]> source.dussan.org Git - nextcloud-server.git/commitdiff
feat(user-prefs): iterator instead of array on search
authorMaxence Lange <maxence@artificial-owl.com>
Mon, 21 Oct 2024 19:31:14 +0000 (18:31 -0100)
committerMaxence Lange <maxence@artificial-owl.com>
Mon, 18 Nov 2024 21:11:31 +0000 (20:11 -0100)
Signed-off-by: Maxence Lange <maxence@artificial-owl.com>
20 files changed:
core/Migrations/Version31000Date20240814184402.php
lib/composer/composer/autoload_classmap.php
lib/composer/composer/autoload_static.php
lib/private/AllConfig.php
lib/private/Config/UserPreferences.php [new file with mode: 0644]
lib/private/Server.php
lib/private/UserPreferences.php [deleted file]
lib/public/Config/Exceptions/IncorrectTypeException.php [new file with mode: 0644]
lib/public/Config/Exceptions/TypeConflictException.php [new file with mode: 0644]
lib/public/Config/Exceptions/UnknownKeyException.php [new file with mode: 0644]
lib/public/Config/IUserPreferences.php [new file with mode: 0644]
lib/public/Config/ValueType.php [new file with mode: 0644]
lib/public/IConfig.php
lib/public/UserPreferences/Exceptions/IncorrectTypeException.php [deleted file]
lib/public/UserPreferences/Exceptions/TypeConflictException.php [deleted file]
lib/public/UserPreferences/Exceptions/UnknownKeyException.php [deleted file]
lib/public/UserPreferences/Exceptions/UserPreferencesException.php [deleted file]
lib/public/UserPreferences/IUserPreferences.php [deleted file]
lib/public/UserPreferences/ValueType.php [deleted file]
tests/lib/UserPreferencesTest.php

index 87b7e93db97d0d8d51cc35079267e8e57edd981f..14b32a704beda68c845493a9e71e2fa02ad246d4 100644 (file)
@@ -38,7 +38,7 @@ class Version31000Date20240814184402 extends SimpleMigrationStep {
                $table->addColumn('lazy', Types::SMALLINT, ['notnull' => true, 'default' => 0, 'length' => 1, 'unsigned' => true]);
                $table->addColumn('type', Types::SMALLINT, ['notnull' => true, 'default' => 0, 'unsigned' => true]);
                $table->addColumn('flags', Types::INTEGER, ['notnull' => true, 'default' => 0, 'unsigned' => true]);
-               $table->addColumn('indexed', Types::STRING, ['notnull' => true, 'default' => '', 'length' => 64]);
+               $table->addColumn('indexed', Types::STRING, ['notnull' => false, 'default' => '', 'length' => 64]);
 
                // removing this index from Version13000Date20170718121200
                // $table->addIndex(['appid', 'configkey'], 'preferences_app_key');
index 3e65291819b88e7cf351a3617e1b271f70737e0b..c3abaefd56c55623e988ffafab9f48326ed2be4b 100644 (file)
@@ -223,6 +223,11 @@ return array(
     'OCP\\Common\\Exception\\NotFoundException' => $baseDir . '/lib/public/Common/Exception/NotFoundException.php',
     'OCP\\Config\\BeforePreferenceDeletedEvent' => $baseDir . '/lib/public/Config/BeforePreferenceDeletedEvent.php',
     'OCP\\Config\\BeforePreferenceSetEvent' => $baseDir . '/lib/public/Config/BeforePreferenceSetEvent.php',
+    'OCP\\Config\\Exceptions\\IncorrectTypeException' => $baseDir . '/lib/public/Config/Exceptions/IncorrectTypeException.php',
+    'OCP\\Config\\Exceptions\\TypeConflictException' => $baseDir . '/lib/public/Config/Exceptions/TypeConflictException.php',
+    'OCP\\Config\\Exceptions\\UnknownKeyException' => $baseDir . '/lib/public/Config/Exceptions/UnknownKeyException.php',
+    'OCP\\Config\\IUserPreferences' => $baseDir . '/lib/public/Config/IUserPreferences.php',
+    'OCP\\Config\\ValueType' => $baseDir . '/lib/public/Config/ValueType.php',
     'OCP\\Console\\ConsoleEvent' => $baseDir . '/lib/public/Console/ConsoleEvent.php',
     'OCP\\Console\\ReservedOptions' => $baseDir . '/lib/public/Console/ReservedOptions.php',
     'OCP\\Constants' => $baseDir . '/lib/public/Constants.php',
@@ -839,13 +844,6 @@ return array(
     'OCP\\UserMigration\\ISizeEstimationMigrator' => $baseDir . '/lib/public/UserMigration/ISizeEstimationMigrator.php',
     'OCP\\UserMigration\\TMigratorBasicVersionHandling' => $baseDir . '/lib/public/UserMigration/TMigratorBasicVersionHandling.php',
     'OCP\\UserMigration\\UserMigrationException' => $baseDir . '/lib/public/UserMigration/UserMigrationException.php',
-    'OCP\\UserPreferences\\Exceptions\\IncorrectTypeException' => $baseDir . '/lib/public/UserPreferences/Exceptions/IncorrectTypeException.php',
-    'OCP\\UserPreferences\\Exceptions\\TypeConflictException' => $baseDir . '/lib/public/UserPreferences/Exceptions/TypeConflictException.php',
-    'OCP\\UserPreferences\\Exceptions\\UnknownKeyException' => $baseDir . '/lib/public/UserPreferences/Exceptions/UnknownKeyException.php',
-    'OCP\\UserPreferences\\Exceptions\\UserPreferencesException' => $baseDir . '/lib/public/UserPreferences/Exceptions/UserPreferencesException.php',
-    'OCP\\UserPreferences\\IUserPreferences' => $baseDir . '/lib/public/UserPreferences/IUserPreferences.php',
-    'OCP\\UserPreferences\\ValueType' => $baseDir . '/lib/public/UserPreferences/ValueType.php',
-    'OCP\\UserPreferences\\ValueTypeDefinition' => $baseDir . '/lib/public/UserPreferences/ValueTypeDefinition.php',
     'OCP\\UserStatus\\IManager' => $baseDir . '/lib/public/UserStatus/IManager.php',
     'OCP\\UserStatus\\IProvider' => $baseDir . '/lib/public/UserStatus/IProvider.php',
     'OCP\\UserStatus\\IUserStatus' => $baseDir . '/lib/public/UserStatus/IUserStatus.php',
@@ -1125,6 +1123,7 @@ return array(
     'OC\\Comments\\Manager' => $baseDir . '/lib/private/Comments/Manager.php',
     'OC\\Comments\\ManagerFactory' => $baseDir . '/lib/private/Comments/ManagerFactory.php',
     'OC\\Config' => $baseDir . '/lib/private/Config.php',
+    'OC\\Config\\UserPreferences' => $baseDir . '/lib/private/Config/UserPreferences.php',
     'OC\\Console\\Application' => $baseDir . '/lib/private/Console/Application.php',
     'OC\\Console\\TimestampFormatter' => $baseDir . '/lib/private/Console/TimestampFormatter.php',
     'OC\\ContactsManager' => $baseDir . '/lib/private/ContactsManager.php',
@@ -2005,7 +2004,6 @@ return array(
     'OC\\Updater\\Exceptions\\ReleaseMetadataException' => $baseDir . '/lib/private/Updater/Exceptions/ReleaseMetadataException.php',
     'OC\\Updater\\ReleaseMetadata' => $baseDir . '/lib/private/Updater/ReleaseMetadata.php',
     'OC\\Updater\\VersionCheck' => $baseDir . '/lib/private/Updater/VersionCheck.php',
-    'OC\\UserPreferences' => $baseDir . '/lib/private/UserPreferences.php',
     'OC\\UserStatus\\ISettableProvider' => $baseDir . '/lib/private/UserStatus/ISettableProvider.php',
     'OC\\UserStatus\\Manager' => $baseDir . '/lib/private/UserStatus/Manager.php',
     'OC\\User\\AvailabilityCoordinator' => $baseDir . '/lib/private/User/AvailabilityCoordinator.php',
index 46e59b2584696271a60410e2b568af5271b822b3..cf821ddeba05823b6c52fcfa602fab0ddabfd045 100644 (file)
@@ -264,6 +264,11 @@ class ComposerStaticInit749170dad3f5e7f9ca158f5a9f04f6a2
         'OCP\\Common\\Exception\\NotFoundException' => __DIR__ . '/../../..' . '/lib/public/Common/Exception/NotFoundException.php',
         'OCP\\Config\\BeforePreferenceDeletedEvent' => __DIR__ . '/../../..' . '/lib/public/Config/BeforePreferenceDeletedEvent.php',
         'OCP\\Config\\BeforePreferenceSetEvent' => __DIR__ . '/../../..' . '/lib/public/Config/BeforePreferenceSetEvent.php',
+        'OCP\\Config\\Exceptions\\IncorrectTypeException' => __DIR__ . '/../../..' . '/lib/public/Config/Exceptions/IncorrectTypeException.php',
+        'OCP\\Config\\Exceptions\\TypeConflictException' => __DIR__ . '/../../..' . '/lib/public/Config/Exceptions/TypeConflictException.php',
+        'OCP\\Config\\Exceptions\\UnknownKeyException' => __DIR__ . '/../../..' . '/lib/public/Config/Exceptions/UnknownKeyException.php',
+        'OCP\\Config\\IUserPreferences' => __DIR__ . '/../../..' . '/lib/public/Config/IUserPreferences.php',
+        'OCP\\Config\\ValueType' => __DIR__ . '/../../..' . '/lib/public/Config/ValueType.php',
         'OCP\\Console\\ConsoleEvent' => __DIR__ . '/../../..' . '/lib/public/Console/ConsoleEvent.php',
         'OCP\\Console\\ReservedOptions' => __DIR__ . '/../../..' . '/lib/public/Console/ReservedOptions.php',
         'OCP\\Constants' => __DIR__ . '/../../..' . '/lib/public/Constants.php',
@@ -880,13 +885,6 @@ class ComposerStaticInit749170dad3f5e7f9ca158f5a9f04f6a2
         'OCP\\UserMigration\\ISizeEstimationMigrator' => __DIR__ . '/../../..' . '/lib/public/UserMigration/ISizeEstimationMigrator.php',
         'OCP\\UserMigration\\TMigratorBasicVersionHandling' => __DIR__ . '/../../..' . '/lib/public/UserMigration/TMigratorBasicVersionHandling.php',
         'OCP\\UserMigration\\UserMigrationException' => __DIR__ . '/../../..' . '/lib/public/UserMigration/UserMigrationException.php',
-        'OCP\\UserPreferences\\Exceptions\\IncorrectTypeException' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/Exceptions/IncorrectTypeException.php',
-        'OCP\\UserPreferences\\Exceptions\\TypeConflictException' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/Exceptions/TypeConflictException.php',
-        'OCP\\UserPreferences\\Exceptions\\UnknownKeyException' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/Exceptions/UnknownKeyException.php',
-        'OCP\\UserPreferences\\Exceptions\\UserPreferencesException' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/Exceptions/UserPreferencesException.php',
-        'OCP\\UserPreferences\\IUserPreferences' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/IUserPreferences.php',
-        'OCP\\UserPreferences\\ValueType' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/ValueType.php',
-        'OCP\\UserPreferences\\ValueTypeDefinition' => __DIR__ . '/../../..' . '/lib/public/UserPreferences/ValueTypeDefinition.php',
         'OCP\\UserStatus\\IManager' => __DIR__ . '/../../..' . '/lib/public/UserStatus/IManager.php',
         'OCP\\UserStatus\\IProvider' => __DIR__ . '/../../..' . '/lib/public/UserStatus/IProvider.php',
         'OCP\\UserStatus\\IUserStatus' => __DIR__ . '/../../..' . '/lib/public/UserStatus/IUserStatus.php',
@@ -1166,6 +1164,7 @@ class ComposerStaticInit749170dad3f5e7f9ca158f5a9f04f6a2
         'OC\\Comments\\Manager' => __DIR__ . '/../../..' . '/lib/private/Comments/Manager.php',
         'OC\\Comments\\ManagerFactory' => __DIR__ . '/../../..' . '/lib/private/Comments/ManagerFactory.php',
         'OC\\Config' => __DIR__ . '/../../..' . '/lib/private/Config.php',
+        'OC\\Config\\UserPreferences' => __DIR__ . '/../../..' . '/lib/private/Config/UserPreferences.php',
         'OC\\Console\\Application' => __DIR__ . '/../../..' . '/lib/private/Console/Application.php',
         'OC\\Console\\TimestampFormatter' => __DIR__ . '/../../..' . '/lib/private/Console/TimestampFormatter.php',
         'OC\\ContactsManager' => __DIR__ . '/../../..' . '/lib/private/ContactsManager.php',
@@ -2046,7 +2045,6 @@ class ComposerStaticInit749170dad3f5e7f9ca158f5a9f04f6a2
         'OC\\Updater\\Exceptions\\ReleaseMetadataException' => __DIR__ . '/../../..' . '/lib/private/Updater/Exceptions/ReleaseMetadataException.php',
         'OC\\Updater\\ReleaseMetadata' => __DIR__ . '/../../..' . '/lib/private/Updater/ReleaseMetadata.php',
         'OC\\Updater\\VersionCheck' => __DIR__ . '/../../..' . '/lib/private/Updater/VersionCheck.php',
-        'OC\\UserPreferences' => __DIR__ . '/../../..' . '/lib/private/UserPreferences.php',
         'OC\\UserStatus\\ISettableProvider' => __DIR__ . '/../../..' . '/lib/private/UserStatus/ISettableProvider.php',
         'OC\\UserStatus\\Manager' => __DIR__ . '/../../..' . '/lib/private/UserStatus/Manager.php',
         'OC\\User\\AvailabilityCoordinator' => __DIR__ . '/../../..' . '/lib/private/User/AvailabilityCoordinator.php',
index b2ebe322d9eef3f51ccc7100dfe1ed3a317902e4..8f7e5359c967fb80830a72a6e724b8306c9b7513 100644 (file)
@@ -6,13 +6,14 @@
  */
 namespace OC;
 
+use OC\Config\UserPreferences;
 use OCP\Cache\CappedMemoryCache;
+use OCP\Config\Exceptions\TypeConflictException;
+use OCP\Config\IUserPreferences;
+use OCP\Config\ValueType;
 use OCP\IConfig;
 use OCP\IDBConnection;
 use OCP\PreConditionNotMetException;
-use OCP\UserPreferences\Exceptions\TypeConflictException;
-use OCP\UserPreferences\IUserPreferences;
-use OCP\UserPreferences\ValueType;
 
 /**
  * Class to combine all the configuration options ownCloud offers
@@ -376,11 +377,11 @@ class AllConfig implements IConfig {
         * @param string $appName the app to get the user for
         * @param string $key the key to get the user for
         * @param string $value the value to get the user for
-        * @return list<string> of user IDs
+        * @return array<string> of user IDs
         * @deprecated 31.0.0 - use {@see IUserPreferences::searchUsersByValueString} directly
         */
        public function getUsersForUserValue($appName, $key, $value) {
-               return \OCP\Server::get(IUserPreferences::class)->searchUsersByValueDeprecated($appName, $key, $value);
+               return iterator_to_array(\OCP\Server::get(IUserPreferences::class)->searchUsersByValueString($appName, $key, $value));
        }
 
        /**
@@ -389,7 +390,7 @@ class AllConfig implements IConfig {
         * @param string $appName the app to get the user for
         * @param string $key the key to get the user for
         * @param string $value the value to get the user for
-        * @return list<string> of user IDs
+        * @return array<string> of user IDs
         * @deprecated 31.0.0 - use {@see IUserPreferences::searchUsersByValueString} directly
         */
        public function getUsersForUserValueCaseInsensitive($appName, $key, $value) {
@@ -397,7 +398,7 @@ class AllConfig implements IConfig {
                        return $this->getUsersForUserValue($appName, $key, strtolower($value));
                }
 
-               return \OCP\Server::get(IUserPreferences::class)->searchUsersByValueDeprecated($appName, $key, $value, true);
+               return iterator_to_array(\OCP\Server::get(IUserPreferences::class)->searchUsersByValueString($appName, $key, $value, true));
        }
 
        public function getSystemConfig() {
diff --git a/lib/private/Config/UserPreferences.php b/lib/private/Config/UserPreferences.php
new file mode 100644 (file)
index 0000000..6dd9c63
--- /dev/null
@@ -0,0 +1,1806 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OC\Config;
+
+use Generator;
+use InvalidArgumentException;
+use JsonException;
+use OCP\Config\Exceptions\IncorrectTypeException;
+use OCP\Config\Exceptions\TypeConflictException;
+use OCP\Config\Exceptions\UnknownKeyException;
+use OCP\Config\IUserPreferences;
+use OCP\Config\ValueType;
+use OCP\DB\Exception as DBException;
+use OCP\DB\IResult;
+use OCP\DB\QueryBuilder\IQueryBuilder;
+use OCP\IConfig;
+use OCP\IDBConnection;
+use OCP\Security\ICrypto;
+use Psr\Log\LoggerInterface;
+
+/**
+ * This class provides an easy way for apps to store user preferences in the
+ * database.
+ * Supports **lazy loading**
+ *
+ * ### What is lazy loading ?
+ * In order to avoid loading useless user preferences into memory for each request,
+ * only non-lazy values are now loaded.
+ *
+ * Once a value that is lazy is requested, all lazy values will be loaded.
+ *
+ * Similarly, some methods from this class are marked with a warning about ignoring
+ * lazy loading. Use them wisely and only on parts of the code that are called
+ * during specific requests or actions to avoid loading the lazy values all the time.
+ *
+ * @since 31.0.0
+ */
+class UserPreferences implements IUserPreferences {
+       private const USER_MAX_LENGTH = 64;
+       private const APP_MAX_LENGTH = 32;
+       private const KEY_MAX_LENGTH = 64;
+       private const INDEX_MAX_LENGTH = 64;
+       private const ENCRYPTION_PREFIX = '$UserPreferencesEncryption$';
+       private const ENCRYPTION_PREFIX_LENGTH = 27; // strlen(self::ENCRYPTION_PREFIX)
+
+       /** @var array<string, array<string, array<string, mixed>>> [ass'user_id' => ['app_id' => ['key' => 'value']]] */
+       private array $fastCache = [];   // cache for normal preference keys
+       /** @var array<string, array<string, array<string, mixed>>> ['user_id' => ['app_id' => ['key' => 'value']]] */
+       private array $lazyCache = [];   // cache for lazy preference keys
+       /** @var array<string, array<string, array<string, array<string, mixed>>>> ['user_id' => ['app_id' => ['key' => ['type' => ValueType, 'flags' => bitflag]]]] */
+       private array $valueDetails = [];  // type for all preference values
+       /** @var array<string, array<string, array<string, ValueType>>> ['user_id' => ['app_id' => ['key' => bitflag]]] */
+       private array $valueTypes = [];  // type for all preference values
+       /** @var array<string, array<string, array<string, int>>> ['user_id' => ['app_id' => ['key' => bitflag]]] */
+       private array $valueFlags = [];  // type for all preference values
+       /** @var array<string, boolean> ['user_id' => bool] */
+       private array $fastLoaded = [];
+       /** @var array<string, boolean> ['user_id' => bool] */
+       private array $lazyLoaded = [];
+
+       public function __construct(
+               protected IDBConnection $connection,
+               protected LoggerInterface $logger,
+               protected ICrypto $crypto,
+       ) {
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $appId optional id of app
+        *
+        * @return list<string> list of userIds
+        * @since 31.0.0
+        */
+       public function getUserIds(string $appId = ''): array {
+               $this->assertParams(app: $appId, allowEmptyUser: true, allowEmptyApp: true);
+
+               $qb = $this->connection->getQueryBuilder();
+               $qb->from('preferences');
+               $qb->select('userid');
+               $qb->groupBy('userid');
+               if ($appId !== '') {
+                       $qb->where($qb->expr()->eq('appid', $qb->createNamedParameter($appId)));
+               }
+
+               $result = $qb->executeQuery();
+               $rows = $result->fetchAll();
+               $userIds = [];
+               foreach ($rows as $row) {
+                       $userIds[] = $row['userid'];
+               }
+
+               return $userIds;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @return list<string> list of app ids
+        * @since 31.0.0
+        */
+       public function getApps(string $userId): array {
+               $this->assertParams($userId, allowEmptyApp: true);
+               $this->loadPreferencesAll($userId);
+               $apps = array_merge(array_keys($this->fastCache[$userId] ?? []), array_keys($this->lazyCache[$userId] ?? []));
+               sort($apps);
+
+               return array_values(array_unique($apps));
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        *
+        * @return list<string> list of stored preference keys
+        * @since 31.0.0
+        */
+       public function getKeys(string $userId, string $app): array {
+               $this->assertParams($userId, $app);
+               $this->loadPreferencesAll($userId);
+               // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
+               $keys = array_map('strval', array_keys(($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? [])));
+               sort($keys);
+
+               return array_values(array_unique($keys));
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
+        *
+        * @return bool TRUE if key exists
+        * @since 31.0.0
+        */
+       public function hasKey(string $userId, string $app, string $key, ?bool $lazy = false): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               if ($lazy === null) {
+                       $appCache = $this->getValues($userId, $app);
+                       return isset($appCache[$key]);
+               }
+
+               if ($lazy) {
+                       return isset($this->lazyCache[$userId][$app][$key]);
+               }
+
+               return isset($this->fastCache[$userId][$app][$key]);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
+        *
+        * @return bool
+        * @throws UnknownKeyException if preference key is not known
+        * @since 31.0.0
+        */
+       public function isSensitive(string $userId, string $app, string $key, ?bool $lazy = false): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               if (!isset($this->valueDetails[$userId][$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               return $this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags']);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
+        *
+        * @return bool
+        * @throws UnknownKeyException if preference key is not known
+        * @since 31.0.0
+        */
+       public function isIndexed(string $userId, string $app, string $key, ?bool $lazy = false): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               if (!isset($this->valueDetails[$userId][$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               return $this->isFlagged(self::FLAG_INDEXED, $this->valueDetails[$userId][$app][$key]['flags']);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app if of the app
+        * @param string $key preference key
+        *
+        * @return bool TRUE if preference is lazy loaded
+        * @throws UnknownKeyException if preference key is not known
+        * @see IUserPreferences for details about lazy loading
+        * @since 31.0.0
+        */
+       public function isLazy(string $userId, string $app, string $key): bool {
+               // there is a huge probability the non-lazy preferences are already loaded
+               // meaning that we can start by only checking if a current non-lazy key exists
+               if ($this->hasKey($userId, $app, $key, false)) {
+                       return false; // meaning key is not lazy.
+               }
+
+               // as key is not found as non-lazy, we load and search in the lazy preferences
+               if ($this->hasKey($userId, $app, $key, true)) {
+                       return true;
+               }
+
+               throw new UnknownKeyException('unknown preference key');
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $prefix preference keys prefix to search
+        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
+        *
+        * @return array<string, string|int|float|bool|array> [key => value]
+        * @since 31.0.0
+        */
+       public function getValues(
+               string $userId,
+               string $app,
+               string $prefix = '',
+               bool $filtered = false,
+       ): array {
+               $this->assertParams($userId, $app, $prefix);
+               // if we want to filter values, we need to get sensitivity
+               $this->loadPreferencesAll($userId);
+               // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
+               $values = array_filter(
+                       $this->formatAppValues($userId, $app, ($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? []), $filtered),
+                       function (string $key) use ($prefix): bool {
+                               return str_starts_with($key, $prefix); // filter values based on $prefix
+                       }, ARRAY_FILTER_USE_KEY
+               );
+
+               return $values;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
+        *
+        * @return array<string, array<string, string|int|float|bool|array>> [appId => [key => value]]
+        * @since 31.0.0
+        */
+       public function getAllValues(string $userId, bool $filtered = false): array {
+               $this->assertParams($userId, allowEmptyApp: true);
+               $this->loadPreferencesAll($userId);
+
+               $result = [];
+               foreach ($this->getApps($userId) as $app) {
+                       // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
+                       $cached = ($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? []);
+                       $result[$app] = $this->formatAppValues($userId, $app, $cached, $filtered);
+               }
+
+               return $result;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $key preference key
+        * @param bool $lazy search within lazy loaded preferences
+        * @param ValueType|null $typedAs enforce type for the returned values
+        *
+        * @return array<string, string|int|float|bool|array> [appId => value]
+        * @since 31.0.0
+        */
+       public function getValuesByApps(string $userId, string $key, bool $lazy = false, ?ValueType $typedAs = null): array {
+               $this->assertParams($userId, '', $key, allowEmptyApp: true);
+               $this->loadPreferences($userId, $lazy);
+
+               /** @var array<array-key, array<array-key, mixed>> $cache */
+               if ($lazy) {
+                       $cache = $this->lazyCache[$userId];
+               } else {
+                       $cache = $this->fastCache[$userId];
+               }
+
+               $values = [];
+               foreach (array_keys($cache) as $app) {
+                       if (isset($cache[$app][$key])) {
+                               $value = $cache[$app][$key];
+                               try {
+                                       $this->decryptSensitiveValue($userId, $app, $key, $value);
+                                       $value = $this->convertTypedValue($value, $typedAs ?? $this->getValueType($userId, $app, $key, $lazy));
+                               } catch (IncorrectTypeException|UnknownKeyException) {
+                               }
+                               $values[$app] = $value;
+                       }
+               }
+
+               return $values;
+       }
+
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param ValueType|null $typedAs enforce type for the returned values
+        * @param array|null $userIds limit to a list of user ids
+        *
+        * @return array<string, string|int|float|bool|array> [userId => value]
+        * @since 31.0.0
+        */
+       public function getValuesByUsers(
+               string $app,
+               string $key,
+               ?ValueType $typedAs = null,
+               ?array $userIds = null,
+       ): array {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+
+               $qb = $this->connection->getQueryBuilder();
+               $qb->select('userid', 'configvalue', 'type')
+                       ->from('preferences')
+                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
+                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
+
+               $values = [];
+               // this nested function will execute current Query and store result within $values.
+               $executeAndStoreValue = function (IQueryBuilder $qb) use (&$values, $typedAs): IResult {
+                       $result = $qb->executeQuery();
+                       while ($row = $result->fetch()) {
+                               $value = $row['configvalue'];
+                               try {
+                                       $value = $this->convertTypedValue($value, $typedAs ?? ValueType::from((int)$row['type']));
+                               } catch (IncorrectTypeException) {
+                               }
+                               $values[$row['userid']] = $value;
+                       }
+                       return $result;
+               };
+
+               // if no userIds to filter, we execute query as it is and returns all values ...
+               if ($userIds === null) {
+                       $result = $executeAndStoreValue($qb);
+                       $result->closeCursor();
+                       return $values;
+               }
+
+               // if userIds to filter, we chunk the list and execute the same query multiple times until we get all values
+               $result = null;
+               $qb->andWhere($qb->expr()->in('userid', $qb->createParameter('userIds')));
+               foreach (array_chunk($userIds, 50, true) as $chunk) {
+                       $qb->setParameter('userIds', $chunk, IQueryBuilder::PARAM_STR_ARRAY);
+                       $result = $executeAndStoreValue($qb);
+               }
+               $result?->closeCursor();
+
+               return $values;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $caseInsensitive non-case-sensitive search, only works if $value is a string
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueString(string $app, string $key, string $value, bool $caseInsensitive = false): Generator {
+               return $this->searchUsersByTypedValue($app, $key, $value, $caseInsensitive);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $value preference value
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueInt(string $app, string $key, int $value): Generator {
+               return $this->searchUsersByValueString($app, $key, (string)$value);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $values list of preference values
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValues(string $app, string $key, array $values): Generator {
+               return $this->searchUsersByTypedValue($app, $key, $values);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $value preference value
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueBool(string $app, string $key, bool $value): Generator {
+               $values = ['0', 'off', 'false', 'no'];
+               if ($value) {
+                       $values = ['1', 'on', 'true', 'yes'];
+               }
+               return $this->searchUsersByValues($app, $key, $values);
+       }
+
+       /**
+        * returns a list of users with preference key set to a specific value, or within the list of
+        * possible values
+        *
+        * @param string $app
+        * @param string $key
+        * @param string|array $value
+        * @param bool $caseInsensitive
+        *
+        * @return Generator<string>
+        */
+       private function searchUsersByTypedValue(string $app, string $key, string|array $value, bool $caseInsensitive = false): Generator {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+
+               $qb = $this->connection->getQueryBuilder();
+               $qb->from('preferences');
+               $qb->select('userid');
+               $qb->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)));
+               $qb->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
+
+               // search within 'indexed' OR 'configvalue' only if 'flags' is set as not indexed
+               // TODO: when implementing config lexicon remove the searches on 'configvalue' if value is set as indexed
+               $configValueColumn = ($this->connection->getDatabaseProvider() === IDBConnection::PLATFORM_ORACLE) ? $qb->expr()->castColumn('configvalue', IQueryBuilder::PARAM_STR) : 'configvalue';
+               if (is_array($value)) {
+                       $where = $qb->expr()->orX(
+                               $qb->expr()->in('indexed', $qb->createNamedParameter($value, IQueryBuilder::PARAM_STR_ARRAY)),
+                               $qb->expr()->andX(
+                                       $qb->expr()->neq($qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED), $qb->createNamedParameter(self::FLAG_INDEXED, IQueryBuilder::PARAM_INT)),
+                                       $qb->expr()->in($configValueColumn, $qb->createNamedParameter($value, IQueryBuilder::PARAM_STR_ARRAY))
+                               )
+                       );
+               } else {
+                       if ($caseInsensitive) {
+                               $where = $qb->expr()->orX(
+                                       $qb->expr()->eq($qb->func()->lower('indexed'), $qb->createNamedParameter(strtolower($value))),
+                                       $qb->expr()->andX(
+                                               $qb->expr()->neq($qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED), $qb->createNamedParameter(self::FLAG_INDEXED, IQueryBuilder::PARAM_INT)),
+                                               $qb->expr()->eq($qb->func()->lower($configValueColumn), $qb->createNamedParameter(strtolower($value)))
+                                       )
+                               );
+                       } else {
+                               $where = $qb->expr()->orX(
+                                       $qb->expr()->eq('indexed', $qb->createNamedParameter($value)),
+                                       $qb->expr()->andX(
+                                               $qb->expr()->neq($qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED), $qb->createNamedParameter(self::FLAG_INDEXED, IQueryBuilder::PARAM_INT)),
+                                               $qb->expr()->eq($configValueColumn, $qb->createNamedParameter($value))
+                                       )
+                               );
+                       }
+               }
+
+               $qb->andWhere($where);
+               $result = $qb->executeQuery();
+               while ($row = $result->fetch()) {
+                       yield $row['userid'];
+               }
+       }
+
+       /**
+        * Get the preference value as string.
+        * If the value does not exist the given default will be returned.
+        *
+        * Set lazy to `null` to ignore it and get the value from either source.
+        *
+        * **WARNING:** Method is internal and **SHOULD** not be used, as it is better to get the value with a type.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $default preference value
+        * @param null|bool $lazy get preference as lazy loaded or not. can be NULL
+        *
+        * @return string the value or $default
+        * @throws TypeConflictException
+        * @internal
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see getValueString()
+        * @see getValueInt()
+        * @see getValueFloat()
+        * @see getValueBool()
+        * @see getValueArray()
+        */
+       public function getValueMixed(
+               string $userId,
+               string $app,
+               string $key,
+               string $default = '',
+               ?bool $lazy = false,
+       ): string {
+               try {
+                       $lazy ??= $this->isLazy($userId, $app, $key);
+               } catch (UnknownKeyException) {
+                       return $default;
+               }
+
+               return $this->getTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       $default,
+                       $lazy,
+                       ValueType::MIXED
+               );
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return string stored preference value or $default if not set in database
+        * @throws InvalidArgumentException if one of the argument format is invalid
+        * @throws TypeConflictException in case of conflict with the value type set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function getValueString(
+               string $userId,
+               string $app,
+               string $key,
+               string $default = '',
+               bool $lazy = false,
+       ): string {
+               return $this->getTypedValue($userId, $app, $key, $default, $lazy, ValueType::STRING);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return int stored preference value or $default if not set in database
+        * @throws InvalidArgumentException if one of the argument format is invalid
+        * @throws TypeConflictException in case of conflict with the value type set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function getValueInt(
+               string $userId,
+               string $app,
+               string $key,
+               int $default = 0,
+               bool $lazy = false,
+       ): int {
+               return (int)$this->getTypedValue($userId, $app, $key, (string)$default, $lazy, ValueType::INT);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param float $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return float stored preference value or $default if not set in database
+        * @throws InvalidArgumentException if one of the argument format is invalid
+        * @throws TypeConflictException in case of conflict with the value type set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function getValueFloat(
+               string $userId,
+               string $app,
+               string $key,
+               float $default = 0,
+               bool $lazy = false,
+       ): float {
+               return (float)$this->getTypedValue($userId, $app, $key, (string)$default, $lazy, ValueType::FLOAT);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return bool stored preference value or $default if not set in database
+        * @throws InvalidArgumentException if one of the argument format is invalid
+        * @throws TypeConflictException in case of conflict with the value type set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function getValueBool(
+               string $userId,
+               string $app,
+               string $key,
+               bool $default = false,
+               bool $lazy = false,
+       ): bool {
+               $b = strtolower($this->getTypedValue($userId, $app, $key, $default ? 'true' : 'false', $lazy, ValueType::BOOL));
+               return in_array($b, ['1', 'true', 'yes', 'on']);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return array stored preference value or $default if not set in database
+        * @throws InvalidArgumentException if one of the argument format is invalid
+        * @throws TypeConflictException in case of conflict with the value type set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function getValueArray(
+               string $userId,
+               string $app,
+               string $key,
+               array $default = [],
+               bool $lazy = false,
+       ): array {
+               try {
+                       $defaultJson = json_encode($default, JSON_THROW_ON_ERROR);
+                       $value = json_decode($this->getTypedValue($userId, $app, $key, $defaultJson, $lazy, ValueType::ARRAY), true, flags: JSON_THROW_ON_ERROR);
+
+                       return is_array($value) ? $value : [];
+               } catch (JsonException) {
+                       return [];
+               }
+       }
+
+       /**
+        * @param string $userId
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        * @param ValueType $type value type
+        *
+        * @return string
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        */
+       private function getTypedValue(
+               string $userId,
+               string $app,
+               string $key,
+               string $default,
+               bool $lazy,
+               ValueType $type,
+       ): string {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               /**
+                * We ignore check if mixed type is requested.
+                * If type of stored value is set as mixed, we don't filter.
+                * If type of stored value is defined, we compare with the one requested.
+                */
+               $knownType = $this->valueDetails[$userId][$app][$key]['type'] ?? null;
+               if ($type !== ValueType::MIXED
+                       && $knownType !== null
+                       && $knownType !== ValueType::MIXED
+                       && $type !== $knownType) {
+                       $this->logger->warning('conflict with value type from database', ['app' => $app, 'key' => $key, 'type' => $type, 'knownType' => $knownType]);
+                       throw new TypeConflictException('conflict with value type from database');
+               }
+
+               /**
+                * - the pair $app/$key cannot exist in both array,
+                * - we should still return an existing non-lazy value even if current method
+                *   is called with $lazy is true
+                *
+                * This way, lazyCache will be empty until the load for lazy preferences value is requested.
+                */
+               if (isset($this->lazyCache[$userId][$app][$key])) {
+                       $value = $this->lazyCache[$userId][$app][$key];
+               } elseif (isset($this->fastCache[$userId][$app][$key])) {
+                       $value = $this->fastCache[$userId][$app][$key];
+               } else {
+                       return $default;
+               }
+
+               $this->decryptSensitiveValue($userId, $app, $key, $value);
+               return $value;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @return ValueType type of the value
+        * @throws UnknownKeyException if preference key is not known
+        * @throws IncorrectTypeException if preferences value type is not known
+        * @since 31.0.0
+        */
+       public function getValueType(string $userId, string $app, string $key, ?bool $lazy = null): ValueType {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               if (!isset($this->valueDetails[$userId][$app][$key]['type'])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               return $this->valueDetails[$userId][$app][$key]['type'];
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy lazy loading
+        *
+        * @return int flags applied to value
+        * @throws UnknownKeyException if preference key is not known
+        * @throws IncorrectTypeException if preferences value type is not known
+        * @since 31.0.0
+        */
+       public function getValueFlags(string $userId, string $app, string $key, bool $lazy = false): int {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               if (!isset($this->valueDetails[$userId][$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               return $this->valueDetails[$userId][$app][$key]['flags'];
+       }
+
+       /**
+        * Store a preference key and its value in database as VALUE_MIXED
+        *
+        * **WARNING:** Method is internal and **MUST** not be used as it is best to set a real value type
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED
+        * @internal
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueString()
+        * @see setValueInt()
+        * @see setValueFloat()
+        * @see setValueBool()
+        * @see setValueArray()
+        */
+       public function setValueMixed(
+               string $userId,
+               string $app,
+               string $key,
+               string $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               return $this->setTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       $value,
+                       $lazy,
+                       $flags,
+                       ValueType::MIXED
+               );
+       }
+
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function setValueString(
+               string $userId,
+               string $app,
+               string $key,
+               string $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               return $this->setTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       $value,
+                       $lazy,
+                       $flags,
+                       ValueType::STRING
+               );
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function setValueInt(
+               string $userId,
+               string $app,
+               string $key,
+               int $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               if ($value > 2000000000) {
+                       $this->logger->debug('You are trying to store an integer value around/above 2,147,483,647. This is a reminder that reaching this theoretical limit on 32 bits system will throw an exception.');
+               }
+
+               return $this->setTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       (string)$value,
+                       $lazy,
+                       $flags,
+                       ValueType::INT
+               );
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param float $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function setValueFloat(
+               string $userId,
+               string $app,
+               string $key,
+               float $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               return $this->setTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       (string)$value,
+                       $lazy,
+                       $flags,
+                       ValueType::FLOAT
+               );
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function setValueBool(
+               string $userId,
+               string $app,
+               string $key,
+               bool $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               return $this->setTypedValue(
+                       $userId,
+                       $app,
+                       $key,
+                       ($value) ? '1' : '0',
+                       $lazy,
+                       $flags,
+                       ValueType::BOOL
+               );
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @throws JsonException
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       public function setValueArray(
+               string $userId,
+               string $app,
+               string $key,
+               array $value,
+               bool $lazy = false,
+               int $flags = 0,
+       ): bool {
+               try {
+                       return $this->setTypedValue(
+                               $userId,
+                               $app,
+                               $key,
+                               json_encode($value, JSON_THROW_ON_ERROR),
+                               $lazy,
+                               $flags,
+                               ValueType::ARRAY
+                       );
+               } catch (JsonException $e) {
+                       $this->logger->warning('could not setValueArray', ['app' => $app, 'key' => $key, 'exception' => $e]);
+                       throw $e;
+               }
+       }
+
+       /**
+        * Store a preference key and its value in database
+        *
+        * If preference key is already known with the exact same preference value and same sensitive/lazy status, the
+        * database is not updated. If preference value was previously stored as sensitive, status will not be
+        * altered.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $lazy preferences set as lazy loaded
+        * @param ValueType $type value type
+        *
+        * @return bool TRUE if value was updated in database
+        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
+        * @see IUserPreferences for explanation about lazy loading
+        */
+       private function setTypedValue(
+               string $userId,
+               string $app,
+               string $key,
+               string $value,
+               bool $lazy,
+               int $flags,
+               ValueType $type,
+       ): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferences($userId, $lazy);
+
+               $inserted = $refreshCache = false;
+               $origValue = $value;
+               $sensitive = $this->isFlagged(self::FLAG_SENSITIVE, $flags);
+               if ($sensitive || ($this->hasKey($userId, $app, $key, $lazy) && $this->isSensitive($userId, $app, $key, $lazy))) {
+                       $value = self::ENCRYPTION_PREFIX . $this->crypto->encrypt($value);
+                       $flags |= UserPreferences::FLAG_SENSITIVE;
+               }
+
+               // if requested, we fill the 'indexed' field with current value
+               $indexed = '';
+               if ($type !== ValueType::ARRAY && $this->isFlagged(self::FLAG_INDEXED, $flags)) {
+                       if ($this->isFlagged(self::FLAG_SENSITIVE, $flags)) {
+                               $this->logger->warning('sensitive value are not to be indexed');
+                       } elseif (strlen($value) > self::USER_MAX_LENGTH) {
+                               $this->logger->warning('value is too lengthy to be indexed');
+                       } else {
+                               $indexed = $value;
+                       }
+               }
+
+               if ($this->hasKey($userId, $app, $key, $lazy)) {
+                       /**
+                        * no update if key is already known with set lazy status and value is
+                        * not different, unless sensitivity is switched from false to true.
+                        */
+                       if ($origValue === $this->getTypedValue($userId, $app, $key, $value, $lazy, $type)
+                               && (!$sensitive || $this->isSensitive($userId, $app, $key, $lazy))) {
+                               return false;
+                       }
+               } else {
+                       /**
+                        * if key is not known yet, we try to insert.
+                        * It might fail if the key exists with a different lazy flag.
+                        */
+                       try {
+                               $insert = $this->connection->getQueryBuilder();
+                               $insert->insert('preferences')
+                                       ->setValue('userid', $insert->createNamedParameter($userId))
+                                       ->setValue('appid', $insert->createNamedParameter($app))
+                                       ->setValue('lazy', $insert->createNamedParameter(($lazy) ? 1 : 0, IQueryBuilder::PARAM_INT))
+                                       ->setValue('type', $insert->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
+                                       ->setValue('flags', $insert->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
+                                       ->setValue('indexed', $insert->createNamedParameter($indexed))
+                                       ->setValue('configkey', $insert->createNamedParameter($key))
+                                       ->setValue('configvalue', $insert->createNamedParameter($value));
+                               $insert->executeStatement();
+                               $inserted = true;
+                       } catch (DBException $e) {
+                               if ($e->getReason() !== DBException::REASON_UNIQUE_CONSTRAINT_VIOLATION) {
+                                       throw $e; // TODO: throw exception or just log and returns false !?
+                               }
+                       }
+               }
+
+               /**
+                * We cannot insert a new row, meaning we need to update an already existing one
+                */
+               if (!$inserted) {
+                       $currType = $this->valueDetails[$userId][$app][$key]['type'] ?? null;
+                       if ($currType === null) { // this might happen when switching lazy loading status
+                               $this->loadPreferencesAll($userId);
+                               $currType = $this->valueDetails[$userId][$app][$key]['type'];
+                       }
+
+                       /**
+                        * We only log a warning and set it to VALUE_MIXED.
+                        */
+                       if ($currType === null) {
+                               $this->logger->warning('Value type is set to zero (0) in database. This is not supposed to happens', ['app' => $app, 'key' => $key]);
+                               $currType = ValueType::MIXED;
+                       }
+
+                       /**
+                        * we only accept a different type from the one stored in database
+                        * if the one stored in database is not-defined (VALUE_MIXED)
+                        */
+                       if ($currType !== ValueType::MIXED &&
+                               $currType !== $type) {
+                               try {
+                                       $currTypeDef = $currType->getDefinition();
+                                       $typeDef = $type->getDefinition();
+                               } catch (IncorrectTypeException) {
+                                       $currTypeDef = $currType->value;
+                                       $typeDef = $type->value;
+                               }
+                               throw new TypeConflictException('conflict between new type (' . $typeDef . ') and old type (' . $currTypeDef . ')');
+                       }
+
+                       if ($lazy !== $this->isLazy($userId, $app, $key)) {
+                               $refreshCache = true;
+                       }
+
+                       $update = $this->connection->getQueryBuilder();
+                       $update->update('preferences')
+                               ->set('configvalue', $update->createNamedParameter($value))
+                               ->set('lazy', $update->createNamedParameter(($lazy) ? 1 : 0, IQueryBuilder::PARAM_INT))
+                               ->set('type', $update->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
+                               ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
+                               ->set('indexed', $update->createNamedParameter($indexed))
+                               ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
+                               ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                               ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+
+                       $update->executeStatement();
+               }
+
+               if ($refreshCache) {
+                       $this->clearCache($userId);
+                       return true;
+               }
+
+               // update local cache
+               if ($lazy) {
+                       $this->lazyCache[$userId][$app][$key] = $value;
+               } else {
+                       $this->fastCache[$userId][$app][$key] = $value;
+               }
+               $this->valueDetails[$userId][$app][$key] = [
+                       'type' => $type,
+                       'flags' => $flags
+               ];
+
+               return true;
+       }
+
+       /**
+        * Change the type of preference value.
+        *
+        * **WARNING:** Method is internal and **MUST** not be used as it may break things.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param ValueType $type value type
+        *
+        * @return bool TRUE if database update were necessary
+        * @throws UnknownKeyException if $key is now known in database
+        * @throws IncorrectTypeException if $type is not valid
+        * @internal
+        * @since 31.0.0
+        */
+       public function updateType(string $userId, string $app, string $key, ValueType $type = ValueType::MIXED): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferencesAll($userId);
+               $this->isLazy($userId, $app, $key); // confirm key exists
+
+               $update = $this->connection->getQueryBuilder();
+               $update->update('preferences')
+                       ->set('type', $update->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
+                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
+                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+               $update->executeStatement();
+
+               $this->valueDetails[$userId][$app][$key]['type'] = $type;
+
+               return true;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
+        *
+        * @return bool TRUE if entry was found in database and an update was necessary
+        * @since 31.0.0
+        */
+       public function updateSensitive(string $userId, string $app, string $key, bool $sensitive): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferencesAll($userId);
+
+               try {
+                       if ($sensitive === $this->isSensitive($userId, $app, $key, null)) {
+                               return false;
+                       }
+               } catch (UnknownKeyException) {
+                       return false;
+               }
+
+               $lazy = $this->isLazy($userId, $app, $key);
+               if ($lazy) {
+                       $cache = $this->lazyCache;
+               } else {
+                       $cache = $this->fastCache;
+               }
+
+               if (!isset($cache[$userId][$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               $value = $cache[$userId][$app][$key];
+               $flags = $this->getValueFlags($userId, $app, $key);
+               if ($sensitive) {
+                       $flags |= self::FLAG_SENSITIVE;
+                       $value = self::ENCRYPTION_PREFIX . $this->crypto->encrypt($value);
+               } else {
+                       $flags &= ~self::FLAG_SENSITIVE;
+                       $this->decryptSensitiveValue($userId, $app, $key, $value);
+               }
+
+               $update = $this->connection->getQueryBuilder();
+               $update->update('preferences')
+                       ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
+                       ->set('configvalue', $update->createNamedParameter($value))
+                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
+                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+               $update->executeStatement();
+
+               $this->valueDetails[$userId][$app][$key]['flags'] = $flags;
+
+               return true;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app
+        * @param string $key
+        * @param bool $sensitive
+        *
+        * @since 31.0.0
+        */
+       public function updateGlobalSensitive(string $app, string $key, bool $sensitive): void {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+               foreach (array_keys($this->getValuesByUsers($app, $key)) as $userId) {
+                       try {
+                               $this->updateSensitive($userId, $app, $key, $sensitive);
+                       } catch (UnknownKeyException) {
+                               // should not happen and can be ignored
+                       }
+               }
+
+               $this->clearCacheAll(); // we clear all cache
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId
+        * @param string $app
+        * @param string $key
+        * @param bool $indexed
+        *
+        * @return bool
+        * @throws DBException
+        * @throws IncorrectTypeException
+        * @throws UnknownKeyException
+        * @since 31.0.0
+        */
+       public function updateIndexed(string $userId, string $app, string $key, bool $indexed): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferencesAll($userId);
+
+               try {
+                       if ($indexed === $this->isIndexed($userId, $app, $key, null)) {
+                               return false;
+                       }
+               } catch (UnknownKeyException) {
+                       return false;
+               }
+
+               $lazy = $this->isLazy($userId, $app, $key);
+               if ($lazy) {
+                       $cache = $this->lazyCache;
+               } else {
+                       $cache = $this->fastCache;
+               }
+
+               if (!isset($cache[$userId][$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               $value = $cache[$userId][$app][$key];
+               $flags = $this->getValueFlags($userId, $app, $key);
+               if ($indexed) {
+                       $indexed = $value;
+               } else {
+                       $flags &= ~self::FLAG_INDEXED;
+                       $indexed = '';
+               }
+
+               $update = $this->connection->getQueryBuilder();
+               $update->update('preferences')
+                       ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
+                       ->set('indexed', $update->createNamedParameter($indexed))
+                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
+                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+               $update->executeStatement();
+
+               $this->valueDetails[$userId][$app][$key]['flags'] = $flags;
+
+               return true;
+       }
+
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app
+        * @param string $key
+        * @param bool $indexed
+        *
+        * @since 31.0.0
+        */
+       public function updateGlobalIndexed(string $app, string $key, bool $indexed): void {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+               foreach (array_keys($this->getValuesByUsers($app, $key)) as $userId) {
+                       try {
+                               $this->updateIndexed($userId, $app, $key, $indexed);
+                       } catch (UnknownKeyException) {
+                               // should not happen and can be ignored
+                       }
+               }
+
+               $this->clearCacheAll(); // we clear all cache
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
+        *
+        * @return bool TRUE if entry was found in database and an update was necessary
+        * @since 31.0.0
+        */
+       public function updateLazy(string $userId, string $app, string $key, bool $lazy): bool {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferencesAll($userId);
+
+               try {
+                       if ($lazy === $this->isLazy($userId, $app, $key)) {
+                               return false;
+                       }
+               } catch (UnknownKeyException) {
+                       return false;
+               }
+
+               $update = $this->connection->getQueryBuilder();
+               $update->update('preferences')
+                       ->set('lazy', $update->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT))
+                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
+                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+               $update->executeStatement();
+
+               // At this point, it is a lot safer to clean cache
+               $this->clearCache($userId);
+
+               return true;
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
+        *
+        * @since 31.0.0
+        */
+       public function updateGlobalLazy(string $app, string $key, bool $lazy): void {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+
+               $update = $this->connection->getQueryBuilder();
+               $update->update('preferences')
+                       ->set('lazy', $update->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT))
+                       ->where($update->expr()->eq('appid', $update->createNamedParameter($app)))
+                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
+               $update->executeStatement();
+
+               $this->clearCacheAll();
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @return array
+        * @throws UnknownKeyException if preference key is not known in database
+        * @since 31.0.0
+        */
+       public function getDetails(string $userId, string $app, string $key): array {
+               $this->assertParams($userId, $app, $key);
+               $this->loadPreferencesAll($userId);
+               $lazy = $this->isLazy($userId, $app, $key);
+
+               if ($lazy) {
+                       $cache = $this->lazyCache[$userId];
+               } else {
+                       $cache = $this->fastCache[$userId];
+               }
+
+               $type = $this->getValueType($userId, $app, $key);
+               try {
+                       $typeString = $type->getDefinition();
+               } catch (IncorrectTypeException $e) {
+                       $this->logger->warning('type stored in database is not correct', ['exception' => $e, 'type' => $type]);
+                       $typeString = (string)$type->value;
+               }
+
+               if (!isset($cache[$app][$key])) {
+                       throw new UnknownKeyException('unknown preference key');
+               }
+
+               $value = $cache[$app][$key];
+               $sensitive = $this->isSensitive($userId, $app, $key, null);
+               $this->decryptSensitiveValue($userId, $app, $key, $value);
+
+               return [
+                       'userId' => $userId,
+                       'app' => $app,
+                       'key' => $key,
+                       'value' => $value,
+                       'type' => $type->value,
+                       'lazy' => $lazy,
+                       'typeString' => $typeString,
+                       'sensitive' => $sensitive
+               ];
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @since 31.0.0
+        */
+       public function deletePreference(string $userId, string $app, string $key): void {
+               $this->assertParams($userId, $app, $key);
+               $qb = $this->connection->getQueryBuilder();
+               $qb->delete('preferences')
+                       ->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)))
+                       ->andWhere($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
+                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
+               $qb->executeStatement();
+
+               unset($this->lazyCache[$userId][$app][$key]);
+               unset($this->fastCache[$userId][$app][$key]);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @since 31.0.0
+        */
+       public function deleteKey(string $app, string $key): void {
+               $this->assertParams('', $app, $key, allowEmptyUser: true);
+               $qb = $this->connection->getQueryBuilder();
+               $qb->delete('preferences')
+                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
+                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
+               $qb->executeStatement();
+
+               $this->clearCacheAll();
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $app id of the app
+        *
+        * @since 31.0.0
+        */
+       public function deleteApp(string $app): void {
+               $this->assertParams('', $app, allowEmptyUser: true);
+               $qb = $this->connection->getQueryBuilder();
+               $qb->delete('preferences')
+                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)));
+               $qb->executeStatement();
+
+               $this->clearCacheAll();
+       }
+
+       public function deleteAllPreferences(string $userId): void {
+               $this->assertParams($userId, '', allowEmptyApp: true);
+               $qb = $this->connection->getQueryBuilder();
+               $qb->delete('preferences')
+                       ->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)));
+               $qb->executeStatement();
+
+               $this->clearCache($userId);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @param string $userId id of the user
+        * @param bool $reload set to TRUE to refill cache instantly after clearing it.
+        *
+        * @since 31.0.0
+        */
+       public function clearCache(string $userId, bool $reload = false): void {
+               $this->assertParams($userId, allowEmptyApp: true);
+               $this->lazyLoaded[$userId] = $this->fastLoaded[$userId] = false;
+               $this->lazyCache[$userId] = $this->fastCache[$userId] = $this->valueDetails[$userId] = [];
+
+               if (!$reload) {
+                       return;
+               }
+
+               $this->loadPreferencesAll($userId);
+       }
+
+       /**
+        * @inheritDoc
+        *
+        * @since 31.0.0
+        */
+       public function clearCacheAll(): void {
+               $this->lazyLoaded = $this->fastLoaded = [];
+               $this->lazyCache = $this->fastCache = $this->valueDetails = [];
+       }
+
+       /**
+        * For debug purpose.
+        * Returns the cached data.
+        *
+        * @return array
+        * @since 31.0.0
+        * @internal
+        */
+       public function statusCache(): array {
+               return [
+                       'fastLoaded' => $this->fastLoaded,
+                       'fastCache' => $this->fastCache,
+                       'lazyLoaded' => $this->lazyLoaded,
+                       'lazyCache' => $this->lazyCache,
+                       'valueDetails' => $this->valueDetails,
+               ];
+       }
+
+       /**
+        * @param int $needle bitflag to search
+        * @param int $flags all flags
+        *
+        * @return bool TRUE if bitflag $needle is set in $flags
+        */
+       private function isFlagged(int $needle, int $flags): bool {
+               return (($needle & $flags) !== 0);
+       }
+
+       /**
+        * Confirm the string set for app and key fit the database description
+        *
+        * @param string $userId
+        * @param string $app assert $app fit in database
+        * @param string $prefKey assert preference key fit in database
+        * @param bool $allowEmptyUser
+        * @param bool $allowEmptyApp $app can be empty string
+        * @param ValueType|null $valueType assert value type is only one type
+        */
+       private function assertParams(
+               string $userId = '',
+               string $app = '',
+               string $prefKey = '',
+               bool $allowEmptyUser = false,
+               bool $allowEmptyApp = false,
+       ): void {
+               if (!$allowEmptyUser && $userId === '') {
+                       throw new InvalidArgumentException('userId cannot be an empty string');
+               }
+               if (!$allowEmptyApp && $app === '') {
+                       throw new InvalidArgumentException('app cannot be an empty string');
+               }
+               if (strlen($userId) > self::USER_MAX_LENGTH) {
+                       throw new InvalidArgumentException('Value (' . $userId . ') for userId is too long (' . self::USER_MAX_LENGTH . ')');
+               }
+               if (strlen($app) > self::APP_MAX_LENGTH) {
+                       throw new InvalidArgumentException('Value (' . $app . ') for app is too long (' . self::APP_MAX_LENGTH . ')');
+               }
+               if (strlen($prefKey) > self::KEY_MAX_LENGTH) {
+                       throw new InvalidArgumentException('Value (' . $prefKey . ') for key is too long (' . self::KEY_MAX_LENGTH . ')');
+               }
+       }
+
+       private function loadPreferencesAll(string $userId): void {
+               $this->loadPreferences($userId, null);
+       }
+
+       /**
+        * Load normal preferences or preferences set as lazy loaded
+        *
+        * @param bool|null $lazy set to TRUE to load preferences set as lazy loaded, set to NULL to load all preferences
+        */
+       private function loadPreferences(string $userId, ?bool $lazy = false): void {
+               if ($this->isLoaded($userId, $lazy)) {
+                       return;
+               }
+
+               if (($lazy ?? true) !== false) { // if lazy is null or true, we debug log
+                       $this->logger->debug('The loading of lazy UserPreferences values have been requested', ['exception' => new \RuntimeException('ignorable exception')]);
+               }
+
+               $qb = $this->connection->getQueryBuilder();
+               $qb->from('preferences');
+               $qb->select('appid', 'configkey', 'configvalue', 'type', 'flags');
+               $qb->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)));
+
+               // we only need value from lazy when loadPreferences does not specify it
+               if ($lazy !== null) {
+                       $qb->andWhere($qb->expr()->eq('lazy', $qb->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT)));
+               } else {
+                       $qb->addSelect('lazy');
+               }
+
+               $result = $qb->executeQuery();
+               $rows = $result->fetchAll();
+               foreach ($rows as $row) {
+                       if (($row['lazy'] ?? ($lazy ?? 0) ? 1 : 0) === 1) {
+                               $this->lazyCache[$userId][$row['appid']][$row['configkey']] = $row['configvalue'] ?? '';
+                       } else {
+                               $this->fastCache[$userId][$row['appid']][$row['configkey']] = $row['configvalue'] ?? '';
+                       }
+                       $this->valueDetails[$userId][$row['appid']][$row['configkey']] = ['type' => ValueType::from((int)($row['type'] ?? 0)), 'flags' => (int)$row['flags']];
+               }
+               $result->closeCursor();
+               $this->setAsLoaded($userId, $lazy);
+       }
+
+       /**
+        * if $lazy is:
+        *  - false: will returns true if fast preferences are loaded
+        *  - true : will returns true if lazy preferences are loaded
+        *  - null : will returns true if both preferences are loaded
+        *
+        * @param string $userId
+        * @param bool $lazy
+        *
+        * @return bool
+        */
+       private function isLoaded(string $userId, ?bool $lazy): bool {
+               if ($lazy === null) {
+                       return ($this->lazyLoaded[$userId] ?? false) && ($this->fastLoaded[$userId] ?? false);
+               }
+
+               return $lazy ? $this->lazyLoaded[$userId] ?? false : $this->fastLoaded[$userId] ?? false;
+       }
+
+       /**
+        * if $lazy is:
+        * - false: set fast preferences as loaded
+        * - true : set lazy preferences as loaded
+        * - null : set both preferences as loaded
+        *
+        * @param string $userId
+        * @param bool $lazy
+        */
+       private function setAsLoaded(string $userId, ?bool $lazy): void {
+               if ($lazy === null) {
+                       $this->fastLoaded[$userId] = $this->lazyLoaded[$userId] = true;
+                       return;
+               }
+
+               // We also create empty entry to keep both fastLoaded/lazyLoaded synced
+               if ($lazy) {
+                       $this->lazyLoaded[$userId] = true;
+                       $this->fastLoaded[$userId] = $this->fastLoaded[$userId] ?? false;
+                       $this->fastCache[$userId] = $this->fastCache[$userId] ?? [];
+               } else {
+                       $this->fastLoaded[$userId] = true;
+                       $this->lazyLoaded[$userId] = $this->lazyLoaded[$userId] ?? false;
+                       $this->lazyCache[$userId] = $this->lazyCache[$userId] ?? [];
+               }
+       }
+
+       /**
+        * **Warning:** this will load all lazy values from the database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
+        *
+        * @return array<string, string|int|float|bool|array>
+        */
+       private function formatAppValues(string $userId, string $app, array $values, bool $filtered = false): array {
+               foreach ($values as $key => $value) {
+                       //$key = (string)$key;
+                       try {
+                               $type = $this->getValueType($userId, $app, (string)$key);
+                       } catch (UnknownKeyException) {
+                               continue;
+                       }
+
+                       if ($this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags'] ?? 0)) {
+                               if ($filtered) {
+                                       $value = IConfig::SENSITIVE_VALUE;
+                                       $type = ValueType::STRING;
+                               } else {
+                                       $this->decryptSensitiveValue($userId, $app, (string)$key, $value);
+                               }
+                       }
+
+                       $values[$key] = $this->convertTypedValue($value, $type);
+               }
+
+               return $values;
+       }
+
+       /**
+        * convert string value to the expected type
+        *
+        * @param string $value
+        * @param ValueType $type
+        *
+        * @return string|int|float|bool|array
+        */
+       private function convertTypedValue(string $value, ValueType $type): string|int|float|bool|array {
+               switch ($type) {
+                       case ValueType::INT:
+                               return (int)$value;
+                       case ValueType::FLOAT:
+                               return (float)$value;
+                       case ValueType::BOOL:
+                               return in_array(strtolower($value), ['1', 'true', 'yes', 'on']);
+                       case ValueType::ARRAY:
+                               try {
+                                       return json_decode($value, true, flags: JSON_THROW_ON_ERROR);
+                               } catch (JsonException) {
+                                       // ignoreable
+                               }
+                               break;
+               }
+               return $value;
+       }
+
+
+       private function decryptSensitiveValue(string $userId, string $app, string $key, string &$value): void {
+               if (!$this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags'] ?? 0)) {
+                       return;
+               }
+
+               if (!str_starts_with($value, self::ENCRYPTION_PREFIX)) {
+                       return;
+               }
+
+               try {
+                       $value = $this->crypto->decrypt(substr($value, self::ENCRYPTION_PREFIX_LENGTH));
+               } catch (\Exception $e) {
+                       $this->logger->warning('could not decrypt sensitive value', [
+                               'userId' => $userId,
+                               'app' => $app,
+                               'key' => $key,
+                               'value' => $value,
+                               'exception' => $e
+                       ]);
+               }
+       }
+}
index 10d6dee9d07f23ec949114d260cab6c71dafc85c..e109636f1fad4ec2a43fa9bae54ad36ba88750b5 100644 (file)
@@ -138,6 +138,7 @@ use OCP\Collaboration\AutoComplete\IManager;
 use OCP\Collaboration\Reference\IReferenceManager;
 use OCP\Command\IBus;
 use OCP\Comments\ICommentsManager;
+use OCP\Config\IUserPreferences;
 use OCP\Contacts\ContactsMenu\IActionFactory;
 use OCP\Contacts\ContactsMenu\IContactsStore;
 use OCP\Defaults;
@@ -237,7 +238,6 @@ use OCP\User\Events\UserLoggedInEvent;
 use OCP\User\Events\UserLoggedInWithCookieEvent;
 use OCP\User\Events\UserLoggedOutEvent;
 use OCP\User\IAvailabilityCoordinator;
-use OCP\UserPreferences\IUserPreferences;
 use Psr\Container\ContainerExceptionInterface;
 use Psr\Container\ContainerInterface;
 use Psr\Log\LoggerInterface;
@@ -568,7 +568,7 @@ class Server extends ServerContainer implements IServerContainer {
                });
 
                $this->registerAlias(IAppConfig::class, \OC\AppConfig::class);
-               $this->registerAlias(IUserPreferences::class, \OC\UserPreferences::class);
+               $this->registerAlias(IUserPreferences::class, \OC\Config\UserPreferences::class);
 
                $this->registerService(IFactory::class, function (Server $c) {
                        return new \OC\L10N\Factory(
diff --git a/lib/private/UserPreferences.php b/lib/private/UserPreferences.php
deleted file mode 100644 (file)
index 2231329..0000000
+++ /dev/null
@@ -1,1834 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OC;
-
-use InvalidArgumentException;
-use JsonException;
-use OCP\DB\Exception as DBException;
-use OCP\DB\IResult;
-use OCP\DB\QueryBuilder\IQueryBuilder;
-use OCP\IConfig;
-use OCP\IDBConnection;
-use OCP\Security\ICrypto;
-use OCP\UserPreferences\Exceptions\IncorrectTypeException;
-use OCP\UserPreferences\Exceptions\TypeConflictException;
-use OCP\UserPreferences\Exceptions\UnknownKeyException;
-use OCP\UserPreferences\IUserPreferences;
-use OCP\UserPreferences\ValueType;
-use Psr\Log\LoggerInterface;
-
-/**
- * This class provides an easy way for apps to store user preferences in the
- * database.
- * Supports **lazy loading**
- *
- * ### What is lazy loading ?
- * In order to avoid loading useless user preferences into memory for each request,
- * only non-lazy values are now loaded.
- *
- * Once a value that is lazy is requested, all lazy values will be loaded.
- *
- * Similarly, some methods from this class are marked with a warning about ignoring
- * lazy loading. Use them wisely and only on parts of the code that are called
- * during specific requests or actions to avoid loading the lazy values all the time.
- *
- * @since 31.0.0
- */
-class UserPreferences implements IUserPreferences {
-       private const USER_MAX_LENGTH = 64;
-       private const APP_MAX_LENGTH = 32;
-       private const KEY_MAX_LENGTH = 64;
-       private const INDEX_MAX_LENGTH = 64;
-       private const ENCRYPTION_PREFIX = '$UserPreferencesEncryption$';
-       private const ENCRYPTION_PREFIX_LENGTH = 27; // strlen(self::ENCRYPTION_PREFIX)
-
-       /** @var array<string, array<string, array<string, mixed>>> [ass'user_id' => ['app_id' => ['key' => 'value']]] */
-       private array $fastCache = [];   // cache for normal preference keys
-       /** @var array<string, array<string, array<string, mixed>>> ['user_id' => ['app_id' => ['key' => 'value']]] */
-       private array $lazyCache = [];   // cache for lazy preference keys
-       /** @var array<string, array<string, array<string, <'type', ValueType>|<'flags', int>> ['user_id' => ['app_id' => ['key' => ['type' => ValueType, 'flags' => bitflag]]] */
-       private array $valueDetails = [];  // type for all preference values
-       /** @var array<string, array<string, array<string, ValueType>>> ['user_id' => ['app_id' => ['key' => bitflag]]] */
-       private array $valueTypes = [];  // type for all preference values
-       /** @var array<string, array<string, array<string, int>>> ['user_id' => ['app_id' => ['key' => bitflag]]] */
-       private array $valueFlags = [];  // type for all preference values
-       /** @var array<string, boolean> ['user_id' => bool] */
-       private array $fastLoaded = [];
-       /** @var array<string, boolean> ['user_id' => bool] */
-       private array $lazyLoaded = [];
-
-       public function __construct(
-               protected IDBConnection $connection,
-               protected LoggerInterface $logger,
-               protected ICrypto $crypto,
-       ) {
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $appId optional id of app
-        *
-        * @return list<string> list of userIds
-        * @since 31.0.0
-        */
-       public function getUserIds(string $appId = ''): array {
-               $this->assertParams(app: $appId, allowEmptyUser: true, allowEmptyApp: true);
-
-               $qb = $this->connection->getQueryBuilder();
-               $qb->from('preferences');
-               $qb->select('userid');
-               $qb->groupBy('userid');
-               if ($appId !== '') {
-                       $qb->where($qb->expr()->eq('appid', $qb->createNamedParameter($appId)));
-               }
-
-               $result = $qb->executeQuery();
-               $rows = $result->fetchAll();
-               $userIds = [];
-               foreach ($rows as $row) {
-                       $userIds[] = $row['userid'];
-               }
-
-               return $userIds;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @return list<string> list of app ids
-        * @since 31.0.0
-        */
-       public function getApps(string $userId): array {
-               $this->assertParams($userId, allowEmptyApp: true);
-               $this->loadPreferencesAll($userId);
-               $apps = array_merge(array_keys($this->fastCache[$userId] ?? []), array_keys($this->lazyCache[$userId] ?? []));
-               sort($apps);
-
-               return array_values(array_unique($apps));
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        *
-        * @return list<string> list of stored preference keys
-        * @since 31.0.0
-        */
-       public function getKeys(string $userId, string $app): array {
-               $this->assertParams($userId, $app);
-               $this->loadPreferencesAll($userId);
-               // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
-               $keys = array_map('strval', array_keys(($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? [])));
-               sort($keys);
-
-               return array_values(array_unique($keys));
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
-        *
-        * @return bool TRUE if key exists
-        * @since 31.0.0
-        */
-       public function hasKey(string $userId, string $app, string $key, ?bool $lazy = false): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferences($userId, $lazy);
-
-               if ($lazy === null) {
-                       $appCache = $this->getValues($userId, $app);
-                       return isset($appCache[$key]);
-               }
-
-               if ($lazy) {
-                       return isset($this->lazyCache[$userId][$app][$key]);
-               }
-
-               return isset($this->fastCache[$userId][$app][$key]);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
-        *
-        * @return bool
-        * @throws UnknownKeyException if preference key is not known
-        * @since 31.0.0
-        */
-       public function isSensitive(string $userId, string $app, string $key, ?bool $lazy = false): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferences($userId, $lazy);
-
-               if (!isset($this->valueDetails[$userId][$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               return $this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags']);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy TRUE to search within lazy loaded preferences, NULL to search within all preferences
-        *
-        * @return bool
-        * @throws UnknownKeyException if preference key is not known
-        * @since 31.0.0
-        */
-       public function isIndexed(string $userId, string $app, string $key, ?bool $lazy = false): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferences($userId, $lazy);
-
-               if (!isset($this->valueDetails[$userId][$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               return $this->isFlagged(self::FLAG_INDEXED, $this->valueDetails[$userId][$app][$key]['flags']);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app if of the app
-        * @param string $key preference key
-        *
-        * @return bool TRUE if preference is lazy loaded
-        * @throws UnknownKeyException if preference key is not known
-        * @see IUserPreferences for details about lazy loading
-        * @since 31.0.0
-        */
-       public function isLazy(string $userId, string $app, string $key): bool {
-               // there is a huge probability the non-lazy preferences are already loaded
-               // meaning that we can start by only checking if a current non-lazy key exists
-               if ($this->hasKey($userId, $app, $key, false)) {
-                       return false; // meaning key is not lazy.
-               }
-
-               // as key is not found as non-lazy, we load and search in the lazy preferences
-               if ($this->hasKey($userId, $app, $key, true)) {
-                       return true;
-               }
-
-               throw new UnknownKeyException('unknown preference key');
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $prefix preference keys prefix to search
-        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
-        *
-        * @return array<string, string|int|float|bool|array> [key => value]
-        * @since 31.0.0
-        */
-       public function getValues(
-               string $userId,
-               string $app,
-               string $prefix = '',
-               bool $filtered = false,
-       ): array {
-               $this->assertParams($userId, $app, $prefix);
-               // if we want to filter values, we need to get sensitivity
-               $this->loadPreferencesAll($userId);
-               // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
-               $values = array_filter(
-                       $this->formatAppValues($userId, $app, ($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? []), $filtered),
-                       function (string $key) use ($prefix): bool {
-                               return str_starts_with($key, $prefix); // filter values based on $prefix
-                       }, ARRAY_FILTER_USE_KEY
-               );
-
-               return $values;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
-        *
-        * @return array<string, array<string, string|int|float|bool|array>> [appId => [key => value]]
-        * @since 31.0.0
-        */
-       public function getAllValues(string $userId, bool $filtered = false): array {
-               $this->assertParams($userId, allowEmptyApp: true);
-               $this->loadPreferencesAll($userId);
-
-               $result = [];
-               foreach ($this->getApps($userId) as $app) {
-                       // array_merge() will remove numeric keys (here preference keys), so addition arrays instead
-                       $cached = ($this->fastCache[$userId][$app] ?? []) + ($this->lazyCache[$userId][$app] ?? []);
-                       $result[$app] = $this->formatAppValues($userId, $app, $cached, $filtered);
-               }
-
-               return $result;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $key preference key
-        * @param bool $lazy search within lazy loaded preferences
-        * @param ValueType|null $typedAs enforce type for the returned values
-        *
-        * @return array<string, string|int|float|bool|array> [appId => value]
-        * @since 31.0.0
-        */
-       public function getValuesByApps(string $userId, string $key, bool $lazy = false, ?ValueType $typedAs = null): array {
-               $this->assertParams($userId, '', $key, allowEmptyApp: true);
-               $this->loadPreferences($userId, $lazy);
-
-               /** @var array<array-key, array<array-key, mixed>> $cache */
-               if ($lazy) {
-                       $cache = $this->lazyCache[$userId];
-               } else {
-                       $cache = $this->fastCache[$userId];
-               }
-
-               $values = [];
-               foreach (array_keys($cache) as $app) {
-                       if (isset($cache[$app][$key])) {
-                               $value = $cache[$app][$key];
-                               try {
-                                       $this->decryptSensitiveValue($userId, $app, $key, $value);
-                                       $value = $this->convertTypedValue($value, $typedAs ?? $this->getValueType($userId, $app, $key, $lazy));
-                               } catch (IncorrectTypeException|UnknownKeyException) {
-                               }
-                               $values[$app] = $value;
-                       }
-               }
-
-               return $values;
-       }
-
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param ValueType|null $typedAs enforce type for the returned values
-        * @param array|null $userIds limit to a list of user ids
-        *
-        * @return array<string, string|int|float|bool|array> [userId => value]
-        * @since 31.0.0
-        */
-       public function getValuesByUsers(
-               string $app,
-               string $key,
-               ?ValueType $typedAs = null,
-               ?array $userIds = null,
-       ): array {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-
-               $qb = $this->connection->getQueryBuilder();
-               $qb->select('userid', 'configvalue', 'type')
-                       ->from('preferences')
-                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
-                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
-
-               $values = [];
-               // this nested function will execute current Query and store result within $values.
-               $executeAndStoreValue = function (IQueryBuilder $qb) use (&$values, $typedAs): IResult {
-                       $result = $qb->executeQuery();
-                       while ($row = $result->fetch()) {
-                               $value = $row['configvalue'];
-                               try {
-                                       $value = $this->convertTypedValue($value, $typedAs ?? ValueType::from((int) $row['type']));
-                               } catch (IncorrectTypeException) {
-                               }
-                               $values[$row['userid']] = $value;
-                       }
-                       return $result;
-               };
-
-               // if no userIds to filter, we execute query as it is and returns all values ...
-               if ($userIds === null) {
-                       $result = $executeAndStoreValue($qb);
-                       $result->closeCursor();
-                       return $values;
-               }
-
-               // if userIds to filter, we chunk the list and execute the same query multiple times until we get all values
-               $result = null;
-               $qb->andWhere($qb->expr()->in('userid', $qb->createParameter('userIds')));
-               foreach (array_chunk($userIds, 50, true) as $chunk) {
-                       $qb->setParameter('userIds', $chunk, IQueryBuilder::PARAM_STR_ARRAY);
-                       $result = $executeAndStoreValue($qb);
-               }
-               $result?->closeCursor();
-
-               return $values;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $caseInsensitive non-case-sensitive search, only works if $value is a string
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueString(string $app, string $key, string $value, bool $caseInsensitive = false): array {
-               return $this->searchUsersByTypedValue($app, $key, $value, $caseInsensitive);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $caseInsensitive non-case-sensitive search, only works if $value is a string
-        * @internal
-        * @deprecated since 31.0.0 - {@see }
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueDeprecated(string $app, string $key, string $value, bool $caseInsensitive = false): array {
-               return $this->searchUsersByTypedValue($app, $key, $value, $caseInsensitive, true);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $value preference value
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueInt(string $app, string $key, int $value): array {
-               return $this->searchUsersByValueString($app, $key, (string)$value);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $values list of preference values
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValues(string $app, string $key, array $values): array {
-               return $this->searchUsersByTypedValue($app, $key, $values);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $value preference value
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueBool(string $app, string $key, bool $value): array {
-               $values = ['0', 'off', 'false', 'no'];
-               if ($value) {
-                       $values = ['1', 'on', 'true', 'yes'];
-               }
-               return $this->searchUsersByValues($app, $key, $values);
-       }
-
-       /**
-        * returns a list of users with preference key set to a specific value, or within the list of
-        * possible values
-        *
-        * @param string $app
-        * @param string $key
-        * @param string|array $value
-        * @param bool $caseInsensitive
-        * @param bool $withinNotIndexedField DEPRECATED: should only be used to stay compatible with not-indexed/pre-31 preferences value
-        *
-        * @return list<string>
-        */
-       private function searchUsersByTypedValue(string $app, string $key, string|array $value, bool $caseInsensitive = false): array {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-
-               $qb = $this->connection->getQueryBuilder();
-               $qb->from('preferences');
-               $qb->select('userid');
-               $qb->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)));
-               $qb->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
-
-               // search within 'indexed' OR 'configvalue' (but if 'flags' is not set as indexed)
-               // TODO: when implementing config lexicon remove the searches on 'configvalue' if value is set as indexed
-               $configValueColumn = ($this->connection->getDatabaseProvider() === IDBConnection::PLATFORM_ORACLE) ? $qb->expr()->castColumn('configvalue', IQueryBuilder::PARAM_STR) : 'configvalue';
-               if (is_array($value)) {
-                       $where = $qb->expr()->orX(
-                               $qb->expr()->in('indexed', $qb->createNamedParameter($value, IQueryBuilder::PARAM_STR_ARRAY)),
-                               $qb->expr()->andX(
-                                       $qb->createFunction('NOT ' . $qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED)),
-                                       $qb->expr()->in($configValueColumn, $qb->createNamedParameter($value, IQueryBuilder::PARAM_STR_ARRAY))
-                               )
-                       );
-               } else {
-                       if ($caseInsensitive) {
-                               $where = $qb->expr()->orX(
-                                       $qb->expr()->eq($qb->func()->lower('indexed'), $qb->createNamedParameter(strtolower($value))),
-                                       $qb->expr()->andX(
-                                               $qb->createFunction('NOT ' . $qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED)),
-                                               $qb->expr()->eq($qb->func()->lower($configValueColumn), $qb->createNamedParameter(strtolower($value)))
-                                       )
-                               );
-                       } else {
-                               $where = $qb->expr()->orX(
-                                       $qb->expr()->eq('indexed', $qb->createNamedParameter($value)),
-                                       $qb->expr()->andX(
-                                               $qb->createFunction('NOT ' . $qb->expr()->bitwiseAnd('flags', self::FLAG_INDEXED)),
-                                               $qb->expr()->eq($configValueColumn, $qb->createNamedParameter($value))
-                                       )
-                               );
-                       }
-               }
-
-               $qb->andWhere($where);
-
-               $userIds = [];
-               $result = $qb->executeQuery();
-               $rows = $result->fetchAll();
-               foreach ($rows as $row) {
-                       $userIds[] = $row['userid'];
-               }
-
-               return $userIds;
-       }
-
-       /**
-        * Get the preference value as string.
-        * If the value does not exist the given default will be returned.
-        *
-        * Set lazy to `null` to ignore it and get the value from either source.
-        *
-        * **WARNING:** Method is internal and **SHOULD** not be used, as it is better to get the value with a type.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $default preference value
-        * @param null|bool $lazy get preference as lazy loaded or not. can be NULL
-        *
-        * @return string the value or $default
-        * @throws TypeConflictException
-        * @internal
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see getValueString()
-        * @see getValueInt()
-        * @see getValueFloat()
-        * @see getValueBool()
-        * @see getValueArray()
-        */
-       public function getValueMixed(
-               string $userId,
-               string $app,
-               string $key,
-               string $default = '',
-               ?bool $lazy = false,
-       ): string {
-               try {
-                       $lazy ??= $this->isLazy($userId, $app, $key);
-               } catch (UnknownKeyException) {
-                       return $default;
-               }
-
-               return $this->getTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       $default,
-                       $lazy,
-                       ValueType::MIXED
-               );
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return string stored preference value or $default if not set in database
-        * @throws InvalidArgumentException if one of the argument format is invalid
-        * @throws TypeConflictException in case of conflict with the value type set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function getValueString(
-               string $userId,
-               string $app,
-               string $key,
-               string $default = '',
-               bool $lazy = false,
-       ): string {
-               return $this->getTypedValue($userId, $app, $key, $default, $lazy, ValueType::STRING);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return int stored preference value or $default if not set in database
-        * @throws InvalidArgumentException if one of the argument format is invalid
-        * @throws TypeConflictException in case of conflict with the value type set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function getValueInt(
-               string $userId,
-               string $app,
-               string $key,
-               int $default = 0,
-               bool $lazy = false,
-       ): int {
-               return (int)$this->getTypedValue($userId, $app, $key, (string)$default, $lazy, ValueType::INT);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param float $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return float stored preference value or $default if not set in database
-        * @throws InvalidArgumentException if one of the argument format is invalid
-        * @throws TypeConflictException in case of conflict with the value type set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function getValueFloat(
-               string $userId,
-               string $app,
-               string $key,
-               float $default = 0,
-               bool $lazy = false,
-       ): float {
-               return (float)$this->getTypedValue($userId, $app, $key, (string)$default, $lazy, ValueType::FLOAT);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return bool stored preference value or $default if not set in database
-        * @throws InvalidArgumentException if one of the argument format is invalid
-        * @throws TypeConflictException in case of conflict with the value type set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function getValueBool(
-               string $userId,
-               string $app,
-               string $key,
-               bool $default = false,
-               bool $lazy = false,
-       ): bool {
-               $b = strtolower($this->getTypedValue($userId, $app, $key, $default ? 'true' : 'false', $lazy, ValueType::BOOL));
-               return in_array($b, ['1', 'true', 'yes', 'on']);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return array stored preference value or $default if not set in database
-        * @throws InvalidArgumentException if one of the argument format is invalid
-        * @throws TypeConflictException in case of conflict with the value type set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function getValueArray(
-               string $userId,
-               string $app,
-               string $key,
-               array $default = [],
-               bool $lazy = false,
-       ): array {
-               try {
-                       $defaultJson = json_encode($default, JSON_THROW_ON_ERROR);
-                       $value = json_decode($this->getTypedValue($userId, $app, $key, $defaultJson, $lazy, ValueType::ARRAY), true, flags: JSON_THROW_ON_ERROR);
-
-                       return is_array($value) ? $value : [];
-               } catch (JsonException) {
-                       return [];
-               }
-       }
-
-       /**
-        * @param string $userId
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        * @param ValueType $type value type
-        *
-        * @return string
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        */
-       private function getTypedValue(
-               string $userId,
-               string $app,
-               string $key,
-               string $default,
-               bool $lazy,
-               ValueType $type,
-       ): string {
-               $this->assertParams($userId, $app, $key, valueType: $type);
-               $this->loadPreferences($userId, $lazy);
-
-               /**
-                * We ignore check if mixed type is requested.
-                * If type of stored value is set as mixed, we don't filter.
-                * If type of stored value is defined, we compare with the one requested.
-                */
-               $knownType = $this->valueDetails[$userId][$app][$key]['type'] ?? null;
-               if ($type !== ValueType::MIXED
-                       && $knownType !== null
-                       && $knownType !== ValueType::MIXED
-                       && $type !== $knownType) {
-                       $this->logger->warning('conflict with value type from database', ['app' => $app, 'key' => $key, 'type' => $type, 'knownType' => $knownType]);
-                       throw new TypeConflictException('conflict with value type from database');
-               }
-
-               /**
-                * - the pair $app/$key cannot exist in both array,
-                * - we should still return an existing non-lazy value even if current method
-                *   is called with $lazy is true
-                *
-                * This way, lazyCache will be empty until the load for lazy preferences value is requested.
-                */
-               if (isset($this->lazyCache[$userId][$app][$key])) {
-                       $value = $this->lazyCache[$userId][$app][$key];
-               } elseif (isset($this->fastCache[$userId][$app][$key])) {
-                       $value = $this->fastCache[$userId][$app][$key];
-               } else {
-                       return $default;
-               }
-
-               $this->decryptSensitiveValue($userId, $app, $key, $value);
-               return $value;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @return ValueType type of the value
-        * @throws UnknownKeyException if preference key is not known
-        * @throws IncorrectTypeException if preferences value type is not known
-        * @since 31.0.0
-        */
-       public function getValueType(string $userId, string $app, string $key, ?bool $lazy = null): ValueType {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferences($userId, $lazy);
-
-               if (!isset($this->valueDetails[$userId][$app][$key]['type'])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               return $this->valueDetails[$userId][$app][$key]['type'];
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy lazy loading
-        *
-        * @return ValueType type of the value
-        * @throws UnknownKeyException if preference key is not known
-        * @throws IncorrectTypeException if preferences value type is not known
-        * @since 31.0.0
-        */
-       public function getValueFlags(string $userId, string $app, string $key, bool $lazy = false): int {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferences($userId, $lazy);
-
-               if (!isset($this->valueDetails[$userId][$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               return $this->valueDetails[$userId][$app][$key]['flags'];
-       }
-
-       /**
-        * Store a preference key and its value in database as VALUE_MIXED
-        *
-        * **WARNING:** Method is internal and **MUST** not be used as it is best to set a real value type
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED
-        * @internal
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueString()
-        * @see setValueInt()
-        * @see setValueFloat()
-        * @see setValueBool()
-        * @see setValueArray()
-        */
-       public function setValueMixed(
-               string $userId,
-               string $app,
-               string $key,
-               string $value,
-               bool $lazy = false,
-               int $flags = 0,
-       ): bool {
-               return $this->setTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       $value,
-                       $lazy,
-                       $flags,
-                       ValueType::MIXED
-               );
-       }
-
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function setValueString(
-               string $userId,
-               string $app,
-               string $key,
-               string $value,
-               bool $lazy = false,
-               int $flags = 0,
-       ): bool {
-               return $this->setTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       $value,
-                       $lazy,
-                       $flags,
-                       ValueType::STRING
-               );
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function setValueInt(
-               string $userId,
-               string $app,
-               string $key,
-               int $value,
-               bool $lazy = false,
-               int $flags = 0,
-       ): bool {
-               if ($value > 2000000000) {
-                       $this->logger->debug('You are trying to store an integer value around/above 2,147,483,647. This is a reminder that reaching this theoretical limit on 32 bits system will throw an exception.');
-               }
-
-               return $this->setTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       (string)$value,
-                       $lazy,
-                       $flags,
-                       ValueType::INT
-               );
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param float $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function setValueFloat(
-               string $userId,
-               string $app,
-               string $key,
-               float $value,
-               bool $lazy = false,
-               int $flags = 0,
-       ): bool {
-               return $this->setTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       (string)$value,
-                       $lazy,
-                       $flags,
-                       ValueType::FLOAT
-               );
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function setValueBool(
-               string $userId,
-               string $app,
-               string $key,
-               bool $value,
-               bool $lazy = false,
-               int $flags = 0
-       ): bool {
-               return $this->setTypedValue(
-                       $userId,
-                       $app,
-                       $key,
-                       ($value) ? '1' : '0',
-                       $lazy,
-                       $flags,
-                       ValueType::BOOL
-               );
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @throws JsonException
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       public function setValueArray(
-               string $userId,
-               string $app,
-               string $key,
-               array $value,
-               bool $lazy = false,
-               int $flags = 0,
-       ): bool {
-               try {
-                       return $this->setTypedValue(
-                               $userId,
-                               $app,
-                               $key,
-                               json_encode($value, JSON_THROW_ON_ERROR),
-                               $lazy,
-                               $flags,
-                               ValueType::ARRAY
-                       );
-               } catch (JsonException $e) {
-                       $this->logger->warning('could not setValueArray', ['app' => $app, 'key' => $key, 'exception' => $e]);
-                       throw $e;
-               }
-       }
-
-       /**
-        * Store a preference key and its value in database
-        *
-        * If preference key is already known with the exact same preference value and same sensitive/lazy status, the
-        * database is not updated. If preference value was previously stored as sensitive, status will not be
-        * altered.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $lazy preferences set as lazy loaded
-        * @param ValueType $type value type
-        *
-        * @return bool TRUE if value was updated in database
-        * @throws TypeConflictException if type from database is not VALUE_MIXED and different from the requested one
-        * @see IUserPreferences for explanation about lazy loading
-        */
-       private function setTypedValue(
-               string $userId,
-               string $app,
-               string $key,
-               string $value,
-               bool $lazy,
-               int $flags,
-               ValueType $type,
-       ): bool {
-               $this->assertParams($userId, $app, $key, valueType: $type);
-               $this->loadPreferences($userId, $lazy);
-
-               $inserted = $refreshCache = false;
-               $origValue = $value;
-               $sensitive = $this->isFlagged(self::FLAG_SENSITIVE, $flags);
-               if ($sensitive || ($this->hasKey($userId, $app, $key, $lazy) && $this->isSensitive($userId, $app, $key, $lazy))) {
-                       $value = self::ENCRYPTION_PREFIX . $this->crypto->encrypt($value);
-                       $flags |= UserPreferences::FLAG_SENSITIVE;
-               }
-
-               // if requested, we fill the 'indexed' field with current value
-               $indexed = '';
-               if ($type !== ValueType::ARRAY && $this->isFlagged(self::FLAG_INDEXED, $flags)) {
-                       if ($this->isFlagged(self::FLAG_SENSITIVE, $flags)) {
-                               $this->logger->warning('sensitive value are not to be indexed');
-                       } else if (strlen($value) > self::USER_MAX_LENGTH) {
-                               $this->logger->warning('value is too lengthy to be indexed');
-                       } else {
-                               $indexed = $value;
-                       }
-               }
-
-               if ($this->hasKey($userId, $app, $key, $lazy)) {
-                       /**
-                        * no update if key is already known with set lazy status and value is
-                        * not different, unless sensitivity is switched from false to true.
-                        */
-                       if ($origValue === $this->getTypedValue($userId, $app, $key, $value, $lazy, $type)
-                               && (!$sensitive || $this->isSensitive($userId, $app, $key, $lazy))) {
-                               return false;
-                       }
-               } else {
-                       /**
-                        * if key is not known yet, we try to insert.
-                        * It might fail if the key exists with a different lazy flag.
-                        */
-                       try {
-                               $insert = $this->connection->getQueryBuilder();
-                               $insert->insert('preferences')
-                                       ->setValue('userid', $insert->createNamedParameter($userId))
-                                       ->setValue('appid', $insert->createNamedParameter($app))
-                                       ->setValue('lazy', $insert->createNamedParameter(($lazy) ? 1 : 0, IQueryBuilder::PARAM_INT))
-                                       ->setValue('type', $insert->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
-                                       ->setValue('flags', $insert->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
-                                       ->setValue('indexed', $insert->createNamedParameter($indexed))
-                                       ->setValue('configkey', $insert->createNamedParameter($key))
-                                       ->setValue('configvalue', $insert->createNamedParameter($value));
-                               $insert->executeStatement();
-                               $inserted = true;
-                       } catch (DBException $e) {
-                               if ($e->getReason() !== DBException::REASON_UNIQUE_CONSTRAINT_VIOLATION) {
-                                       throw $e; // TODO: throw exception or just log and returns false !?
-                               }
-                       }
-               }
-
-               /**
-                * We cannot insert a new row, meaning we need to update an already existing one
-                */
-               if (!$inserted) {
-                       $currType = $this->valueDetails[$userId][$app][$key]['type'] ?? null;
-                       if ($currType === null) { // this might happen when switching lazy loading status
-                               $this->loadPreferencesAll($userId);
-                               $currType = $this->valueDetails[$userId][$app][$key]['type'];
-                       }
-
-                       /**
-                        * We only log a warning and set it to VALUE_MIXED.
-                        */
-                       if ($currType === null) {
-                               $this->logger->warning('Value type is set to zero (0) in database. This is not supposed to happens', ['app' => $app, 'key' => $key]);
-                               $currType = ValueType::MIXED;
-                       }
-
-                       /**
-                        * we only accept a different type from the one stored in database
-                        * if the one stored in database is not-defined (VALUE_MIXED)
-                        */
-                       if ($currType !== ValueType::MIXED &&
-                               $currType !== $type) {
-                               try {
-                                       $currTypeDef = $currType->getDefinition();
-                                       $typeDef = $type->getDefinition();
-                               } catch (IncorrectTypeException) {
-                                       $currTypeDef = $currType->value;
-                                       $typeDef = $type->value;
-                               }
-                               throw new TypeConflictException('conflict between new type (' . $typeDef . ') and old type (' . $currTypeDef . ')');
-                       }
-
-                       if ($lazy !== $this->isLazy($userId, $app, $key)) {
-                               $refreshCache = true;
-                       }
-
-                       $update = $this->connection->getQueryBuilder();
-                       $update->update('preferences')
-                               ->set('configvalue', $update->createNamedParameter($value))
-                               ->set('lazy', $update->createNamedParameter(($lazy) ? 1 : 0, IQueryBuilder::PARAM_INT))
-                               ->set('type', $update->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
-                               ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
-                               ->set('indexed', $update->createNamedParameter($indexed))
-                               ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
-                               ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                               ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-
-                       $update->executeStatement();
-               }
-
-               if ($refreshCache) {
-                       $this->clearCache($userId);
-                       return true;
-               }
-
-               // update local cache
-               if ($lazy) {
-                       $this->lazyCache[$userId][$app][$key] = $value;
-               } else {
-                       $this->fastCache[$userId][$app][$key] = $value;
-               }
-               $this->valueDetails[$userId][$app][$key] = [
-                       'type' => $type,
-                       'flags' => $flags
-               ];
-
-               return true;
-       }
-
-       /**
-        * Change the type of preference value.
-        *
-        * **WARNING:** Method is internal and **MUST** not be used as it may break things.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param ValueType $type value type
-        *
-        * @return bool TRUE if database update were necessary
-        * @throws UnknownKeyException if $key is now known in database
-        * @throws IncorrectTypeException if $type is not valid
-        * @internal
-        * @since 31.0.0
-        */
-       public function updateType(string $userId, string $app, string $key, ValueType $type = ValueType::MIXED): bool {
-               $this->assertParams($userId, $app, $key, valueType: $type);
-               $this->loadPreferencesAll($userId);
-               $this->isLazy($userId, $app, $key); // confirm key exists
-
-               $update = $this->connection->getQueryBuilder();
-               $update->update('preferences')
-                       ->set('type', $update->createNamedParameter($type->value, IQueryBuilder::PARAM_INT))
-                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
-                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-               $update->executeStatement();
-
-               $this->valueDetails[$userId][$app][$key]['type'] = $type;
-
-               return true;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
-        *
-        * @return bool TRUE if entry was found in database and an update was necessary
-        * @since 31.0.0
-        */
-       public function updateSensitive(string $userId, string $app, string $key, bool $sensitive): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferencesAll($userId);
-
-               try {
-                       if ($sensitive === $this->isSensitive($userId, $app, $key, null)) {
-                               return false;
-                       }
-               } catch (UnknownKeyException) {
-                       return false;
-               }
-
-               $lazy = $this->isLazy($userId, $app, $key);
-               if ($lazy) {
-                       $cache = $this->lazyCache;
-               } else {
-                       $cache = $this->fastCache;
-               }
-
-               if (!isset($cache[$userId][$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               $value = $cache[$userId][$app][$key];
-               $flags = $this->getValueFlags($userId, $app, $key);
-               if ($sensitive) {
-                       $flags |= self::FLAG_SENSITIVE;
-                       $value = self::ENCRYPTION_PREFIX . $this->crypto->encrypt($value);
-               } else {
-                       $flags &= ~self::FLAG_SENSITIVE;
-                       $this->decryptSensitiveValue($userId, $app, $key, $value);
-               }
-
-               $update = $this->connection->getQueryBuilder();
-               $update->update('preferences')
-                       ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
-                       ->set('configvalue', $update->createNamedParameter($value))
-                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
-                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-               $update->executeStatement();
-
-               $this->valueDetails[$userId][$app][$key]['flags'] = $flags;
-
-               return true;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app
-        * @param string $key
-        * @param bool $sensitive
-        *
-        * @since 31.0.0
-        */
-       public function updateGlobalSensitive(string $app, string $key, bool $sensitive): void {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-               foreach (array_keys($this->getValuesByUsers($app, $key)) as $userId) {
-                       try {
-                               $this->updateSensitive($userId, $app, $key, $sensitive);
-                       } catch (UnknownKeyException) {
-                               // should not happen and can be ignored
-                       }
-               }
-
-               $this->clearCacheAll(); // we clear all cache
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId
-        * @param string $app
-        * @param string $key
-        * @param bool $indexed
-        *
-        * @return bool
-        * @throws DBException
-        * @throws IncorrectTypeException
-        * @throws UnknownKeyException
-        * @since 31.0.0
-        */
-       public function updateIndexed(string $userId, string $app, string $key, bool $indexed): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferencesAll($userId);
-
-               try {
-                       if ($indexed === $this->isIndexed($userId, $app, $key, null)) {
-                               return false;
-                       }
-               } catch (UnknownKeyException) {
-                       return false;
-               }
-
-               $lazy = $this->isLazy($userId, $app, $key);
-               if ($lazy) {
-                       $cache = $this->lazyCache;
-               } else {
-                       $cache = $this->fastCache;
-               }
-
-               if (!isset($cache[$userId][$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               $value = $cache[$userId][$app][$key];
-               $flags = $this->getValueFlags($userId, $app, $key);
-               if ($indexed) {
-                       $indexed = $value;
-               } else {
-                       $flags &= ~self::FLAG_INDEXED;
-                       $indexed = '';
-               }
-
-               $update = $this->connection->getQueryBuilder();
-               $update->update('preferences')
-                          ->set('flags', $update->createNamedParameter($flags, IQueryBuilder::PARAM_INT))
-                          ->set('indexed', $update->createNamedParameter($indexed))
-                          ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
-                          ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                          ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-               $update->executeStatement();
-
-               $this->valueDetails[$userId][$app][$key]['flags'] = $flags;
-
-               return true;
-       }
-
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app
-        * @param string $key
-        * @param bool $indexed
-        *
-        * @since 31.0.0
-        */
-       public function updateGlobalIndexed(string $app, string $key, bool $indexed): void {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-               foreach (array_keys($this->getValuesByUsers($app, $key)) as $userId) {
-                       try {
-                               $this->updateIndexed($userId, $app, $key, $indexed);
-                       } catch (UnknownKeyException) {
-                               // should not happen and can be ignored
-                       }
-               }
-
-               $this->clearCacheAll(); // we clear all cache
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
-        *
-        * @return bool TRUE if entry was found in database and an update was necessary
-        * @since 31.0.0
-        */
-       public function updateLazy(string $userId, string $app, string $key, bool $lazy): bool {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferencesAll($userId);
-
-               try {
-                       if ($lazy === $this->isLazy($userId, $app, $key)) {
-                               return false;
-                       }
-               } catch (UnknownKeyException) {
-                       return false;
-               }
-
-               $update = $this->connection->getQueryBuilder();
-               $update->update('preferences')
-                       ->set('lazy', $update->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT))
-                       ->where($update->expr()->eq('userid', $update->createNamedParameter($userId)))
-                       ->andWhere($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-               $update->executeStatement();
-
-               // At this point, it is a lot safer to clean cache
-               $this->clearCache($userId);
-
-               return true;
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
-        *
-        * @since 31.0.0
-        */
-       public function updateGlobalLazy(string $app, string $key, bool $lazy): void {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-
-               $update = $this->connection->getQueryBuilder();
-               $update->update('preferences')
-                       ->set('lazy', $update->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT))
-                       ->where($update->expr()->eq('appid', $update->createNamedParameter($app)))
-                       ->andWhere($update->expr()->eq('configkey', $update->createNamedParameter($key)));
-               $update->executeStatement();
-
-               $this->clearCacheAll();
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @return array
-        * @throws UnknownKeyException if preference key is not known in database
-        * @since 31.0.0
-        */
-       public function getDetails(string $userId, string $app, string $key): array {
-               $this->assertParams($userId, $app, $key);
-               $this->loadPreferencesAll($userId);
-               $lazy = $this->isLazy($userId, $app, $key);
-
-               if ($lazy) {
-                       $cache = $this->lazyCache[$userId];
-               } else {
-                       $cache = $this->fastCache[$userId];
-               }
-
-               $type = $this->getValueType($userId, $app, $key);
-               try {
-                       $typeString = $type->getDefinition();
-               } catch (IncorrectTypeException $e) {
-                       $this->logger->warning('type stored in database is not correct', ['exception' => $e, 'type' => $type]);
-                       $typeString = (string)$type->value;
-               }
-
-               if (!isset($cache[$app][$key])) {
-                       throw new UnknownKeyException('unknown preference key');
-               }
-
-               $value = $cache[$app][$key];
-               $sensitive = $this->isSensitive($userId, $app, $key, null);
-               $this->decryptSensitiveValue($userId, $app, $key, $value);
-
-               return [
-                       'userId' => $userId,
-                       'app' => $app,
-                       'key' => $key,
-                       'value' => $value,
-                       'type' => $type->value,
-                       'lazy' => $lazy,
-                       'typeString' => $typeString,
-                       'sensitive' => $sensitive
-               ];
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @since 31.0.0
-        */
-       public function deletePreference(string $userId, string $app, string $key): void {
-               $this->assertParams($userId, $app, $key);
-               $qb = $this->connection->getQueryBuilder();
-               $qb->delete('preferences')
-                       ->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)))
-                       ->andWhere($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
-                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
-               $qb->executeStatement();
-
-               unset($this->lazyCache[$userId][$app][$key]);
-               unset($this->fastCache[$userId][$app][$key]);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @since 31.0.0
-        */
-       public function deleteKey(string $app, string $key): void {
-               $this->assertParams('', $app, $key, allowEmptyUser: true);
-               $qb = $this->connection->getQueryBuilder();
-               $qb->delete('preferences')
-                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)))
-                       ->andWhere($qb->expr()->eq('configkey', $qb->createNamedParameter($key)));
-               $qb->executeStatement();
-
-               $this->clearCacheAll();
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $app id of the app
-        *
-        * @since 31.0.0
-        */
-       public function deleteApp(string $app): void {
-               $this->assertParams('', $app, allowEmptyUser: true);
-               $qb = $this->connection->getQueryBuilder();
-               $qb->delete('preferences')
-                       ->where($qb->expr()->eq('appid', $qb->createNamedParameter($app)));
-               $qb->executeStatement();
-
-               $this->clearCacheAll();
-       }
-
-       public function deleteAllPreferences(string $userId): void {
-               $this->assertParams($userId, '', allowEmptyApp: true);
-               $qb = $this->connection->getQueryBuilder();
-               $qb->delete('preferences')
-                       ->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)));
-               $qb->executeStatement();
-
-               $this->clearCache($userId);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @param string $userId id of the user
-        * @param bool $reload set to TRUE to refill cache instantly after clearing it.
-        *
-        * @since 31.0.0
-        */
-       public function clearCache(string $userId, bool $reload = false): void {
-               $this->assertParams($userId, allowEmptyApp: true);
-               $this->lazyLoaded[$userId] = $this->fastLoaded[$userId] = false;
-               $this->lazyCache[$userId] = $this->fastCache[$userId] = $this->valueDetails[$userId] = [];
-
-               if (!$reload) {
-                       return;
-               }
-
-               $this->loadPreferencesAll($userId);
-       }
-
-       /**
-        * @inheritDoc
-        *
-        * @since 31.0.0
-        */
-       public function clearCacheAll(): void {
-               $this->lazyLoaded = $this->fastLoaded = [];
-               $this->lazyCache = $this->fastCache = $this->valueDetails = [];
-       }
-
-       /**
-        * For debug purpose.
-        * Returns the cached data.
-        *
-        * @return array
-        * @since 31.0.0
-        * @internal
-        */
-       public function statusCache(): array {
-               return [
-                       'fastLoaded' => $this->fastLoaded,
-                       'fastCache' => $this->fastCache,
-                       'lazyLoaded' => $this->lazyLoaded,
-                       'lazyCache' => $this->lazyCache,
-                       'valueDetails' => $this->valueDetails,
-               ];
-       }
-
-       /**
-        * @param int $needle bitflag to search
-        * @param int $flags all flags
-        *
-        * @return bool TRUE if bitflag $needle is set in $flags
-        */
-       private function isFlagged(int $needle, int $flags): bool {
-               return (($needle & $flags) !== 0);
-       }
-
-       /**
-        * Confirm the string set for app and key fit the database description
-        *
-        * @param string $userId
-        * @param string $app assert $app fit in database
-        * @param string $prefKey assert preference key fit in database
-        * @param bool $allowEmptyUser
-        * @param bool $allowEmptyApp $app can be empty string
-        * @param ValueType|null $valueType assert value type is only one type
-        */
-       private function assertParams(
-               string $userId = '',
-               string $app = '',
-               string $prefKey = '',
-               bool $allowEmptyUser = false,
-               bool $allowEmptyApp = false,
-               ?ValueType $valueType = null,
-       ): void {
-               if (!$allowEmptyUser && $userId === '') {
-                       throw new InvalidArgumentException('userId cannot be an empty string');
-               }
-               if (!$allowEmptyApp && $app === '') {
-                       throw new InvalidArgumentException('app cannot be an empty string');
-               }
-               if (strlen($userId) > self::USER_MAX_LENGTH) {
-                       throw new InvalidArgumentException('Value (' . $userId . ') for userId is too long (' . self::USER_MAX_LENGTH . ')');
-               }
-               if (strlen($app) > self::APP_MAX_LENGTH) {
-                       throw new InvalidArgumentException('Value (' . $app . ') for app is too long (' . self::APP_MAX_LENGTH . ')');
-               }
-               if (strlen($prefKey) > self::KEY_MAX_LENGTH) {
-                       throw new InvalidArgumentException('Value (' . $prefKey . ') for key is too long (' . self::KEY_MAX_LENGTH . ')');
-               }
-               if ($valueType !== null) {
-//                     $valueFlag = $valueType->value;
-//                     if (ValueType::tryFrom($valueFlag) === null) {
-//                             throw new InvalidArgumentException('Unknown value type');
-//                     }
-               }
-       }
-
-       private function loadPreferencesAll(string $userId): void {
-               $this->loadPreferences($userId, null);
-       }
-
-       /**
-        * Load normal preferences or preferences set as lazy loaded
-        *
-        * @param bool|null $lazy set to TRUE to load preferences set as lazy loaded, set to NULL to load all preferences
-        */
-       private function loadPreferences(string $userId, ?bool $lazy = false): void {
-               if ($this->isLoaded($userId, $lazy)) {
-                       return;
-               }
-
-               if (($lazy ?? true) !== false) { // if lazy is null or true, we debug log
-                       $this->logger->debug('The loading of lazy UserPreferences values have been requested', ['exception' => new \RuntimeException('ignorable exception')]);
-               }
-
-               $qb = $this->connection->getQueryBuilder();
-               $qb->from('preferences');
-               $qb->select('appid', 'configkey', 'configvalue', 'type', 'flags');
-               $qb->where($qb->expr()->eq('userid', $qb->createNamedParameter($userId)));
-
-               // we only need value from lazy when loadPreferences does not specify it
-               if ($lazy !== null) {
-                       $qb->andWhere($qb->expr()->eq('lazy', $qb->createNamedParameter($lazy ? 1 : 0, IQueryBuilder::PARAM_INT)));
-               } else {
-                       $qb->addSelect('lazy');
-               }
-
-               $result = $qb->executeQuery();
-               $rows = $result->fetchAll();
-               foreach ($rows as $row) {
-                       if (($row['lazy'] ?? ($lazy ?? 0) ? 1 : 0) === 1) {
-                               $this->lazyCache[$userId][$row['appid']][$row['configkey']] = $row['configvalue'] ?? '';
-                       } else {
-                               $this->fastCache[$userId][$row['appid']][$row['configkey']] = $row['configvalue'] ?? '';
-                       }
-                       $this->valueDetails[$userId][$row['appid']][$row['configkey']] = ['type' => ValueType::from((int)($row['type'] ?? 0)), 'flags' => (int) $row['flags']];
-               }
-               $result->closeCursor();
-               $this->setAsLoaded($userId, $lazy);
-       }
-
-       /**
-        * if $lazy is:
-        *  - false: will returns true if fast preferences are loaded
-        *  - true : will returns true if lazy preferences are loaded
-        *  - null : will returns true if both preferences are loaded
-        *
-        * @param string $userId
-        * @param bool $lazy
-        *
-        * @return bool
-        */
-       private function isLoaded(string $userId, ?bool $lazy): bool {
-               if ($lazy === null) {
-                       return ($this->lazyLoaded[$userId] ?? false) && ($this->fastLoaded[$userId] ?? false);
-               }
-
-               return $lazy ? $this->lazyLoaded[$userId] ?? false : $this->fastLoaded[$userId] ?? false;
-       }
-
-       /**
-        * if $lazy is:
-        * - false: set fast preferences as loaded
-        * - true : set lazy preferences as loaded
-        * - null : set both preferences as loaded
-        *
-        * @param string $userId
-        * @param bool $lazy
-        */
-       private function setAsLoaded(string $userId, ?bool $lazy): void {
-               if ($lazy === null) {
-                       $this->fastLoaded[$userId] = $this->lazyLoaded[$userId] = true;
-                       return;
-               }
-
-               // We also create empty entry to keep both fastLoaded/lazyLoaded synced
-               if ($lazy) {
-                       $this->lazyLoaded[$userId] = true;
-                       $this->fastLoaded[$userId] = $this->fastLoaded[$userId] ?? false;
-                       $this->fastCache[$userId] = $this->fastCache[$userId] ?? [];
-               } else {
-                       $this->fastLoaded[$userId] = true;
-                       $this->lazyLoaded[$userId] = $this->lazyLoaded[$userId] ?? false;
-                       $this->lazyCache[$userId] = $this->lazyCache[$userId] ?? [];
-               }
-       }
-
-       /**
-        * **Warning:** this will load all lazy values from the database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param bool $filtered TRUE to hide sensitive preference values. Value are replaced by {@see IConfig::SENSITIVE_VALUE}
-        *
-        * @return array<string, string|int|float|bool|array>
-        */
-       private function formatAppValues(string $userId, string $app, array $values, bool $filtered = false): array {
-               foreach ($values as $key => $value) {
-                       //$key = (string)$key;
-                       try {
-                               $type = $this->getValueType($userId, $app, (string)$key);
-                       } catch (UnknownKeyException) {
-                               continue;
-                       }
-
-                       if ($this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags'] ?? 0)) {
-                               if ($filtered) {
-                                       $value = IConfig::SENSITIVE_VALUE;
-                                       $type = ValueType::STRING;
-                               } else {
-                                       $this->decryptSensitiveValue($userId, $app, (string)$key, $value);
-                               }
-                       }
-
-                       $values[$key] = $this->convertTypedValue($value, $type);
-               }
-
-               return $values;
-       }
-
-       /**
-        * convert string value to the expected type
-        *
-        * @param string $value
-        * @param ValueType $type
-        *
-        * @return string|int|float|bool|array
-        */
-       private function convertTypedValue(string $value, ValueType $type): string|int|float|bool|array {
-               switch ($type) {
-                       case ValueType::INT:
-                               return (int)$value;
-                       case ValueType::FLOAT:
-                               return (float)$value;
-                       case ValueType::BOOL:
-                               return in_array(strtolower($value), ['1', 'true', 'yes', 'on']);
-                       case ValueType::ARRAY:
-                               try {
-                                       return json_decode($value, true, flags: JSON_THROW_ON_ERROR);
-                               } catch (JsonException) {
-                                       // ignoreable
-                               }
-                               break;
-               }
-               return $value;
-       }
-
-
-       private function decryptSensitiveValue(string $userId, string $app, string $key, string &$value): void {
-               if (!$this->isFlagged(self::FLAG_SENSITIVE, $this->valueDetails[$userId][$app][$key]['flags'] ?? 0)) {
-                       return;
-               }
-
-               if (!str_starts_with($value, self::ENCRYPTION_PREFIX)) {
-                       return;
-               }
-
-               try {
-                       $value = $this->crypto->decrypt(substr($value, self::ENCRYPTION_PREFIX_LENGTH));
-               } catch (\Exception $e) {
-                       $this->logger->warning('could not decrypt sensitive value', [
-                               'userId' => $userId,
-                               'app' => $app,
-                               'key' => $key,
-                               'value' => $value,
-                               'exception' => $e
-                       ]);
-               }
-       }
-}
diff --git a/lib/public/Config/Exceptions/IncorrectTypeException.php b/lib/public/Config/Exceptions/IncorrectTypeException.php
new file mode 100644 (file)
index 0000000..dafbbac
--- /dev/null
@@ -0,0 +1,17 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCP\Config\Exceptions;
+
+use Exception;
+
+/**
+ * @since 31.0.0
+ */
+class IncorrectTypeException extends Exception {
+}
diff --git a/lib/public/Config/Exceptions/TypeConflictException.php b/lib/public/Config/Exceptions/TypeConflictException.php
new file mode 100644 (file)
index 0000000..0b3c903
--- /dev/null
@@ -0,0 +1,17 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCP\Config\Exceptions;
+
+use Exception;
+
+/**
+ * @since 31.0.0
+ */
+class TypeConflictException extends Exception {
+}
diff --git a/lib/public/Config/Exceptions/UnknownKeyException.php b/lib/public/Config/Exceptions/UnknownKeyException.php
new file mode 100644 (file)
index 0000000..2150246
--- /dev/null
@@ -0,0 +1,17 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCP\Config\Exceptions;
+
+use Exception;
+
+/**
+ * @since 31.0.0
+ */
+class UnknownKeyException extends Exception {
+}
diff --git a/lib/public/Config/IUserPreferences.php b/lib/public/Config/IUserPreferences.php
new file mode 100644 (file)
index 0000000..18c7fdc
--- /dev/null
@@ -0,0 +1,694 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCP\Config;
+
+use Generator;
+use OCP\Config\Exceptions\IncorrectTypeException;
+use OCP\Config\Exceptions\UnknownKeyException;
+
+/**
+ * This class provides an easy way for apps to store user preferences in the
+ * database.
+ * Supports **lazy loading**
+ *
+ * ### What is lazy loading ?
+ * In order to avoid loading useless user preferences into memory for each request,
+ * only non-lazy values are now loaded.
+ *
+ * Once a value that is lazy is requested, all lazy values will be loaded.
+ *
+ * Similarly, some methods from this class are marked with a warning about ignoring
+ * lazy loading. Use them wisely and only on parts of the code that are called
+ * during specific requests or actions to avoid loading the lazy values all the time.
+ *
+ * @since 31.0.0
+ */
+
+interface IUserPreferences {
+       /** @since 31.0.0 */
+       public const FLAG_SENSITIVE = 1;   // value is sensitive
+       /** @since 31.0.0 */
+       public const FLAG_INDEXED = 2;    // value should be indexed
+
+       /**
+        * Get list of all userIds with preferences stored in database.
+        * If $appId is specified, will only limit the search to this value
+        *
+        * **WARNING:** ignore any cache and get data directly from database.
+        *
+        * @param string $appId optional id of app
+        *
+        * @return list<string> list of userIds
+        * @since 31.0.0
+        */
+       public function getUserIds(string $appId = ''): array;
+
+       /**
+        * Get list of all apps that have at least one preference
+        * value related to $userId stored in database
+        *
+        * **WARNING:** ignore lazy filtering, all user preferences are loaded from database
+        *
+        * @param string $userId id of the user
+        *
+        * @return list<string> list of app ids
+        * @since 31.0.0
+        */
+       public function getApps(string $userId): array;
+
+       /**
+        * Returns all keys stored in database, related to user+app.
+        * Please note that the values are not returned.
+        *
+        * **WARNING:** ignore lazy filtering, all user preferences are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        *
+        * @return list<string> list of stored preference keys
+        * @since 31.0.0
+        */
+       public function getKeys(string $userId, string $app): array;
+
+       /**
+        * Check if a key exists in the list of stored preference values.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return bool TRUE if key exists
+        * @since 31.0.0
+        */
+       public function hasKey(string $userId, string $app, string $key, ?bool $lazy = false): bool;
+
+       /**
+        * best way to see if a value is set as sensitive (not displayed in report)
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy search within lazy loaded preferences
+        *
+        * @return bool TRUE if value is sensitive
+        * @throws UnknownKeyException if preference key is not known
+        * @since 31.0.0
+        */
+       public function isSensitive(string $userId, string $app, string $key, ?bool $lazy = false): bool;
+
+       /**
+        * best way to see if a value is set as indexed (so it can be search)
+        *
+        * @see self::searchUsersByValueString()
+        * @see self::searchUsersByValueInt()
+        * @see self::searchUsersByValueBool()
+        * @see self::searchUsersByValues()
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy search within lazy loaded preferences
+        *
+        * @return bool TRUE if value is sensitive
+        * @throws UnknownKeyException if preference key is not known
+        * @since 31.0.0
+        */
+       public function isIndexed(string $userId, string $app, string $key, ?bool $lazy = false): bool;
+
+       /**
+        * Returns if the preference key stored in database is lazy loaded
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @return bool TRUE if preference is lazy loaded
+        * @throws UnknownKeyException if preference key is not known
+        * @see IUserPreferences for details about lazy loading
+        * @since 31.0.0
+        */
+       public function isLazy(string $userId, string $app, string $key): bool;
+
+       /**
+        * List all preference values from an app with preference key starting with $key.
+        * Returns an array with preference key as key, stored value as value.
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $prefix preference keys prefix to search, can be empty.
+        * @param bool $filtered filter sensitive preference values
+        *
+        * @return array<string, string|int|float|bool|array> [key => value]
+        * @since 31.0.0
+        */
+       public function getValues(string $userId, string $app, string $prefix = '', bool $filtered = false): array;
+
+       /**
+        * List all preference values of a user.
+        * Returns an array with preference key as key, stored value as value.
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param bool $filtered filter sensitive preference values
+        *
+        * @return array<string, string|int|float|bool|array> [key => value]
+        * @since 31.0.0
+        */
+       public function getAllValues(string $userId, bool $filtered = false): array;
+
+       /**
+        * List all apps storing a specific preference key and its stored value.
+        * Returns an array with appId as key, stored value as value.
+        *
+        * @param string $userId id of the user
+        * @param string $key preference key
+        * @param bool $lazy search within lazy loaded preferences
+        * @param ValueType|null $typedAs enforce type for the returned values
+        *
+        * @return array<string, string|int|float|bool|array> [appId => value]
+        * @since 31.0.0
+        */
+       public function getValuesByApps(string $userId, string $key, bool $lazy = false, ?ValueType $typedAs = null): array;
+
+       /**
+        * List all users storing a specific preference key and its stored value.
+        * Returns an array with userId as key, stored value as value.
+        *
+        * **WARNING:** no caching, generate a fresh request
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param ValueType|null $typedAs enforce type for the returned values
+        * @param array|null $userIds limit the search to a list of user ids
+        *
+        * @return array<string, string|int|float|bool|array> [userId => value]
+        * @since 31.0.0
+        */
+       public function getValuesByUsers(string $app, string $key, ?ValueType $typedAs = null, ?array $userIds = null): array;
+
+       /**
+        * List all users storing a specific preference key/value pair.
+        * Returns a list of user ids.
+        *
+        * **WARNING:** no caching, generate a fresh request
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $caseInsensitive non-case-sensitive search, only works if $value is a string
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueString(string $app, string $key, string $value, bool $caseInsensitive = false): Generator;
+
+       /**
+        * List all users storing a specific preference key/value pair.
+        * Returns a list of user ids.
+        *
+        * **WARNING:** no caching, generate a fresh request
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $value preference value
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueInt(string $app, string $key, int $value): Generator;
+
+       /**
+        * List all users storing a specific preference key/value pair.
+        * Returns a list of user ids.
+        *
+        * **WARNING:** no caching, generate a fresh request
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $values list of possible preference values
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValues(string $app, string $key, array $values): Generator;
+
+       /**
+        * List all users storing a specific preference key/value pair.
+        * Returns a list of user ids.
+        *
+        * **WARNING:** no caching, generate a fresh request
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $value preference value
+        *
+        * @return Generator<string>
+        * @since 31.0.0
+        */
+       public function searchUsersByValueBool(string $app, string $key, bool $value): Generator;
+
+       /**
+        * Get user preference assigned to a preference key.
+        * If preference key is not found in database, default value is returned.
+        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return string stored preference value or $default if not set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see getValueInt()
+        * @see getValueFloat()
+        * @see getValueBool()
+        * @see getValueArray()
+        */
+       public function getValueString(string $userId, string $app, string $key, string $default = '', bool $lazy = false): string;
+
+       /**
+        * Get preference value assigned to a preference key.
+        * If preference key is not found in database, default value is returned.
+        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return int stored preference value or $default if not set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see getValueString()
+        * @see getValueFloat()
+        * @see getValueBool()
+        * @see getValueArray()
+        */
+       public function getValueInt(string $userId, string $app, string $key, int $default = 0, bool $lazy = false): int;
+
+       /**
+        * Get preference value assigned to a preference key.
+        * If preference key is not found in database, default value is returned.
+        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param float $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return float stored preference value or $default if not set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see getValueString()
+        * @see getValueInt()
+        * @see getValueBool()
+        * @see getValueArray()
+        */
+       public function getValueFloat(string $userId, string $app, string $key, float $default = 0, bool $lazy = false): float;
+
+       /**
+        * Get preference value assigned to a preference key.
+        * If preference key is not found in database, default value is returned.
+        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $default default value
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return bool stored preference value or $default if not set in database
+        * @since 31.0.0
+        * @see IUserPrefences for explanation about lazy loading
+        * @see getValueString()
+        * @see getValueInt()
+        * @see getValueFloat()
+        * @see getValueArray()
+        */
+       public function getValueBool(string $userId, string $app, string $key, bool $default = false, bool $lazy = false): bool;
+
+       /**
+        * Get preference value assigned to a preference key.
+        * If preference key is not found in database, default value is returned.
+        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $default default value`
+        * @param bool $lazy search within lazy loaded preferences
+        *
+        * @return array stored preference value or $default if not set in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see getValueString()
+        * @see getValueInt()
+        * @see getValueFloat()
+        * @see getValueBool()
+        */
+       public function getValueArray(string $userId, string $app, string $key, array $default = [], bool $lazy = false): array;
+
+       /**
+        * returns the type of preference value
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *              unless lazy is set to false
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool|null $lazy
+        *
+        * @return ValueType type of the value
+        * @throws UnknownKeyException if preference key is not known
+        * @throws IncorrectTypeException if preferences value type is not known
+        * @since 31.0.0
+        */
+       public function getValueType(string $userId, string $app, string $key, ?bool $lazy = null): ValueType;
+
+       /**
+        * returns a bitflag related to preference value
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *              unless lazy is set to false
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy lazy loading
+        *
+        * @return int a bitflag in relation to the preference value
+        * @throws UnknownKeyException if preference key is not known
+        * @throws IncorrectTypeException if preferences value type is not known
+        * @since 31.0.0
+        */
+       public function getValueFlags(string $userId, string $app, string $key, bool $lazy = false): int;
+
+       /**
+        * Store a preference key and its value in database
+        *
+        * If preference key is already known with the exact same preference value, the database is not updated.
+        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
+        *
+        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param string $value preference value
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueInt()
+        * @see setValueFloat()
+        * @see setValueBool()
+        * @see setValueArray()
+        */
+       public function setValueString(string $userId, string $app, string $key, string $value, bool $lazy = false, int $flags = 0): bool;
+
+       /**
+        * Store a preference key and its value in database
+        *
+        * When handling huge value around and/or above 2,147,483,647, a debug log will be generated
+        * on 64bits system, as php int type reach its limit (and throw an exception) on 32bits when using huge numbers.
+        *
+        * When using huge numbers, it is advised to use {@see \OCP\Util::numericToNumber()} and {@see setValueString()}
+        *
+        * If preference key is already known with the exact same preference value, the database is not updated.
+        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
+        *
+        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param int $value preference value
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueString()
+        * @see setValueFloat()
+        * @see setValueBool()
+        * @see setValueArray()
+        */
+       public function setValueInt(string $userId, string $app, string $key, int $value, bool $lazy = false, int $flags = 0): bool;
+
+       /**
+        * Store a preference key and its value in database.
+        *
+        * If preference key is already known with the exact same preference value, the database is not updated.
+        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
+        *
+        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param float $value preference value
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueString()
+        * @see setValueInt()
+        * @see setValueBool()
+        * @see setValueArray()
+        */
+       public function setValueFloat(string $userId, string $app, string $key, float $value, bool $lazy = false, int $flags = 0): bool;
+
+       /**
+        * Store a preference key and its value in database
+        *
+        * If preference key is already known with the exact same preference value, the database is not updated.
+        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
+        *
+        * If preference value was previously stored as lazy loaded, status cannot be altered without using {@see deleteKey()} first
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $value preference value
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueString()
+        * @see setValueInt()
+        * @see setValueFloat()
+        * @see setValueArray()
+        */
+       public function setValueBool(string $userId, string $app, string $key, bool $value, bool $lazy = false): bool;
+
+       /**
+        * Store a preference key and its value in database
+        *
+        * If preference key is already known with the exact same preference value, the database is not updated.
+        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
+        *
+        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param array $value preference value
+        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
+        * @param bool $lazy set preference as lazy loaded
+        *
+        * @return bool TRUE if value was different, therefor updated in database
+        * @since 31.0.0
+        * @see IUserPreferences for explanation about lazy loading
+        * @see setValueString()
+        * @see setValueInt()
+        * @see setValueFloat()
+        * @see setValueBool()
+        */
+       public function setValueArray(string $userId, string $app, string $key, array $value, bool $lazy = false, int $flags = 0): bool;
+
+       /**
+        * switch sensitive status of a preference value
+        *
+        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
+        *
+        * @return bool TRUE if database update were necessary
+        * @since 31.0.0
+        */
+       public function updateSensitive(string $userId, string $app, string $key, bool $sensitive): bool;
+
+       /**
+        * switch sensitive loading status of a preference key for all users
+        *
+        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
+        *
+        * @since 31.0.0
+        */
+       public function updateGlobalSensitive(string $app, string $key, bool $sensitive): void;
+
+
+       /**
+        * switch indexed status of a preference value
+        *
+        *  **WARNING:** ignore lazy filtering, all preference values are loaded from database
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $indexed TRUE to set as indexed, FALSE to unset
+        *
+        * @return bool TRUE if database update were necessary
+        * @since 31.0.0
+        */
+       public function updateIndexed(string $userId, string $app, string $key, bool $indexed): bool;
+
+       /**
+        * switch sensitive loading status of a preference key for all users
+        *
+        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $indexed TRUE to set as indexed, FALSE to unset
+        * @since 31.0.0
+        */
+       public function updateGlobalIndexed(string $app, string $key, bool $indexed): void;
+
+       /**
+        * switch lazy loading status of a preference value
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
+        *
+        * @return bool TRUE if database update was necessary
+        * @since 31.0.0
+        */
+       public function updateLazy(string $userId, string $app, string $key, bool $lazy): bool;
+
+       /**
+        * switch lazy loading status of a preference key for all users
+        *
+        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
+        * @since 31.0.0
+        */
+       public function updateGlobalLazy(string $app, string $key, bool $lazy): void;
+
+       /**
+        * returns an array contains details about a preference value
+        *
+        * ```
+        * [
+        *   "app" => "myapp",
+        *   "key" => "mykey",
+        *   "value" => "its_value",
+        *   "lazy" => false,
+        *   "type" => 4,
+        *   "typeString" => "string",
+        *   'sensitive' => true
+        * ]
+        * ```
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @return array
+        * @throws UnknownKeyException if preference key is not known in database
+        * @since 31.0.0
+        */
+       public function getDetails(string $userId, string $app, string $key): array;
+
+       /**
+        * Delete single preference key from database.
+        *
+        * @param string $userId id of the user
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @since 31.0.0
+        */
+       public function deletePreference(string $userId, string $app, string $key): void;
+
+       /**
+        * Delete preference values from all users linked to a specific preference keys
+        *
+        * @param string $app id of the app
+        * @param string $key preference key
+        *
+        * @since 31.0.0
+        */
+       public function deleteKey(string $app, string $key): void;
+
+       /**
+        * delete all preference keys linked to an app
+        *
+        * @param string $app id of the app
+        * @since 31.0.0
+        */
+       public function deleteApp(string $app): void;
+
+       /**
+        * delete all preference keys linked to a user
+        *
+        * @param string $userId id of the user
+        * @since 31.0.0
+        */
+       public function deleteAllPreferences(string $userId): void;
+
+       /**
+        * Clear the cache for a single user
+        *
+        * The cache will be rebuilt only the next time a user preference is requested.
+        *
+        * @param string $userId id of the user
+        * @param bool $reload set to TRUE to refill cache instantly after clearing it
+        *
+        * @since 31.0.0
+        */
+       public function clearCache(string $userId, bool $reload = false): void;
+
+       /**
+        * Clear the cache for all users.
+        * The cache will be rebuilt only the next time a user preference is requested.
+        *
+        * @since 31.0.0
+        */
+       public function clearCacheAll(): void;
+}
diff --git a/lib/public/Config/ValueType.php b/lib/public/Config/ValueType.php
new file mode 100644 (file)
index 0000000..caf5100
--- /dev/null
@@ -0,0 +1,78 @@
+<?php
+
+declare(strict_types=1);
+/**
+ * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ */
+
+namespace OCP\Config;
+
+use OCP\Config\Exceptions\IncorrectTypeException;
+use UnhandledMatchError;
+
+/**
+ * Listing of available value type for typed config value
+ *
+ * @since 31.0.0
+ */
+enum ValueType: int {
+       /** @since 31.0.0 */
+       case MIXED = 0;
+       /** @since 31.0.0 */
+       case STRING = 1;
+       /** @since 31.0.0 */
+       case INT = 2;
+       /** @since 31.0.0 */
+       case FLOAT = 3;
+       /** @since 31.0.0 */
+       case BOOL = 4;
+       /** @since 31.0.0 */
+       case ARRAY = 5;
+
+       /**
+        * get ValueType from string
+        *
+        * @param string $definition
+        *
+        * @return self
+        * @throws IncorrectTypeException
+        * @since 31.0.0
+        */
+       public static function fromStringDefinition(string $definition): self {
+               try {
+                       return match ($definition) {
+                               'mixed' => self::MIXED,
+                               'string' => self::STRING,
+                               'int' => self::INT,
+                               'float' => self::FLOAT,
+                               'bool' => self::BOOL,
+                               'array' => self::ARRAY
+                       };
+               } catch (\UnhandledMatchError) {
+                       throw new IncorrectTypeException('unknown string definition');
+               }
+       }
+
+       /**
+        * get string definition for current enum value
+        *
+        * @return string
+        * @throws IncorrectTypeException
+        * @since 31.0.0
+        */
+       public function getDefinition(): string {
+               try {
+                       return match ($this) {
+                               self::MIXED => 'mixed',
+                               self::STRING => 'string',
+                               self::INT => 'int',
+                               self::FLOAT => 'float',
+                               self::BOOL => 'bool',
+                               self::ARRAY => 'array',
+                       };
+               } catch (UnhandledMatchError) {
+                       throw new IncorrectTypeException('unknown type definition ' . $this->value);
+               }
+       }
+}
index 3bc64c5e13337b0691ce1eb1e695a870bfc1b1e0..f434f838d45c3d80436203fbfeda04a26b014896 100644 (file)
@@ -245,7 +245,7 @@ interface IConfig {
         * @param string $appName the app to get the user for
         * @param string $key the key to get the user for
         * @param string $value the value to get the user for
-        * @return list<string> of user IDs
+        * @return array<string> of user IDs
         * @since 31.0.0 return type of `list<string>`
         * @since 8.0.0
         */
diff --git a/lib/public/UserPreferences/Exceptions/IncorrectTypeException.php b/lib/public/UserPreferences/Exceptions/IncorrectTypeException.php
deleted file mode 100644 (file)
index 5c8f83d..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences\Exceptions;
-
-/**
- * @since 31.0.0
- */
-class IncorrectTypeException extends UserPreferencesException {
-}
diff --git a/lib/public/UserPreferences/Exceptions/TypeConflictException.php b/lib/public/UserPreferences/Exceptions/TypeConflictException.php
deleted file mode 100644 (file)
index b67113f..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences\Exceptions;
-
-/**
- * @since 31.0.0
- */
-class TypeConflictException extends UserPreferencesException {
-}
diff --git a/lib/public/UserPreferences/Exceptions/UnknownKeyException.php b/lib/public/UserPreferences/Exceptions/UnknownKeyException.php
deleted file mode 100644 (file)
index 3df6668..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences\Exceptions;
-
-/**
- * @since 31.0.0
- */
-class UnknownKeyException extends UserPreferencesException {
-}
diff --git a/lib/public/UserPreferences/Exceptions/UserPreferencesException.php b/lib/public/UserPreferences/Exceptions/UserPreferencesException.php
deleted file mode 100644 (file)
index 664181d..0000000
+++ /dev/null
@@ -1,17 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences\Exceptions;
-
-use Exception;
-
-/**
- * @since 31.0.0
- */
-class UserPreferencesException extends Exception {
-}
diff --git a/lib/public/UserPreferences/IUserPreferences.php b/lib/public/UserPreferences/IUserPreferences.php
deleted file mode 100644 (file)
index f2635c0..0000000
+++ /dev/null
@@ -1,676 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences;
-
-use OCP\UserPreferences\Exceptions\IncorrectTypeException;
-use OCP\UserPreferences\Exceptions\UnknownKeyException;
-
-/**
- * @since 31.0.0
- */
-interface IUserPreferences {
-       public const FLAG_SENSITIVE = 1;   // value is sensitive
-       public const FLAG_INDEXED = 2;    // value should be indexed
-
-       /**
-        * Get list of all userIds with preferences stored in database.
-        * If $appId is specified, will only limit the search to this value
-        *
-        * **WARNING:** ignore any cache and get data directly from database.
-        *
-        * @param string $appId optional id of app
-        *
-        * @return list<string> list of userIds
-        * @since 31.0.0
-        */
-       public function getUserIds(string $appId = ''): array;
-
-       /**
-        * Get list of all apps that have at least one preference
-        * value related to $userId stored in database
-        *
-        * **WARNING:** ignore lazy filtering, all user preferences are loaded from database
-        *
-        * @param string $userId id of the user
-        *
-        * @return list<string> list of app ids
-        * @since 31.0.0
-        */
-       public function getApps(string $userId): array;
-
-       /**
-        * Returns all keys stored in database, related to user+app.
-        * Please note that the values are not returned.
-        *
-        * **WARNING:** ignore lazy filtering, all user preferences are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        *
-        * @return list<string> list of stored preference keys
-        * @since 31.0.0
-        */
-       public function getKeys(string $userId, string $app): array;
-
-       /**
-        * Check if a key exists in the list of stored preference values.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return bool TRUE if key exists
-        * @since 31.0.0
-        */
-       public function hasKey(string $userId, string $app, string $key, ?bool $lazy = false): bool;
-
-       /**
-        * best way to see if a value is set as sensitive (not displayed in report)
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy search within lazy loaded preferences
-        *
-        * @return bool TRUE if value is sensitive
-        * @throws UnknownKeyException if preference key is not known
-        * @since 31.0.0
-        */
-       public function isSensitive(string $userId, string $app, string $key, ?bool $lazy = false): bool;
-
-       /**
-        * best way to see if a value is set as indexed (so it can be search)
-        *
-        * @see self::searchUsersByValueString()
-        * @see self::searchUsersByValueInt()
-        * @see self::searchUsersByValueBool()
-        * @see self::searchUsersByValues()
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy search within lazy loaded preferences
-        *
-        * @return bool TRUE if value is sensitive
-        * @throws UnknownKeyException if preference key is not known
-        * @since 31.0.0
-        */
-       public function isIndexed(string $userId, string $app, string $key, ?bool $lazy = false): bool;
-
-       /**
-        * Returns if the preference key stored in database is lazy loaded
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @return bool TRUE if preference is lazy loaded
-        * @throws UnknownKeyException if preference key is not known
-        * @see IUserPreferences for details about lazy loading
-        * @since 31.0.0
-        */
-       public function isLazy(string $userId, string $app, string $key): bool;
-
-       /**
-        * List all preference values from an app with preference key starting with $key.
-        * Returns an array with preference key as key, stored value as value.
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $prefix preference keys prefix to search, can be empty.
-        * @param bool $filtered filter sensitive preference values
-        *
-        * @return array<string, string|int|float|bool|array> [key => value]
-        * @since 31.0.0
-        */
-       public function getValues(string $userId, string $app, string $prefix = '', bool $filtered = false): array;
-
-       /**
-        * List all preference values of a user.
-        * Returns an array with preference key as key, stored value as value.
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param bool $filtered filter sensitive preference values
-        *
-        * @return array<string, string|int|float|bool|array> [key => value]
-        * @since 31.0.0
-        */
-       public function getAllValues(string $userId, bool $filtered = false): array;
-
-       /**
-        * List all apps storing a specific preference key and its stored value.
-        * Returns an array with appId as key, stored value as value.
-        *
-        * @param string $userId id of the user
-        * @param string $key preference key
-        * @param bool $lazy search within lazy loaded preferences
-        * @param ValueType|null $typedAs enforce type for the returned values
-        *
-        * @return array<string, string|int|float|bool|array> [appId => value]
-        * @since 31.0.0
-        */
-       public function getValuesByApps(string $userId, string $key, bool $lazy = false, ?ValueType $typedAs = null): array;
-
-       /**
-        * List all users storing a specific preference key and its stored value.
-        * Returns an array with userId as key, stored value as value.
-        *
-        * **WARNING:** no caching, generate a fresh request
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param ValueType|null $typedAs enforce type for the returned values
-        * @param array|null $userIds limit the search to a list of user ids
-        *
-        * @return array<string, string|int|float|bool|array> [userId => value]
-        * @since 31.0.0
-        */
-       public function getValuesByUsers(string $app, string $key, ?ValueType $typedAs = null, ?array $userIds = null): array;
-
-       /**
-        * List all users storing a specific preference key/value pair.
-        * Returns a list of user ids.
-        *
-        * **WARNING:** no caching, generate a fresh request
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $caseInsensitive non-case-sensitive search, only works if $value is a string
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueString(string $app, string $key, string $value, bool $caseInsensitive = false): array;
-
-       /**
-        * List all users storing a specific preference key/value pair.
-        * Returns a list of user ids.
-        *
-        * **WARNING:** no caching, generate a fresh request
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $value preference value
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueInt(string $app, string $key, int $value): array;
-
-       /**
-        * List all users storing a specific preference key/value pair.
-        * Returns a list of user ids.
-        *
-        * **WARNING:** no caching, generate a fresh request
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $values list of possible preference values
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValues(string $app, string $key, array $values): array;
-
-       /**
-        * List all users storing a specific preference key/value pair.
-        * Returns a list of user ids.
-        *
-        * **WARNING:** no caching, generate a fresh request
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $value preference value
-        *
-        * @return list<string>
-        * @since 31.0.0
-        */
-       public function searchUsersByValueBool(string $app, string $key, bool $value): array;
-
-       /**
-        * Get user preference assigned to a preference key.
-        * If preference key is not found in database, default value is returned.
-        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return string stored preference value or $default if not set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see getValueInt()
-        * @see getValueFloat()
-        * @see getValueBool()
-        * @see getValueArray()
-        */
-       public function getValueString(string $userId, string $app, string $key, string $default = '', bool $lazy = false): string;
-
-       /**
-        * Get preference value assigned to a preference key.
-        * If preference key is not found in database, default value is returned.
-        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return int stored preference value or $default if not set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see getValueString()
-        * @see getValueFloat()
-        * @see getValueBool()
-        * @see getValueArray()
-        */
-       public function getValueInt(string $userId, string $app, string $key, int $default = 0, bool $lazy = false): int;
-
-       /**
-        * Get preference value assigned to a preference key.
-        * If preference key is not found in database, default value is returned.
-        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param float $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return float stored preference value or $default if not set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see getValueString()
-        * @see getValueInt()
-        * @see getValueBool()
-        * @see getValueArray()
-        */
-       public function getValueFloat(string $userId, string $app, string $key, float $default = 0, bool $lazy = false): float;
-
-       /**
-        * Get preference value assigned to a preference key.
-        * If preference key is not found in database, default value is returned.
-        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $default default value
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return bool stored preference value or $default if not set in database
-        * @since 31.0.0
-        * @see IUserPrefences for explanation about lazy loading
-        * @see getValueString()
-        * @see getValueInt()
-        * @see getValueFloat()
-        * @see getValueArray()
-        */
-       public function getValueBool(string $userId, string $app, string $key, bool $default = false, bool $lazy = false): bool;
-
-       /**
-        * Get preference value assigned to a preference key.
-        * If preference key is not found in database, default value is returned.
-        * If preference key is set as lazy loaded, the $lazy argument needs to be set to TRUE.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $default default value`
-        * @param bool $lazy search within lazy loaded preferences
-        *
-        * @return array stored preference value or $default if not set in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see getValueString()
-        * @see getValueInt()
-        * @see getValueFloat()
-        * @see getValueBool()
-        */
-       public function getValueArray(string $userId, string $app, string $key, array $default = [], bool $lazy = false): array;
-
-       /**
-        * returns the type of preference value
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *              unless lazy is set to false
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool|null $lazy
-        *
-        * @return ValueType type of the value
-        * @throws UnknownKeyException if preference key is not known
-        * @throws IncorrectTypeException if preferences value type is not known
-        * @since 31.0.0
-        */
-       public function getValueType(string $userId, string $app, string $key, ?bool $lazy = null): ValueType;
-
-       /**
-        * returns a bitflag related to preference value
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *              unless lazy is set to false
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy lazy loading
-        *
-        * @return int a bitflag in relation to the preference value
-        * @throws UnknownKeyException if preference key is not known
-        * @throws IncorrectTypeException if preferences value type is not known
-        * @since 31.0.0
-        */
-       public function getValueFlags(string $userId, string $app, string $key, bool $lazy = false): int;
-
-       /**
-        * Store a preference key and its value in database
-        *
-        * If preference key is already known with the exact same preference value, the database is not updated.
-        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
-        *
-        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param string $value preference value
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueInt()
-        * @see setValueFloat()
-        * @see setValueBool()
-        * @see setValueArray()
-        */
-       public function setValueString(string $userId, string $app, string $key, string $value, bool $lazy = false, int $flags = 0): bool;
-
-       /**
-        * Store a preference key and its value in database
-        *
-        * When handling huge value around and/or above 2,147,483,647, a debug log will be generated
-        * on 64bits system, as php int type reach its limit (and throw an exception) on 32bits when using huge numbers.
-        *
-        * When using huge numbers, it is advised to use {@see \OCP\Util::numericToNumber()} and {@see setValueString()}
-        *
-        * If preference key is already known with the exact same preference value, the database is not updated.
-        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
-        *
-        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param int $value preference value
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueString()
-        * @see setValueFloat()
-        * @see setValueBool()
-        * @see setValueArray()
-        */
-       public function setValueInt(string $userId, string $app, string $key, int $value, bool $lazy = false, int $flags = 0): bool;
-
-       /**
-        * Store a preference key and its value in database.
-        *
-        * If preference key is already known with the exact same preference value, the database is not updated.
-        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
-        *
-        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param float $value preference value
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueString()
-        * @see setValueInt()
-        * @see setValueBool()
-        * @see setValueArray()
-        */
-       public function setValueFloat(string $userId, string $app, string $key, float $value, bool $lazy = false, int $flags = 0): bool;
-
-       /**
-        * Store a preference key and its value in database
-        *
-        * If preference key is already known with the exact same preference value, the database is not updated.
-        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
-        *
-        * If preference value was previously stored as lazy loaded, status cannot be altered without using {@see deleteKey()} first
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $value preference value
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueString()
-        * @see setValueInt()
-        * @see setValueFloat()
-        * @see setValueArray()
-        */
-       public function setValueBool(string $userId, string $app, string $key, bool $value, bool $lazy = false): bool;
-
-       /**
-        * Store a preference key and its value in database
-        *
-        * If preference key is already known with the exact same preference value, the database is not updated.
-        * If preference key is not supposed to be read during the boot of the cloud, it is advised to set it as lazy loaded.
-        *
-        * If preference value was previously stored as sensitive or lazy loaded, status cannot be altered without using {@see deleteKey()} first
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param array $value preference value
-        * @param bool $sensitive if TRUE value will be hidden when listing preference values.
-        * @param bool $lazy set preference as lazy loaded
-        *
-        * @return bool TRUE if value was different, therefor updated in database
-        * @since 31.0.0
-        * @see IUserPreferences for explanation about lazy loading
-        * @see setValueString()
-        * @see setValueInt()
-        * @see setValueFloat()
-        * @see setValueBool()
-        */
-       public function setValueArray(string $userId, string $app, string $key, array $value, bool $lazy = false, int $flags = 0): bool;
-
-       /**
-        * switch sensitive status of a preference value
-        *
-        * **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
-        *
-        * @return bool TRUE if database update were necessary
-        * @since 31.0.0
-        */
-       public function updateSensitive(string $userId, string $app, string $key, bool $sensitive): bool;
-
-       /**
-        * switch sensitive loading status of a preference key for all users
-        *
-        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $sensitive TRUE to set as sensitive, FALSE to unset
-        *
-        * @since 31.0.0
-        */
-       public function updateGlobalSensitive(string $app, string $key, bool $sensitive): void;
-
-
-       /**
-        * switch indexed status of a preference value
-        *
-        *  **WARNING:** ignore lazy filtering, all preference values are loaded from database
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $indexed TRUE to set as indexed, FALSE to unset
-        *
-        * @return bool TRUE if database update were necessary
-        * @since 31.0.0
-        */
-       public function updateIndexed(string $userId, string $app, string $key, bool $indexed): bool;
-
-       /**
-        * switch sensitive loading status of a preference key for all users
-        *
-        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $indexed TRUE to set as indexed, FALSE to unset
-        * @since 31.0.0
-        */
-       public function updateGlobalIndexed(string $app, string $key, bool $indexed): void;
-
-       /**
-        * switch lazy loading status of a preference value
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
-        *
-        * @return bool TRUE if database update was necessary
-        * @since 31.0.0
-        */
-       public function updateLazy(string $userId, string $app, string $key, bool $lazy): bool;
-
-       /**
-        * switch lazy loading status of a preference key for all users
-        *
-        * **Warning:** heavy on resources, MUST only be used on occ command or migrations
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        * @param bool $lazy TRUE to set as lazy loaded, FALSE to unset
-        * @since 31.0.0
-        */
-       public function updateGlobalLazy(string $app, string $key, bool $lazy): void;
-
-       /**
-        * returns an array contains details about a preference value
-        *
-        * ```
-        * [
-        *   "app" => "myapp",
-        *   "key" => "mykey",
-        *   "value" => "its_value",
-        *   "lazy" => false,
-        *   "type" => 4,
-        *   "typeString" => "string",
-        *   'sensitive' => true
-        * ]
-        * ```
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @return array
-        * @throws UnknownKeyException if preference key is not known in database
-        * @since 31.0.0
-        */
-       public function getDetails(string $userId, string $app, string $key): array;
-
-       /**
-        * Delete single preference key from database.
-        *
-        * @param string $userId id of the user
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @since 31.0.0
-        */
-       public function deletePreference(string $userId, string $app, string $key): void;
-
-       /**
-        * Delete preference values from all users linked to a specific preference keys
-        *
-        * @param string $app id of the app
-        * @param string $key preference key
-        *
-        * @since 31.0.0
-        */
-       public function deleteKey(string $app, string $key): void;
-
-       /**
-        * delete all preference keys linked to an app
-        *
-        * @param string $app id of the app
-        * @since 31.0.0
-        */
-       public function deleteApp(string $app): void;
-
-       /**
-        * delete all preference keys linked to a user
-        *
-        * @param string $userId id of the user
-        * @since 31.0.0
-        */
-       public function deleteAllPreferences(string $userId): void;
-
-       /**
-        * Clear the cache for a single user
-        *
-        * The cache will be rebuilt only the next time a user preference is requested.
-        *
-        * @param string $userId id of the user
-        * @param bool $reload set to TRUE to refill cache instantly after clearing it
-        *
-        * @since 31.0.0
-        */
-       public function clearCache(string $userId, bool $reload = false): void;
-
-       /**
-        * Clear the cache for all users.
-        * The cache will be rebuilt only the next time a user preference is requested.
-        *
-        * @since 31.0.0
-        */
-       public function clearCacheAll(): void;
-}
diff --git a/lib/public/UserPreferences/ValueType.php b/lib/public/UserPreferences/ValueType.php
deleted file mode 100644 (file)
index 280dc5a..0000000
+++ /dev/null
@@ -1,79 +0,0 @@
-<?php
-
-declare(strict_types=1);
-/**
- * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-or-later
- */
-
-namespace OCP\UserPreferences;
-
-use OCP\UserPreferences\Exceptions\IncorrectTypeException;
-use UnhandledMatchError;
-
-/**
- * Listing of available value type for user preferences
- *
- * @see IUserPreferences
- * @since 31.0.0
- */
-enum ValueType: int {
-       /** @since 31.0.0 */
-       case MIXED = 0;
-       /** @since 31.0.0 */
-       case STRING = 1;
-       /** @since 31.0.0 */
-       case INT = 2;
-       /** @since 31.0.0 */
-       case FLOAT = 3;
-       /** @since 31.0.0 */
-       case BOOL = 4;
-       /** @since 31.0.0 */
-       case ARRAY = 5;
-
-       /**
-        * get ValueType from string
-        *
-        * @param string $definition
-        *
-        * @return self
-        * @throws IncorrectTypeException
-        * @since 31.0.0
-        */
-       public static function fromStringDefinition(string $definition): self {
-               try {
-                       return match ($definition) {
-                               'mixed' => self::MIXED,
-                               'string' => self::STRING,
-                               'int' => self::INT,
-                               'float' => self::FLOAT,
-                               'bool' => self::BOOL,
-                               'array' => self::ARRAY
-                       };
-               } catch (\UnhandledMatchError ) {
-                       throw new IncorrectTypeException('unknown string definition');
-               }
-       }
-
-       /**
-        * get string definition for current enum value
-        *
-        * @return string
-        * @throws IncorrectTypeException
-        * @since 31.0.0
-        */
-       public function getDefinition(): string {
-               try {
-                       return match ($this) {
-                               self::MIXED => 'mixed',
-                               self::STRING => 'string',
-                               self::INT => 'int',
-                               self::FLOAT => 'float',
-                               self::BOOL => 'bool',
-                               self::ARRAY => 'array',
-                       };
-               } catch (UnhandledMatchError) {
-                       throw new IncorrectTypeException('unknown type definition ' . $this->value);
-               }
-       }
-}
index bf3353646f5f1873bac32c657f75b45a4ee3dd38..19e6d6161bfc82a67aee861a695787f886a40d10 100644 (file)
@@ -7,13 +7,13 @@ declare(strict_types=1);
  */
 namespace lib;
 
-use OC\UserPreferences;
+use OC\Config\UserPreferences;
+use OCP\Config\Exceptions\TypeConflictException;
+use OCP\Config\Exceptions\UnknownKeyException;
+use OCP\Config\IUserPreferences;
+use OCP\Config\ValueType;
 use OCP\IDBConnection;
 use OCP\Security\ICrypto;
-use OCP\UserPreferences\Exceptions\TypeConflictException;
-use OCP\UserPreferences\Exceptions\UnknownKeyException;
-use OCP\UserPreferences\IUserPreferences;
-use OCP\UserPreferences\ValueType;
 use Psr\Log\LoggerInterface;
 use Test\TestCase;
 
@@ -275,7 +275,7 @@ class UserPreferencesTest extends TestCase {
         * @return IUserPreferences
         */
        private function generateUserPreferences(array $preLoading = []): IUserPreferences {
-               $preferences = new \OC\UserPreferences(
+               $preferences = new \OC\Config\UserPreferences(
                        $this->connection,
                        $this->logger,
                        $this->crypto,
@@ -775,7 +775,7 @@ class UserPreferencesTest extends TestCase {
                array $result,
        ): void {
                $preferences = $this->generateUserPreferences();
-               $this->assertEqualsCanonicalizing($result, $preferences->searchUsersByValueString($app, $key, $value, $ci));
+               $this->assertEqualsCanonicalizing($result, iterator_to_array($preferences->searchUsersByValueString($app, $key, $value, $ci)));
        }
 
        public function providerSearchValuesByValueInt(): array {
@@ -796,7 +796,7 @@ class UserPreferencesTest extends TestCase {
                array $result,
        ): void {
                $preferences = $this->generateUserPreferences();
-               $this->assertEqualsCanonicalizing($result, $preferences->searchUsersByValueInt($app, $key, $value));
+               $this->assertEqualsCanonicalizing($result, iterator_to_array($preferences->searchUsersByValueInt($app, $key, $value)));
        }
 
        public function providerSearchValuesByValues(): array {
@@ -816,7 +816,7 @@ class UserPreferencesTest extends TestCase {
                array $result,
        ): void {
                $preferences = $this->generateUserPreferences();
-               $this->assertEqualsCanonicalizing($result, $preferences->searchUsersByValues($app, $key, $values));
+               $this->assertEqualsCanonicalizing($result, iterator_to_array($preferences->searchUsersByValues($app, $key, $values)));
        }
 
        public function providerSearchValuesByValueBool(): array {
@@ -836,7 +836,7 @@ class UserPreferencesTest extends TestCase {
                array $result,
        ): void {
                $preferences = $this->generateUserPreferences();
-               $this->assertEqualsCanonicalizing($result, $preferences->searchUsersByValueBool($app, $key, $value));
+               $this->assertEqualsCanonicalizing($result, iterator_to_array($preferences->searchUsersByValueBool($app, $key, $value)));
        }
 
        public function providerGetValueMixed(): array {