refactor: standardize protocol
Signed-off-by: Sebastian Krupinski <krupinski01@gmail.com>
This commit is contained in:
@@ -14,357 +14,622 @@ use KTXC\Http\Response\JsonResponse;
|
|||||||
use KTXC\SessionIdentity;
|
use KTXC\SessionIdentity;
|
||||||
use KTXC\SessionTenant;
|
use KTXC\SessionTenant;
|
||||||
use KTXF\Controller\ControllerAbstract;
|
use KTXF\Controller\ControllerAbstract;
|
||||||
use KTXF\People\Collection\ICollectionBase;
|
|
||||||
use KTXF\Resource\Selector\SourceSelector;
|
use KTXF\Resource\Selector\SourceSelector;
|
||||||
use KTXF\Routing\Attributes\AuthenticatedRoute;
|
use KTXF\Routing\Attributes\AuthenticatedRoute;
|
||||||
use KTXM\PeopleManager\Manager;
|
use KTXM\PeopleManager\Manager;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
class DefaultController extends ControllerAbstract {
|
class DefaultController extends ControllerAbstract {
|
||||||
|
|
||||||
|
private const ERR_MISSING_PROVIDER = 'Missing parameter: provider';
|
||||||
|
private const ERR_MISSING_IDENTIFIER = 'Missing parameter: identifier';
|
||||||
|
private const ERR_MISSING_SERVICE = 'Missing parameter: service';
|
||||||
|
private const ERR_MISSING_COLLECTION = 'Missing parameter: collection';
|
||||||
|
private const ERR_MISSING_DATA = 'Missing parameter: data';
|
||||||
|
private const ERR_MISSING_SOURCES = 'Missing parameter: sources';
|
||||||
|
private const ERR_MISSING_IDENTIFIERS = 'Missing parameter: identifiers';
|
||||||
|
private const ERR_INVALID_OPERATION = 'Invalid operation: ';
|
||||||
|
private const ERR_INVALID_PROVIDER = 'Invalid parameter: provider must be a string';
|
||||||
|
private const ERR_INVALID_SERVICE = 'Invalid parameter: service must be a string';
|
||||||
|
private const ERR_INVALID_IDENTIFIER = 'Invalid parameter: identifier must be a string';
|
||||||
|
private const ERR_INVALID_COLLECTION = 'Invalid parameter: collection must be a string or integer';
|
||||||
|
private const ERR_INVALID_SOURCES = 'Invalid parameter: sources must be an array';
|
||||||
|
private const ERR_INVALID_IDENTIFIERS = 'Invalid parameter: identifiers must be an array';
|
||||||
|
private const ERR_INVALID_DATA = 'Invalid parameter: data must be an array';
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly SessionTenant $tenantIdentity,
|
private readonly SessionTenant $tenantIdentity,
|
||||||
private readonly SessionIdentity $userIdentity,
|
private readonly SessionIdentity $userIdentity,
|
||||||
private Manager $peopleManager,
|
private readonly Manager $manager,
|
||||||
|
private readonly LoggerInterface $logger,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve list of available providers
|
* Main API endpoint for mail operations
|
||||||
*
|
*
|
||||||
* @return JsonResponse
|
* Single operation:
|
||||||
*/
|
* {
|
||||||
#[AuthenticatedRoute('/v1', name: 'peoplemanager.v1', methods: ['POST'])]
|
* "version": 1,
|
||||||
|
* "transaction": "tx-1",
|
||||||
|
* "operation": "entity.create",
|
||||||
|
* "data": {...}
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* @return JsonResponse
|
||||||
|
*/
|
||||||
|
#[AuthenticatedRoute('/v1', name: 'people.manager.v1', methods: ['POST'])]
|
||||||
|
public function index(
|
||||||
|
int $version,
|
||||||
|
string $transaction,
|
||||||
|
string|null $operation = null,
|
||||||
|
array|null $data = null,
|
||||||
|
string|null $user = null
|
||||||
|
): JsonResponse {
|
||||||
|
|
||||||
public function index(int $version, string $transaction, string $operation, array $data = [], string|null $user = null): JsonResponse {
|
// authorize request
|
||||||
|
$tenantId = $this->tenantIdentity->identifier();
|
||||||
|
$userId = $this->userIdentity->identifier();
|
||||||
|
|
||||||
// authorize request
|
try {
|
||||||
$tenantId = $this->tenantIdentity->identifier();
|
|
||||||
$userId = $this->userIdentity->identifier();
|
|
||||||
|
|
||||||
try {
|
if ($operation !== null) {
|
||||||
$data = $this->process($tenantId, $userId, $operation, $data);
|
$result = $this->processOperation($tenantId, $userId, $operation, $data ?? [], []);
|
||||||
return new JsonResponse([
|
return new JsonResponse([
|
||||||
'version' => $version,
|
'version' => $version,
|
||||||
'transaction' => $transaction,
|
'transaction' => $transaction,
|
||||||
'operation' => $operation,
|
'operation' => $operation,
|
||||||
'status' => 'success',
|
'status' => 'success',
|
||||||
'data' => $data,
|
'data' => $result
|
||||||
], JsonResponse::HTTP_OK);
|
], JsonResponse::HTTP_OK);
|
||||||
} catch (\Throwable $t) {
|
}
|
||||||
return new JsonResponse([
|
|
||||||
'version' => $version,
|
|
||||||
'transaction' => $transaction,
|
|
||||||
'operation' => $operation,
|
|
||||||
'status' => 'error',
|
|
||||||
'data' => [
|
|
||||||
'code' => $t->getCode(),
|
|
||||||
'message' => $t->getMessage(),
|
|
||||||
]
|
|
||||||
], JsonResponse::HTTP_INTERNAL_SERVER_ERROR);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function process(string $tenantId, string $userId, string $operation, array $data): mixed {
|
throw new InvalidArgumentException('Operation must be provided');
|
||||||
|
|
||||||
return match ($operation) {
|
} catch (Throwable $t) {
|
||||||
'provider.list' => $this->providerList($tenantId, $userId, $data),
|
$this->logger->error('Error processing request', ['exception' => $t]);
|
||||||
'provider.extant' => $this->providerExtant($tenantId, $userId, $data),
|
return new JsonResponse([
|
||||||
'service.list' => $this->serviceList($tenantId, $userId, $data),
|
'version' => $version,
|
||||||
'service.extant' => $this->serviceExtant($tenantId, $userId, $data),
|
'transaction' => $transaction,
|
||||||
'service.fetch' => $this->serviceFetch($tenantId, $userId, $data),
|
'operation' => $operation,
|
||||||
'collection.list' => $this->collectionList($tenantId, $userId, $data),
|
'status' => 'error',
|
||||||
'collection.extant' => $this->collectionExtant($tenantId, $userId, $data),
|
'data' => [
|
||||||
'collection.fetch' => $this->collectionFetch($tenantId, $userId, $data),
|
'code' => $t->getCode(),
|
||||||
'collection.create' => $this->collectionCreate($tenantId, $userId, $data),
|
'message' => $t->getMessage()
|
||||||
'collection.modify' => $this->collectionModify($tenantId, $userId, $data),
|
]
|
||||||
'collection.destroy' => $this->collectionDestroy($tenantId, $userId, $data),
|
], JsonResponse::HTTP_INTERNAL_SERVER_ERROR);
|
||||||
'entity.list' => $this->entityList($tenantId, $userId, $data),
|
}
|
||||||
'entity.delta' => $this->entityDelta($tenantId, $userId, $data),
|
}
|
||||||
'entity.extant' => $this->entityExtant($tenantId, $userId, $data),
|
|
||||||
'entity.fetch' => $this->entityFetch($tenantId, $userId, $data),
|
|
||||||
'entity.create' => $this->entityCreate($tenantId, $userId, $data),
|
|
||||||
'entity.modify' => $this->entityModify($tenantId, $userId, $data),
|
|
||||||
'entity.destroy' => $this->entityDestroy($tenantId, $userId, $data),
|
|
||||||
default => throw new InvalidArgumentException("Invalid operation: $operation"),
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
/**
|
||||||
|
* Process a single operation
|
||||||
|
*/
|
||||||
|
private function processOperation(string $tenantId, string $userId, string $operation, array $data): mixed {
|
||||||
|
return match ($operation) {
|
||||||
|
// Provider operations
|
||||||
|
'provider.list' => $this->providerList($tenantId, $userId, $data),
|
||||||
|
'provider.fetch' => $this->providerFetch($tenantId, $userId, $data),
|
||||||
|
'provider.extant' => $this->providerExtant($tenantId, $userId, $data),
|
||||||
|
|
||||||
// ==================== Provider Operations ====================
|
// Service operations
|
||||||
|
'service.list' => $this->serviceList($tenantId, $userId, $data),
|
||||||
|
'service.fetch' => $this->serviceFetch($tenantId, $userId, $data),
|
||||||
|
'service.extant' => $this->serviceExtant($tenantId, $userId, $data),
|
||||||
|
'service.create' => $this->serviceCreate($tenantId, $userId, $data),
|
||||||
|
'service.update' => $this->serviceUpdate($tenantId, $userId, $data),
|
||||||
|
'service.delete' => $this->serviceDelete($tenantId, $userId, $data),
|
||||||
|
'service.test' => $this->serviceTest($tenantId, $userId, $data),
|
||||||
|
|
||||||
private function providerList(string $tenantId, string $userId, array $data): mixed {
|
// Collection operations
|
||||||
|
'collection.list' => $this->collectionList($tenantId, $userId, $data),
|
||||||
|
'collection.fetch' => $this->collectionFetch($tenantId, $userId, $data),
|
||||||
|
'collection.extant' => $this->collectionExtant($tenantId, $userId, $data),
|
||||||
|
'collection.create' => $this->collectionCreate($tenantId, $userId, $data),
|
||||||
|
'collection.update' => $this->collectionUpdate($tenantId, $userId, $data),
|
||||||
|
'collection.delete' => $this->collectionDelete($tenantId, $userId, $data),
|
||||||
|
|
||||||
$sources = null;
|
// Entity operations
|
||||||
|
'entity.list' => $this->entityList($tenantId, $userId, $data),
|
||||||
|
'entity.fetch' => $this->entityFetch($tenantId, $userId, $data),
|
||||||
|
'entity.extant' => $this->entityExtant($tenantId, $userId, $data),
|
||||||
|
'entity.create' => $this->entityCreate($tenantId, $userId, $data),
|
||||||
|
'entity.update' => $this->entityUpdate($tenantId, $userId, $data),
|
||||||
|
'entity.delete' => $this->entityDelete($tenantId, $userId, $data),
|
||||||
|
'entity.delta' => $this->entityDelta($tenantId, $userId, $data),
|
||||||
|
'entity.move' => throw new InvalidArgumentException('Operation not implemented: ' . $operation),
|
||||||
|
'entity.copy' => throw new InvalidArgumentException('Operation not implemented: ' . $operation),
|
||||||
|
|
||||||
|
default => throw new InvalidArgumentException(self::ERR_INVALID_OPERATION . $operation)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== Provider Operations ====================
|
||||||
|
|
||||||
|
private function providerList(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
|
||||||
|
$sources = null;
|
||||||
if (isset($data['sources']) && is_array($data['sources'])) {
|
if (isset($data['sources']) && is_array($data['sources'])) {
|
||||||
$sources = new SourceSelector();
|
$sources = new SourceSelector();
|
||||||
$sources->jsonDeserialize($data['sources']);
|
$sources->jsonDeserialize($data['sources']);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->peopleManager->providerList($tenantId, $userId, $sources);
|
return $this->manager->providerList($tenantId, $userId, $sources);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function providerExtant(string $tenantId, string $userId, array $data): mixed {
|
private function providerFetch(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
|
||||||
if (!isset($data['sources']) || !is_array($data['sources'])) {
|
if (!isset($data['identifier'])) {
|
||||||
throw new InvalidArgumentException('Invalid sources selector provided');
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->manager->providerFetch($tenantId, $userId, $data['identifier']);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function providerExtant(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
|
||||||
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
|
}
|
||||||
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
}
|
}
|
||||||
$sources = new SourceSelector();
|
$sources = new SourceSelector();
|
||||||
$sources->jsonDeserialize($data['sources']);
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
|
||||||
return $this->peopleManager->providerExtant($tenantId, $userId, $sources);
|
return $this->manager->providerExtant($tenantId, $userId, $sources);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== Service Operations =====================
|
|
||||||
|
|
||||||
private function serviceList(string $tenantId, string $userId, array $data): mixed {
|
// ==================== Service Operations =====================
|
||||||
|
|
||||||
$sources = null;
|
private function serviceList(string $tenantId, string $userId, array $data): mixed {
|
||||||
if (isset($data['sources']) && is_array($data['sources'])) {
|
|
||||||
$sources = new SourceSelector();
|
$sources = null;
|
||||||
$sources->jsonDeserialize($data['sources']);
|
if (isset($data['sources']) && is_array($data['sources'])) {
|
||||||
|
$sources = new SourceSelector();
|
||||||
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->manager->serviceList($tenantId, $userId, $sources);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private function serviceFetch(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->peopleManager->serviceList($tenantId, $userId, $sources);
|
return $this->manager->serviceFetch($tenantId, $userId, $data['provider'], $data['identifier']);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
private function serviceExtant(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
|
||||||
private function serviceExtant(string $tenantId, string $userId, array $data): mixed {
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
if (!isset($data['sources']) || !is_array($data['sources'])) {
|
}
|
||||||
throw new InvalidArgumentException('Invalid sources selector provided');
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
}
|
}
|
||||||
$sources = new SourceSelector();
|
$sources = new SourceSelector();
|
||||||
$sources->jsonDeserialize($data['sources']);
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
|
||||||
return $this->peopleManager->serviceExtant($tenantId, $userId, $sources);
|
return $this->manager->serviceExtant($tenantId, $userId, $sources);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
private function serviceCreate(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['data'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_DATA);
|
||||||
|
}
|
||||||
|
if (!is_array($data['data'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
private function serviceFetch(string $tenantId, string $userId, array $data): mixed {
|
return $this->manager->serviceCreate(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['data']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
private function serviceUpdate(string $tenantId, string $userId, array $data): mixed {
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
if (!isset($data['provider'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
if (!isset($data['identifier']) || !is_string($data['identifier'])) {
|
}
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
if (!is_string($data['provider'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!isset($data['data'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_DATA);
|
||||||
|
}
|
||||||
|
if (!is_array($data['data'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
return $this->peopleManager->serviceFetch($tenantId, $userId, $data['provider'], $data['identifier']);
|
return $this->manager->serviceUpdate(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['identifier'],
|
||||||
|
$data['data']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
private function serviceDelete(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
|
}
|
||||||
|
|
||||||
private function collectionList(string $tenantId, string $userId, array $data): mixed {
|
return $this->manager->serviceDelete(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['identifier']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
$sources = null;
|
private function serviceTest(string $tenantId, string $userId, array $data): mixed {
|
||||||
if (isset($data['sources']) && is_array($data['sources'])) {
|
|
||||||
$sources = new SourceSelector();
|
|
||||||
$sources->jsonDeserialize($data['sources']);
|
|
||||||
}
|
|
||||||
$filter = $data['filter'] ?? null;
|
|
||||||
$sort = $data['sort'] ?? null;
|
|
||||||
// retrieve collections
|
|
||||||
return $this->peopleManager->collectionList($tenantId, $userId, $sources, $filter, $sort);
|
|
||||||
|
|
||||||
}
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
|
||||||
private function collectionExtant(string $tenantId, string $userId, array $data = []): mixed {
|
if (!isset($data['identifier']) && !isset($data['location']) && !isset($data['identity'])) {
|
||||||
|
throw new InvalidArgumentException('Either a service identifier or location and identity must be provided for service test');
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['sources']) || !is_array($data['sources'])) {
|
return $this->manager->serviceTest(
|
||||||
throw new InvalidArgumentException('Invalid sources selector provided');
|
$tenantId,
|
||||||
}
|
$userId,
|
||||||
$sources = new SourceSelector();
|
$data['provider'],
|
||||||
$sources->jsonDeserialize($data['sources']);
|
$data['identifier'] ?? null,
|
||||||
// retrieve collection status
|
$data['location'] ?? null,
|
||||||
return $this->peopleManager->collectionExtant($tenantId, $userId, $sources);
|
$data['identity'] ?? null,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
// ==================== Collection Operations ====================
|
||||||
|
|
||||||
private function collectionFetch(string $tenantId, string $userId, array $data = []): mixed {
|
private function collectionList(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
$sources = null;
|
||||||
|
if (isset($data['sources']) && is_array($data['sources'])) {
|
||||||
|
$sources = new SourceSelector();
|
||||||
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
$filter = $data['filter'] ?? null;
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
$sort = $data['sort'] ?? null;
|
||||||
}
|
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
|
||||||
}
|
|
||||||
if (!isset($data['identifier'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
|
||||||
}
|
|
||||||
// retrieve collection
|
|
||||||
return $this->peopleManager->collectionFetch($tenantId, $userId, $data['provider'], $data['service'], $data['identifier']);
|
|
||||||
|
|
||||||
}
|
return $this->manager->collectionList($tenantId, $userId, $sources, $filter, $sort);
|
||||||
|
}
|
||||||
|
|
||||||
private function collectionCreate(string $tenantId, string $userId, array $data = []): mixed {
|
private function collectionExtant(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
|
}
|
||||||
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
$sources = new SourceSelector();
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
$sources->jsonDeserialize($data['sources']);
|
||||||
}
|
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
|
||||||
}
|
|
||||||
if (!isset($data['data'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid collection data provided');
|
|
||||||
}
|
|
||||||
$options = $data['options'] ?? [];
|
|
||||||
// create collection
|
|
||||||
return $this->peopleManager->collectionCreate($tenantId, $userId, $data['provider'], $data['service'], $data['data'], $options);
|
|
||||||
|
|
||||||
}
|
return $this->manager->collectionExtant($tenantId, $userId, $sources);
|
||||||
|
}
|
||||||
|
|
||||||
private function collectionModify(string $tenantId, string $userId, array $data = []): mixed {
|
private function collectionFetch(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SERVICE);
|
||||||
|
}
|
||||||
|
if (!is_string($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier']) && !is_int($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
return $this->manager->collectionFetch(
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
$tenantId,
|
||||||
}
|
$userId,
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
$data['provider'],
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
$data['service'],
|
||||||
}
|
$data['identifier']
|
||||||
if (!isset($data['identifier'])) {
|
);
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
}
|
||||||
}
|
|
||||||
if (!isset($data['data'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid collection data provided');
|
|
||||||
}
|
|
||||||
// modify collection
|
|
||||||
return $this->peopleManager->collectionModify($tenantId, $userId, $data['provider'], $data['service'], $data['identifier'], $data['data']);
|
|
||||||
|
|
||||||
}
|
private function collectionCreate(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SERVICE);
|
||||||
|
}
|
||||||
|
if (!is_string($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
|
}
|
||||||
|
if (isset($data['collection']) && !is_string($data['collection']) && !is_int($data['collection'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
|
}
|
||||||
|
if (!isset($data['properties'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_DATA);
|
||||||
|
}
|
||||||
|
if (!is_array($data['properties'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
private function collectionDestroy(string $tenantId, string $userId, array $data = []): mixed {
|
return $this->manager->collectionCreate(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['service'],
|
||||||
|
$data['collection'] ?? null,
|
||||||
|
$data['properties']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
private function collectionUpdate(string $tenantId, string $userId, array $data): mixed {
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
if (!isset($data['provider'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
}
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
if (!is_string($data['provider'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
if (!isset($data['identifier'])) {
|
}
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
if (!isset($data['service'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_MISSING_SERVICE);
|
||||||
// destroy collection
|
}
|
||||||
return ['success' => $this->peopleManager->collectionDestroy($tenantId, $userId, $data['provider'], $data['service'], $data['identifier'])];
|
if (!is_string($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier']) && !is_int($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
|
}
|
||||||
|
if (!isset($data['properties'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_DATA);
|
||||||
|
}
|
||||||
|
if (!is_array($data['properties'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
return $this->manager->collectionUpdate(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['service'],
|
||||||
|
$data['identifier'],
|
||||||
|
$data['properties']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private function entityList(string $tenantId, string $userId, array $data = []): mixed {
|
private function collectionDelete(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['provider'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
|
}
|
||||||
|
if (!isset($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SERVICE);
|
||||||
|
}
|
||||||
|
if (!is_string($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIER);
|
||||||
|
}
|
||||||
|
if (!is_string($data['identifier']) && !is_int($data['identifier'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->manager->collectionDelete(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['service'],
|
||||||
|
$data['identifier'],
|
||||||
|
$data['options'] ?? []
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== Entity Operations ====================
|
||||||
|
|
||||||
|
private function entityList(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
|
}
|
||||||
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
|
}
|
||||||
|
|
||||||
|
$sources = new SourceSelector();
|
||||||
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
|
||||||
$sources = null;
|
|
||||||
if (isset($data['sources']) && is_array($data['sources'])) {
|
|
||||||
$sources = new SourceSelector();
|
|
||||||
$sources->jsonDeserialize($data['sources']);
|
|
||||||
}
|
|
||||||
$filter = $data['filter'] ?? null;
|
$filter = $data['filter'] ?? null;
|
||||||
$sort = $data['sort'] ?? null;
|
$sort = $data['sort'] ?? null;
|
||||||
$range = $data['range'] ?? null;
|
$range = $data['range'] ?? null;
|
||||||
// retrieve entities
|
|
||||||
return $this->peopleManager->entityList($tenantId, $userId, $sources, $filter, $sort, $range);
|
|
||||||
|
|
||||||
}
|
return $this->manager->entityList($tenantId, $userId, $sources, $filter, $sort, $range);
|
||||||
|
|
||||||
private function entityDelta(string $tenantId, string $userId, array $data = []): mixed {
|
}
|
||||||
|
|
||||||
if (!isset($data['sources']) || !is_array($data['sources'])) {
|
private function entityFetch(string $tenantId, string $userId, array $data): mixed {
|
||||||
throw new InvalidArgumentException('Invalid sources selector provided');
|
if (!isset($data['provider'])) {
|
||||||
}
|
throw new InvalidArgumentException(self::ERR_MISSING_PROVIDER);
|
||||||
$sources = new SourceSelector();
|
}
|
||||||
$sources->jsonDeserialize($data['sources']);
|
if (!is_string($data['provider'])) {
|
||||||
// retrieve entity delta
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
return $this->peopleManager->entityDelta($tenantId, $userId, $sources);
|
}
|
||||||
|
if (!isset($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SERVICE);
|
||||||
|
}
|
||||||
|
if (!is_string($data['service'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
|
}
|
||||||
|
if (!isset($data['collection'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_COLLECTION);
|
||||||
|
}
|
||||||
|
if (!is_string($data['collection']) && !is_int($data['collection'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
|
}
|
||||||
|
if (!isset($data['identifiers'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_IDENTIFIERS);
|
||||||
|
}
|
||||||
|
if (!is_array($data['identifiers'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIERS);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
return $this->manager->entityFetch(
|
||||||
|
$tenantId,
|
||||||
|
$userId,
|
||||||
|
$data['provider'],
|
||||||
|
$data['service'],
|
||||||
|
$data['collection'],
|
||||||
|
$data['identifiers']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private function entityExtant(string $tenantId, string $userId, array $data = []): mixed {
|
private function entityExtant(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
|
}
|
||||||
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
|
}
|
||||||
|
|
||||||
if (!isset($data['sources']) || !is_array($data['sources'])) {
|
$sources = new SourceSelector();
|
||||||
throw new InvalidArgumentException('Invalid sources selector provided');
|
$sources->jsonDeserialize($data['sources']);
|
||||||
}
|
|
||||||
$sources = new SourceSelector();
|
|
||||||
$sources->jsonDeserialize($data['sources']);
|
|
||||||
// retrieve entity status
|
|
||||||
return $this->peopleManager->entityExtant($tenantId, $userId, $sources);
|
|
||||||
|
|
||||||
}
|
return $this->manager->entityExtant($tenantId, $userId, $sources);
|
||||||
|
}
|
||||||
private function entityFetch(string $tenantId, string $userId, array $data = []): mixed {
|
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
|
||||||
}
|
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
|
||||||
}
|
|
||||||
if (!isset($data['collection'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
|
||||||
}
|
|
||||||
if (!isset($data['identifiers']) || !is_array($data['identifiers'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid entity identifiers provided');
|
|
||||||
}
|
|
||||||
// retrieve entities
|
|
||||||
return $this->peopleManager->entityFetch($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['identifiers']);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private function entityCreate(string $tenantId, string $userId, array $data = []): mixed {
|
private function entityCreate(string $tenantId, string $userId, array $data = []): mixed {
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
}
|
}
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
if (!isset($data['service']) || !is_string($data['service'])) {
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
}
|
}
|
||||||
if (!isset($data['collection'])) {
|
if (!isset($data['collection'])) {
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
}
|
|
||||||
if (!isset($data['data']) || !is_array($data['data'])) {
|
|
||||||
throw new InvalidArgumentException('Invalid entity data provided');
|
|
||||||
}
|
}
|
||||||
|
$properties = $data['properties'] ?? $data['data'] ?? null;
|
||||||
|
if (!is_array($properties)) {
|
||||||
|
throw new InvalidArgumentException('Invalid parameter: properties must be an array');
|
||||||
|
}
|
||||||
$options = $data['options'] ?? [];
|
$options = $data['options'] ?? [];
|
||||||
// create entity
|
|
||||||
return $this->peopleManager->entityCreate($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['data'], $options);
|
return $this->manager->entityCreate($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $properties, $options);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function entityModify(string $tenantId, string $userId, array $data = []): mixed {
|
private function entityUpdate(string $tenantId, string $userId, array $data = []): mixed {
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
}
|
}
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
if (!isset($data['service']) || !is_string($data['service'])) {
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
}
|
}
|
||||||
if (!isset($data['collection'])) {
|
if (!isset($data['collection'])) {
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
}
|
}
|
||||||
if (!isset($data['identifier'])) {
|
if (!isset($data['identifier'])) {
|
||||||
throw new InvalidArgumentException('Invalid entity identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
}
|
}
|
||||||
if (!isset($data['data']) || !is_array($data['data'])) {
|
$properties = $data['properties'] ?? $data['data'] ?? null;
|
||||||
throw new InvalidArgumentException('Invalid entity data provided');
|
if (!is_array($properties)) {
|
||||||
}
|
throw new InvalidArgumentException('Invalid parameter: properties must be an array');
|
||||||
// modify entity
|
}
|
||||||
return $this->peopleManager->entityModify($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['identifier'], $data['data']);
|
|
||||||
|
return $this->manager->entityUpdate($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['identifier'], $properties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function entityDestroy(string $tenantId, string $userId, array $data = []): mixed {
|
private function entityDelete(string $tenantId, string $userId, array $data = []): mixed {
|
||||||
|
|
||||||
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
if (!isset($data['provider']) || !is_string($data['provider'])) {
|
||||||
throw new InvalidArgumentException('Invalid provider identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_PROVIDER);
|
||||||
}
|
}
|
||||||
if (!isset($data['service']) || !is_string($data['service'])) {
|
if (!isset($data['service']) || !is_string($data['service'])) {
|
||||||
throw new InvalidArgumentException('Invalid service identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_SERVICE);
|
||||||
}
|
}
|
||||||
if (!isset($data['collection'])) {
|
if (!isset($data['collection'])) {
|
||||||
throw new InvalidArgumentException('Invalid collection identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_COLLECTION);
|
||||||
}
|
}
|
||||||
if (!isset($data['identifier'])) {
|
if (!isset($data['identifier'])) {
|
||||||
throw new InvalidArgumentException('Invalid entity identifier provided');
|
throw new InvalidArgumentException(self::ERR_INVALID_IDENTIFIER);
|
||||||
}
|
}
|
||||||
// destroy entity
|
|
||||||
return ['success' => $this->peopleManager->entityDestroy($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['identifier'])];
|
return $this->manager->entityDelete($tenantId, $userId, $data['provider'], $data['service'], $data['collection'], $data['identifier']);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function entityDelta(string $tenantId, string $userId, array $data): mixed {
|
||||||
|
if (!isset($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_MISSING_SOURCES);
|
||||||
|
}
|
||||||
|
if (!is_array($data['sources'])) {
|
||||||
|
throw new InvalidArgumentException(self::ERR_INVALID_SOURCES);
|
||||||
|
}
|
||||||
|
|
||||||
|
$sources = new SourceSelector();
|
||||||
|
$sources->jsonDeserialize($data['sources']);
|
||||||
|
|
||||||
|
return $this->manager->entityDelta($tenantId, $userId, $sources);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
616
lib/Manager.php
616
lib/Manager.php
@@ -6,12 +6,16 @@ namespace KTXM\PeopleManager;
|
|||||||
|
|
||||||
use InvalidArgumentException;
|
use InvalidArgumentException;
|
||||||
use KTXC\Resource\ProviderManager;
|
use KTXC\Resource\ProviderManager;
|
||||||
use KTXF\People\Collection\ICollectionBase;
|
use KTXF\People\Collection\CollectionBaseInterface;
|
||||||
use KTXF\People\Entity\IEntityBase;
|
use KTXF\People\Collection\CollectionMutableInterface;
|
||||||
use KTXF\People\Provider\IProviderBase;
|
use KTXF\People\Entity\EntityBaseInterface;
|
||||||
use KTXF\People\Service\IServiceBase;
|
use KTXF\People\Entity\EntityMutableInterface;
|
||||||
use KTXF\People\Service\IServiceCollectionMutable;
|
use KTXF\People\Provider\ProviderBaseInterface;
|
||||||
use KTXF\People\Service\IServiceEntityMutable;
|
use KTXF\People\Provider\ProviderServiceMutateInterface;
|
||||||
|
use KTXF\People\Service\ServiceBaseInterface;
|
||||||
|
use KTXF\People\Service\ServiceCollectionMutableInterface;
|
||||||
|
use KTXF\People\Service\ServiceEntityMutableInterface;
|
||||||
|
use KTXF\Resource\Filter\IFilter;
|
||||||
use KTXF\Resource\Provider\ProviderInterface;
|
use KTXF\Resource\Provider\ProviderInterface;
|
||||||
use KTXF\Resource\Range\RangeAnchorType;
|
use KTXF\Resource\Range\RangeAnchorType;
|
||||||
use KTXF\Resource\Range\RangeType;
|
use KTXF\Resource\Range\RangeType;
|
||||||
@@ -19,6 +23,7 @@ use KTXF\Resource\Selector\CollectionSelector;
|
|||||||
use KTXF\Resource\Selector\EntitySelector;
|
use KTXF\Resource\Selector\EntitySelector;
|
||||||
use KTXF\Resource\Selector\ServiceSelector;
|
use KTXF\Resource\Selector\ServiceSelector;
|
||||||
use KTXF\Resource\Selector\SourceSelector;
|
use KTXF\Resource\Selector\SourceSelector;
|
||||||
|
use KTXF\Resource\Sort\ISort;
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
|
|
||||||
class Manager {
|
class Manager {
|
||||||
@@ -33,13 +38,32 @@ class Manager {
|
|||||||
*
|
*
|
||||||
* @param SourceSelector|null $sources collection of provider identifiers
|
* @param SourceSelector|null $sources collection of provider identifiers
|
||||||
*
|
*
|
||||||
* @return array<string,IProviderBase> collection of available providers e.g. ['provider1' => IProvider, 'provider2' => IProvider]
|
* @return array<string,ProviderBaseInterface> collection of available providers e.g. ['provider1' => IProvider, 'provider2' => IProvider]
|
||||||
*/
|
*/
|
||||||
public function providerList(string $tenantId, string $userId, ?SourceSelector $sources = null): array {
|
public function providerList(string $tenantId, string $userId, ?SourceSelector $sources = null): array {
|
||||||
// determine filter from sources
|
// determine filter from sources
|
||||||
$filter = ($sources !== null && $sources->identifiers() !== []) ? $sources->identifiers() : null;
|
$filter = ($sources !== null && $sources->identifiers() !== []) ? $sources->identifiers() : null;
|
||||||
// retrieve providers from provider manager
|
// retrieve providers from provider manager
|
||||||
return $this->providerManager->providers(ProviderInterface::TYPE_PEOPLE, $filter);
|
return $this->providerManager->providers(ProviderBaseInterface::TYPE_PEOPLE, $filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve specific provider for specific user
|
||||||
|
*
|
||||||
|
* @param string $tenantId tenant identifier
|
||||||
|
* @param string $userId user identifier
|
||||||
|
* @param string $provider provider identifier
|
||||||
|
*
|
||||||
|
* @return ProviderBaseInterface
|
||||||
|
* @throws InvalidArgumentException
|
||||||
|
*/
|
||||||
|
public function providerFetch(string $tenantId, string $userId, string $provider): ProviderBaseInterface {
|
||||||
|
// retrieve provider
|
||||||
|
$providers = $this->providerList($tenantId, $userId, new SourceSelector([$provider => true]));
|
||||||
|
if (!isset($providers[$provider])) {
|
||||||
|
throw new InvalidArgumentException("Provider '$provider' not found");
|
||||||
|
}
|
||||||
|
return $providers[$provider];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -62,25 +86,6 @@ class Manager {
|
|||||||
return $responseData;
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve specific provider for specific user
|
|
||||||
*
|
|
||||||
* @param string $tenantId tenant identifier
|
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param string $provider provider identifier
|
|
||||||
*
|
|
||||||
* @return IProviderBase
|
|
||||||
* @throws InvalidArgumentException
|
|
||||||
*/
|
|
||||||
public function providerFetch(string $tenantId, string $userId, string $provider): IProviderBase {
|
|
||||||
// retrieve provider
|
|
||||||
$providers = $this->providerList($tenantId, $userId, new SourceSelector([$provider => true]));
|
|
||||||
if (!isset($providers[$provider])) {
|
|
||||||
throw new InvalidArgumentException("Provider '$provider' not found");
|
|
||||||
}
|
|
||||||
return $providers[$provider];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve available services for specific user
|
* Retrieve available services for specific user
|
||||||
*
|
*
|
||||||
@@ -88,7 +93,7 @@ class Manager {
|
|||||||
* @param string $userId user identifier
|
* @param string $userId user identifier
|
||||||
* @param SourceSelector|null $sources list of provider and service identifiers
|
* @param SourceSelector|null $sources list of provider and service identifiers
|
||||||
*
|
*
|
||||||
* @return array<string,<string,IServiceBase>> collections of available services e.g. ['provider1' => ['service1' => IServiceBase], 'provider2' => ['service2' => IServiceBase]]
|
* @return array<string,<string,ServiceBaseInterface>> collections of available services e.g. ['provider1' => ['service1' => IServiceBase], 'provider2' => ['service2' => IServiceBase]]
|
||||||
*/
|
*/
|
||||||
public function serviceList(string $tenantId, string $userId, ?SourceSelector $sources = null): array {
|
public function serviceList(string $tenantId, string $userId, ?SourceSelector $sources = null): array {
|
||||||
// retrieve providers
|
// retrieve providers
|
||||||
@@ -96,13 +101,34 @@ class Manager {
|
|||||||
// retrieve services for each provider
|
// retrieve services for each provider
|
||||||
$responseData = [];
|
$responseData = [];
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
$serviceFilter = $sources[$provider->id()] instanceof ServiceSelector ? $sources[$provider->id()]->identifiers() : [];
|
$serviceFilter = $sources[$provider->identifier()] instanceof ServiceSelector ? $sources[$provider->identifier()]->identifiers() : [];
|
||||||
$services = $provider->serviceList($tenantId, $userId, $serviceFilter);
|
$services = $provider->serviceList($tenantId, $userId, $serviceFilter);
|
||||||
$responseData[$provider->id()] = $services;
|
$responseData[$provider->identifier()] = $services;
|
||||||
}
|
}
|
||||||
return $responseData;
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve service for specific user
|
||||||
|
*
|
||||||
|
* @param string $tenantId tenant identifier
|
||||||
|
* @param string $userId user identifier
|
||||||
|
* @param string $providerId provider identifier
|
||||||
|
* @param string|int $serviceId service identifier
|
||||||
|
*
|
||||||
|
* @return ServiceBaseInterface
|
||||||
|
* @throws InvalidArgumentException
|
||||||
|
*/
|
||||||
|
public function serviceFetch(string $tenantId, string $userId, string $providerId, string|int $serviceId): ServiceBaseInterface {
|
||||||
|
// retrieve provider and service
|
||||||
|
$service = $this->providerFetch($tenantId, $userId, $providerId)->serviceFetch($tenantId, $userId, $serviceId);
|
||||||
|
if ($service === null) {
|
||||||
|
throw new InvalidArgumentException("Service '$serviceId' not found for provider '$providerId'");
|
||||||
|
}
|
||||||
|
// retrieve services
|
||||||
|
return $service;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Confirm which services are available
|
* Confirm which services are available
|
||||||
*
|
*
|
||||||
@@ -123,44 +149,132 @@ class Manager {
|
|||||||
|
|
||||||
// retrieve services for each available provider
|
// retrieve services for each available provider
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
$serviceSelector = $sources[$provider->id()];
|
$serviceSelector = $sources[$provider->identifier()];
|
||||||
$serviceAvailability = $provider->serviceExtant($tenantId, $userId, ...$serviceSelector->identifiers());
|
$serviceAvailability = $provider->serviceExtant($tenantId, $userId, ...$serviceSelector->identifiers());
|
||||||
$responseData[$provider->id()] = $serviceAvailability;
|
$responseData[$provider->identifier()] = $serviceAvailability;
|
||||||
}
|
}
|
||||||
return $responseData;
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve service for specific user
|
* Create a new service
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param string $providerId provider identifier
|
|
||||||
* @param string|int $serviceId service identifier
|
|
||||||
*
|
*
|
||||||
* @return IServiceBase
|
* @param string $tenantId Tenant identifier
|
||||||
* @throws InvalidArgumentException
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param array $data Service configuration data
|
||||||
|
*
|
||||||
|
* @return ServiceBaseInterface Created service
|
||||||
|
*
|
||||||
|
* @throws InvalidArgumentException If provider doesn't support service creation
|
||||||
*/
|
*/
|
||||||
public function serviceFetch(string $tenantId, string $userId, string $providerId, string|int $serviceId): IServiceBase {
|
public function serviceCreate(string $tenantId, ?string $userId, string $providerId, array $data): ServiceBaseInterface {
|
||||||
// retrieve provider and service
|
// retrieve provider and service
|
||||||
$service = $this->providerFetch($tenantId, $userId, $providerId)->serviceFetch($tenantId, $userId, $serviceId);
|
$provider = $this->providerFetch($tenantId, $userId, $providerId);
|
||||||
if ($service === null) {
|
if ($provider instanceof ProviderServiceMutateInterface === false) {
|
||||||
throw new InvalidArgumentException("Service '$serviceId' not found for provider '$providerId'");
|
throw new InvalidArgumentException("Provider '$providerId' does not support service creation");
|
||||||
}
|
}
|
||||||
// retrieve services
|
|
||||||
return $service;
|
// Create a fresh service instance
|
||||||
|
$service = $provider->serviceFresh();
|
||||||
|
|
||||||
|
// Deserialize the data into the service
|
||||||
|
$service->jsonDeserialize($data);
|
||||||
|
|
||||||
|
// Create the service
|
||||||
|
$serviceId = $provider->serviceCreate($tenantId, $userId, $service);
|
||||||
|
|
||||||
|
// Fetch and return the created service
|
||||||
|
return $provider->serviceFetch($tenantId, $userId, $serviceId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve available collections for specific user
|
* Update an existing service
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param SourceSelector $sources list of provider and service identifiers
|
|
||||||
*
|
*
|
||||||
* @return array<string,<string,IServiceBase>> collections of available services e.g. ['provider1' => ['service1' => [ICollectionBase], 'service2' => [ICollectionBase]]]
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param string|int $serviceId Service identifier
|
||||||
|
* @param array $data Updated service configuration data
|
||||||
|
*
|
||||||
|
* @return ServiceBaseInterface Updated service
|
||||||
|
*
|
||||||
|
* @throws InvalidArgumentException If provider doesn't support service modification or service not found
|
||||||
*/
|
*/
|
||||||
public function collectionList(string $tenantId, string $userId, ?SourceSelector $sources = null, ?array $filter = null, ?array $sort = null): array {
|
public function serviceUpdate(string $tenantId, string $userId, string $providerId, string|int $serviceId, array $data): ServiceBaseInterface {
|
||||||
|
// retrieve provider and service
|
||||||
|
$provider = $this->providerFetch($tenantId, $userId, $providerId);
|
||||||
|
if ($provider instanceof ProviderServiceMutateInterface === false) {
|
||||||
|
throw new InvalidArgumentException("Provider '$providerId' does not support service modification");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch existing service
|
||||||
|
$service = $provider->serviceFetch($tenantId, $userId, $serviceId);
|
||||||
|
if ($service === null) {
|
||||||
|
throw new InvalidArgumentException("Service '$serviceId' not found");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update with new data
|
||||||
|
$service->jsonDeserialize($data);
|
||||||
|
|
||||||
|
// Modify the service
|
||||||
|
$provider->serviceModify($tenantId, $userId, $service);
|
||||||
|
|
||||||
|
// Fetch and return the updated service
|
||||||
|
return $provider->serviceFetch($tenantId, $userId, $serviceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a service
|
||||||
|
*
|
||||||
|
* @since 2025.05.01
|
||||||
|
*
|
||||||
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param string|int $serviceId Service identifier
|
||||||
|
*
|
||||||
|
* @return bool True if service was deleted
|
||||||
|
*
|
||||||
|
* @throws InvalidArgumentException If provider doesn't support service deletion or service not found
|
||||||
|
*/
|
||||||
|
public function serviceDelete(string $tenantId, string $userId, string $providerId, string|int $serviceId): bool {
|
||||||
|
// retrieve provider and service
|
||||||
|
$provider = $this->providerFetch($tenantId, $userId, $providerId);
|
||||||
|
if ($provider instanceof ProviderServiceMutateInterface === false) {
|
||||||
|
throw new InvalidArgumentException("Provider '$providerId' does not support service creation");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch existing service
|
||||||
|
$service = $provider->serviceFetch($tenantId, $userId, $serviceId);
|
||||||
|
if ($service === null) {
|
||||||
|
throw new InvalidArgumentException("Service '$serviceId' not found");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the service
|
||||||
|
return $provider->serviceDestroy($tenantId, $userId, $service);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== Collection Operations ====================
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List collections across services
|
||||||
|
*
|
||||||
|
* @since 2025.05.01
|
||||||
|
*
|
||||||
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param SourceSelector|null $sources Provider/service sources
|
||||||
|
* @param IFilter|null $filter Collection filter
|
||||||
|
* @param ISort|null $sort Collection sort
|
||||||
|
*
|
||||||
|
* @return array<string, array<string|int, array<string|int, ICollectionBase>>> Collections grouped by provider/service
|
||||||
|
*/
|
||||||
|
public function collectionList(string $tenantId, ?string $userId, ?SourceSelector $sources = null, ?IFilter $filter = null, ?ISort $sort = null): array {
|
||||||
// confirm that sources are provided
|
// confirm that sources are provided
|
||||||
if ($sources === null) {
|
if ($sources === null) {
|
||||||
$sources = new SourceSelector([]);
|
$sources = new SourceSelector([]);
|
||||||
@@ -170,7 +284,8 @@ class Manager {
|
|||||||
// retrieve services for each provider
|
// retrieve services for each provider
|
||||||
$responseData = [];
|
$responseData = [];
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
$serviceFilter = $sources[$provider->id()] instanceof ServiceSelector ? $sources[$provider->id()]->identifiers() : [];
|
$serviceFilter = $sources[$provider->identifier()] instanceof ServiceSelector ? $sources[$provider->identifier()]->identifiers() : [];
|
||||||
|
/** @var ServiceBaseInterface[] $services */
|
||||||
$services = $provider->serviceList($tenantId, $userId, $serviceFilter);
|
$services = $provider->serviceList($tenantId, $userId, $serviceFilter);
|
||||||
// retrieve collections for each service
|
// retrieve collections for each service
|
||||||
foreach ($services as $service) {
|
foreach ($services as $service) {
|
||||||
@@ -190,9 +305,9 @@ class Manager {
|
|||||||
$collectionSort->condition($attribute, $direction);
|
$collectionSort->condition($attribute, $direction);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$collections = $service->collectionList($collectionFilter, $collectionSort);
|
$collections = $service->collectionList('', $collectionFilter, $collectionSort);
|
||||||
if ($collections !== []) {
|
if ($collections !== []) {
|
||||||
$responseData[$provider->id()][$service->id()] = $collections;
|
$responseData[$provider->identifier()][$service->identifier()] = $collections;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -200,15 +315,17 @@ class Manager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Confirm which collections are available
|
* Check if collections exist
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param SourceSelector $sources collection of provider and service identifiers to confirm
|
|
||||||
*
|
*
|
||||||
* @return array<string,bool> collection of providers and their availability status e.g. ['provider1' => ['service1' => ['collection1' => true, 'collection2' => false]]]]
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param SourceSelector $sources Collection sources with identifiers
|
||||||
|
*
|
||||||
|
* @return array<string, array<string|int, array<string|int, bool>>> Existence map grouped by provider/service
|
||||||
*/
|
*/
|
||||||
public function collectionExtant(string $tenantId, string $userId, SourceSelector $sources): array {
|
public function collectionExtant(string $tenantId, ?string $userId, SourceSelector $sources): array {
|
||||||
// retrieve available providers
|
// retrieve available providers
|
||||||
$providers = $this->providerList($tenantId, $userId, $sources);
|
$providers = $this->providerList($tenantId, $userId, $sources);
|
||||||
$providersRequested = $sources->identifiers();
|
$providersRequested = $sources->identifiers();
|
||||||
@@ -219,19 +336,20 @@ class Manager {
|
|||||||
|
|
||||||
// check services and collections for each available provider
|
// check services and collections for each available provider
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
$serviceSelector = $sources[$provider->id()];
|
$serviceSelector = $sources[$provider->identifier()];
|
||||||
$servicesRequested = $serviceSelector->identifiers();
|
$servicesRequested = $serviceSelector->identifiers();
|
||||||
|
/** @var ServiceBaseInterface[] $servicesAvailable */
|
||||||
$servicesAvailable = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
$servicesAvailable = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
||||||
$servicesUnavailable = array_diff($servicesRequested, array_keys($servicesAvailable));
|
$servicesUnavailable = array_diff($servicesRequested, array_keys($servicesAvailable));
|
||||||
|
|
||||||
// mark unavailable services as false
|
// mark unavailable services as false
|
||||||
if ($servicesUnavailable !== []) {
|
if ($servicesUnavailable !== []) {
|
||||||
$responseData[$provider->id()] = array_fill_keys($servicesUnavailable, false);
|
$responseData[$provider->identifier()] = array_fill_keys($servicesUnavailable, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// confirm collections for each available service
|
// confirm collections for each available service
|
||||||
foreach ($servicesAvailable as $service) {
|
foreach ($servicesAvailable as $service) {
|
||||||
$collectionSelector = $serviceSelector[$service->id()];
|
$collectionSelector = $serviceSelector[$service->identifier()];
|
||||||
$collectionsRequested = $collectionSelector->identifiers();
|
$collectionsRequested = $collectionSelector->identifiers();
|
||||||
|
|
||||||
if ($collectionsRequested === []) {
|
if ($collectionsRequested === []) {
|
||||||
@@ -239,29 +357,36 @@ class Manager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// check each requested collection
|
// check each requested collection
|
||||||
foreach ($collectionsRequested as $collectionId) {
|
$collectionsAvailable = $service->collectionExtant(...$collectionsRequested);
|
||||||
$responseData[$provider->id()][$service->id()][$collectionId] = $service->collectionExtant((string)$collectionId);
|
$collectionsUnavailable = array_diff($collectionsRequested, array_keys($collectionsAvailable));
|
||||||
}
|
$responseData[$provider->identifier()][$service->identifier()] = array_merge(
|
||||||
|
$collectionsAvailable,
|
||||||
|
array_fill_keys($collectionsUnavailable, false)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return $responseData;
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve collection for specific user
|
* Fetch a specific collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param string $providerId provider identifier
|
|
||||||
* @param string|int $serviceId service identifier
|
|
||||||
* @param string|int $collectionId collection identifier
|
|
||||||
*
|
*
|
||||||
* @return ICollectionBase
|
* @param string $tenantId Tenant identifier
|
||||||
* @throws InvalidArgumentException
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param string|int $serviceId Service identifier
|
||||||
|
* @param string|int $collectionId Collection identifier
|
||||||
|
*
|
||||||
|
* @return CollectionBaseInterface|null
|
||||||
*/
|
*/
|
||||||
public function collectionFetch(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId): ICollectionBase {
|
public function collectionFetch(string $tenantId, ?string $userId, string $providerId, string|int $serviceId, string|int $collectionId): ?CollectionBaseInterface {
|
||||||
// retrieve services
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
if ($service === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
// retrieve collection
|
// retrieve collection
|
||||||
return $service->collectionFetch($collectionId);
|
return $service->collectionFetch($collectionId);
|
||||||
}
|
}
|
||||||
@@ -273,30 +398,34 @@ class Manager {
|
|||||||
* @param string $userId user identifier
|
* @param string $userId user identifier
|
||||||
* @param string $providerId provider identifier
|
* @param string $providerId provider identifier
|
||||||
* @param string|int $serviceId service identifier
|
* @param string|int $serviceId service identifier
|
||||||
* @param ICollectionBase|array $collection collection to create
|
* @param string|int|null $collectionId collection identifier (parent collection)
|
||||||
|
* @param CollectionMutableInterface|array $object collection to create
|
||||||
* @param array $options additional options for creation
|
* @param array $options additional options for creation
|
||||||
*
|
*
|
||||||
* @return ICollectionBase
|
* @return CollectionBaseInterface
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function collectionCreate(string $tenantId, string $userId, string $providerId, string|int $serviceId, ICollectionBase|array $collection, array $options = []): ICollectionBase {
|
public function collectionCreate(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int|null $collectionId, CollectionMutableInterface|array $object, array $options = []): CollectionBaseInterface {
|
||||||
// retrieve service
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports collection creation
|
// Check if service supports collection creation
|
||||||
if (!($service instanceof IServiceCollectionMutable)) {
|
if (!($service instanceof ServiceCollectionMutableInterface)) {
|
||||||
throw new InvalidArgumentException("Service does not support collection mutations");
|
throw new InvalidArgumentException("Service does not support collection mutations");
|
||||||
}
|
}
|
||||||
if (!$service->capable(IServiceCollectionMutable::CAPABILITY_COLLECTION_CREATE)) {
|
if (!$service->capable(ServiceCollectionMutableInterface::CAPABILITY_COLLECTION_CREATE)) {
|
||||||
throw new InvalidArgumentException("Service is not capable of creating collections");
|
throw new InvalidArgumentException("Service is not capable of creating collections");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_array($collection)) {
|
if (is_array($object)) {
|
||||||
$collection = $service->collectionFresh()->jsonDeserialize($collection);
|
$collection = $service->collectionFresh();
|
||||||
|
$collection->getProperties()->jsonDeserialize($object);
|
||||||
|
} else {
|
||||||
|
$collection = $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create collection (location is empty string for root)
|
// Create collection
|
||||||
return $service->collectionCreate('', $collection, $options);
|
return $service->collectionCreate($collectionId, $collection, $options);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -307,89 +436,100 @@ class Manager {
|
|||||||
* @param string $providerId provider identifier
|
* @param string $providerId provider identifier
|
||||||
* @param string|int $serviceId service identifier
|
* @param string|int $serviceId service identifier
|
||||||
* @param string|int $collectionId collection identifier
|
* @param string|int $collectionId collection identifier
|
||||||
* @param ICollectionBase $collectionData collection with modifications
|
* @param CollectionMutableInterface|array $object collection to modify
|
||||||
*
|
*
|
||||||
* @return ICollectionBase
|
* @return CollectionBaseInterface
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function collectionModify(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, ICollectionBase|array $collectionData): ICollectionBase {
|
public function collectionUpdate(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, CollectionMutableInterface|array $object): CollectionBaseInterface {
|
||||||
// retrieve service
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports collection modification
|
// Check if service supports collection creation
|
||||||
if (!($service instanceof IServiceCollectionMutable)) {
|
if (!($service instanceof ServiceCollectionMutableInterface)) {
|
||||||
throw new InvalidArgumentException("Service does not support collection mutations");
|
throw new InvalidArgumentException("Service does not support collection mutations");
|
||||||
}
|
}
|
||||||
if (!$service->capable(IServiceCollectionMutable::CAPABILITY_COLLECTION_MODIFY)) {
|
if (!$service->capable(ServiceCollectionMutableInterface::CAPABILITY_COLLECTION_UPDATE)) {
|
||||||
throw new InvalidArgumentException("Service is not capable of modifying collections");
|
throw new InvalidArgumentException("Service is not capable of updating collections");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_array($collectionData)) {
|
if (is_array($object)) {
|
||||||
$collectionData = $service->collectionFresh()->jsonDeserialize($collectionData);
|
$collection = $service->collectionFresh();
|
||||||
|
$collection->getProperties()->jsonDeserialize($object);
|
||||||
|
} else {
|
||||||
|
$collection = $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Modify collection
|
// Update collection
|
||||||
return $service->collectionModify($collectionId, $collectionData);
|
return $service->collectionUpdate($collectionId, $collection);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete a collection for a specific user
|
* Delete a specific collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param string $providerId provider identifier
|
|
||||||
* @param string|int $serviceId service identifier
|
|
||||||
* @param string|int $collectionId collection identifier
|
|
||||||
*
|
*
|
||||||
* @return bool
|
* @param string $tenantId Tenant identifier
|
||||||
* @throws InvalidArgumentException
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param string|int $serviceId Service identifier
|
||||||
|
* @param string|int $collectionId Collection identifier
|
||||||
|
*
|
||||||
|
* @return CollectionBaseInterface|null
|
||||||
*/
|
*/
|
||||||
public function collectionDestroy(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId): bool {
|
public function collectionDelete(string $tenantId, ?string $userId, string $providerId, string|int $serviceId, string|int $collectionId, array $options = []): bool {
|
||||||
// retrieve service
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports collection destruction
|
// Check if service supports collection deletion
|
||||||
if (!($service instanceof IServiceCollectionMutable)) {
|
if (!($service instanceof ServiceCollectionMutableInterface)) {
|
||||||
throw new InvalidArgumentException("Service does not support collection mutations");
|
throw new InvalidArgumentException("Service does not support collection mutations");
|
||||||
}
|
}
|
||||||
if (!$service->capable(IServiceCollectionMutable::CAPABILITY_COLLECTION_DESTROY)) {
|
if (!$service->capable(ServiceCollectionMutableInterface::CAPABILITY_COLLECTION_DELETE)) {
|
||||||
throw new InvalidArgumentException("Service is not capable of destroying collections");
|
throw new InvalidArgumentException("Service is not capable of deleting collections");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Destroy collection and cast result to bool
|
$force = $options['force'] ?? false;
|
||||||
return (bool)$service->collectionDestroy($collectionId);
|
$recursive = $options['recursive'] ?? false;
|
||||||
|
|
||||||
|
// delete collection
|
||||||
|
return $service->collectionDelete($collectionId, $force, $recursive);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ==================== Entity Operations ====================
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve available entities for specific user
|
* List Entities in a collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param SourceSelector $sources list of provider and service identifiers
|
|
||||||
*
|
*
|
||||||
* @return array<string,<string,IServiceBase>> collections of store enteties e.g. ['provider1' => ['service1' => [ICollectionBase], 'service2' => [ICollectionBase]]]
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string $userId User identifier
|
||||||
|
* @param SourceSelector $sources Entity sources with collection identifiers
|
||||||
|
* @param array|null $filter Entity filter
|
||||||
|
* @param array|null $sort Entity sort
|
||||||
|
* @param array|null $range Entity range/pagination
|
||||||
|
*
|
||||||
|
* @return array<string, array<string|int, array<string|int, array<string|int, IEntityBase>>>> Entities grouped by provider/service/collection
|
||||||
*/
|
*/
|
||||||
public function entityList(string $tenantId, string $userId, ?SourceSelector $sources = null, ?array $filter = null, ?array $sort = null, ?array $range = null): array {
|
public function entityList(string $tenantId, string $userId, SourceSelector $sources, array|null $filter = null, array|null $sort = null, array|null $range = null): array {
|
||||||
// confirm that sources are provided
|
|
||||||
if ($sources === null) {
|
|
||||||
$sources = new SourceSelector([]);
|
|
||||||
}
|
|
||||||
// retrieve providers
|
// retrieve providers
|
||||||
$providers = $this->providerList($tenantId, $userId, $sources);
|
$providers = $this->providerList($tenantId, $userId, $sources);
|
||||||
// retrieve services for each provider
|
// retrieve services for each provider
|
||||||
$responseData = [];
|
$responseData = [];
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
// retrieve services for each provider
|
// retrieve services for each provider
|
||||||
$serviceSelector = $sources[$provider->id()];
|
$serviceSelector = $sources[$provider->identifier()];
|
||||||
$servicesSelected = $provider->serviceList($tenantId,$userId, $serviceSelector->identifiers());
|
$servicesSelected = $provider->serviceList($tenantId,$userId, $serviceSelector->identifiers());
|
||||||
|
/** @var ServiceBaseInterface $service */
|
||||||
foreach ($servicesSelected as $service) {
|
foreach ($servicesSelected as $service) {
|
||||||
// retrieve collections for each service
|
// retrieve collections for each service
|
||||||
$collectionSelector = $serviceSelector[$service->id()];
|
$collectionSelector = $serviceSelector[$service->identifier()];
|
||||||
$collectionSelected = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
$collectionSelected = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
||||||
if ($collectionSelected === []) {
|
if ($collectionSelected === []) {
|
||||||
$collections = $service->collectionList();
|
$collections = $service->collectionList('');
|
||||||
$collectionSelected = array_map(
|
$collectionSelected = array_map(
|
||||||
fn($collection) => $collection->id(),
|
fn($collection) => $collection->identifier(),
|
||||||
$collections
|
$collections
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -437,67 +577,45 @@ class Manager {
|
|||||||
if ($entities === []) {
|
if ($entities === []) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$responseData[$provider->id()][$service->id()][$collectionId] = $entities;
|
$responseData[$provider->identifier()][$service->identifier()][$collectionId] = $entities;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return $responseData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Confirm which entities are available
|
|
||||||
*
|
|
||||||
* @param string $tenantId tenant identifier
|
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param array $sources collection of provider and service identifiers to confirm
|
|
||||||
*
|
|
||||||
* @return array collection of providers and their availability status
|
|
||||||
*/
|
|
||||||
public function entityDelta(string $tenantId, string $userId, SourceSelector $sources): array {
|
|
||||||
// confirm that sources are provided
|
|
||||||
if ($sources === null) {
|
|
||||||
$sources = new SourceSelector([]);
|
|
||||||
}
|
|
||||||
// retrieve providers
|
|
||||||
$providers = $this->providerList($tenantId, $userId, $sources);
|
|
||||||
$providersRequested = $sources->identifiers();
|
|
||||||
$providersUnavailable = array_diff($providersRequested, array_keys($providers));
|
|
||||||
// initialize response with unavailable providers
|
|
||||||
$responseData = array_fill_keys($providersUnavailable, false);
|
|
||||||
// iterate through available providers
|
|
||||||
foreach ($providers as $provider) {
|
|
||||||
$serviceSelector = $sources[$provider->id()];
|
|
||||||
$servicesRequested = $serviceSelector instanceof ServiceSelector ? $serviceSelector->identifiers() : [];
|
|
||||||
$services = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
|
||||||
$servicesUnavailable = array_diff($servicesRequested, array_keys($services));
|
|
||||||
if ($servicesUnavailable !== []) {
|
|
||||||
$responseData[$provider->id()] = array_fill_keys($servicesUnavailable, false);
|
|
||||||
}
|
|
||||||
// iterate through available services
|
|
||||||
foreach ($services as $service) {
|
|
||||||
$collectionSelector = $serviceSelector[$service->id()];
|
|
||||||
$collectionsRequested = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
|
||||||
if ($collectionsRequested === []) {
|
|
||||||
$responseData[$provider->id()][$service->id()] = false;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
foreach ($collectionsRequested as $collection) {
|
|
||||||
$entitySelector = $collectionSelector[$collection] ?? null;
|
|
||||||
$responseData[$provider->id()][$service->id()][$collection] = $service->entityDelta($collection, $entitySelector);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $responseData;
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Confirm which entities are available
|
* Fetch specific messages
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param SourceSelector $sources collection of provider and service identifiers to confirm
|
|
||||||
*
|
*
|
||||||
* @return array<string,bool> collection of providers and their availability status e.g. ['provider1' => ['service1' => ['collection1' => ['entity1' => true, 'entity2' => false]]]]
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param string $providerId Provider identifier
|
||||||
|
* @param string|int $serviceId Service identifier
|
||||||
|
* @param string|int $collectionId Collection identifier
|
||||||
|
* @param array<string|int> $identifiers Message identifiers
|
||||||
|
*
|
||||||
|
* @return array<string|int, IMessageBase> Messages indexed by ID
|
||||||
|
*/
|
||||||
|
public function entityFetch(string $tenantId, ?string $userId, string $providerId, string|int $serviceId, string|int $collectionId, array $identifiers): array {
|
||||||
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
|
// retrieve collection
|
||||||
|
return $service->entityFetch($collectionId, ...$identifiers);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if messages exist
|
||||||
|
*
|
||||||
|
* @since 2025.05.01
|
||||||
|
*
|
||||||
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param SourceSelector $sources Message sources with identifiers
|
||||||
|
*
|
||||||
|
* @return array<string, array<string|int, array<string|int, array<string|int, bool>>>> Existence map grouped by provider/service/collection
|
||||||
*/
|
*/
|
||||||
public function entityExtant(string $tenantId, string $userId, SourceSelector $sources): array {
|
public function entityExtant(string $tenantId, string $userId, SourceSelector $sources): array {
|
||||||
// confirm that sources are provided
|
// confirm that sources are provided
|
||||||
@@ -514,19 +632,20 @@ class Manager {
|
|||||||
|
|
||||||
// check services, collections, and entities for each available provider
|
// check services, collections, and entities for each available provider
|
||||||
foreach ($providers as $provider) {
|
foreach ($providers as $provider) {
|
||||||
$serviceSelector = $sources[$provider->id()];
|
$serviceSelector = $sources[$provider->identifier()];
|
||||||
$servicesRequested = $serviceSelector->identifiers();
|
$servicesRequested = $serviceSelector->identifiers();
|
||||||
|
/** @var ServiceBaseInterface[] $servicesAvailable */
|
||||||
$servicesAvailable = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
$servicesAvailable = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
||||||
$servicesUnavailable = array_diff($servicesRequested, array_keys($servicesAvailable));
|
$servicesUnavailable = array_diff($servicesRequested, array_keys($servicesAvailable));
|
||||||
|
|
||||||
// mark unavailable services as false
|
// mark unavailable services as false
|
||||||
if ($servicesUnavailable !== []) {
|
if ($servicesUnavailable !== []) {
|
||||||
$responseData[$provider->id()] = array_fill_keys($servicesUnavailable, false);
|
$responseData[$provider->identifier()] = array_fill_keys($servicesUnavailable, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// check collections and entities for each available service
|
// check collections and entities for each available service
|
||||||
foreach ($servicesAvailable as $service) {
|
foreach ($servicesAvailable as $service) {
|
||||||
$collectionSelector = $serviceSelector[$service->id()];
|
$collectionSelector = $serviceSelector[$service->identifier()];
|
||||||
$collectionsRequested = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
$collectionsRequested = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
||||||
|
|
||||||
if ($collectionsRequested === []) {
|
if ($collectionsRequested === []) {
|
||||||
@@ -540,7 +659,7 @@ class Manager {
|
|||||||
|
|
||||||
if (!$collectionExists) {
|
if (!$collectionExists) {
|
||||||
// collection doesn't exist, mark as false
|
// collection doesn't exist, mark as false
|
||||||
$responseData[$provider->id()][$service->id()][$collectionId] = false;
|
$responseData[$provider->identifier()][$service->identifier()][$collectionId] = false;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -550,10 +669,10 @@ class Manager {
|
|||||||
// handle both array of entity IDs and boolean true (meaning check if collection exists)
|
// handle both array of entity IDs and boolean true (meaning check if collection exists)
|
||||||
if ($entitySelector instanceof EntitySelector) {
|
if ($entitySelector instanceof EntitySelector) {
|
||||||
// check specific entities within the collection
|
// check specific entities within the collection
|
||||||
$responseData[$provider->id()][$service->id()][$collectionId] = $service->entityExtant($collectionId, ...$entitySelector->identifiers());
|
$responseData[$provider->identifier()][$service->identifier()][$collectionId] = $service->entityExtant($collectionId, ...$entitySelector->identifiers());
|
||||||
} elseif ($entitySelector === true) {
|
} elseif ($entitySelector === true) {
|
||||||
// just checking if collection exists (already confirmed above)
|
// just checking if collection exists (already confirmed above)
|
||||||
$responseData[$provider->id()][$service->id()][$collectionId] = true;
|
$responseData[$provider->identifier()][$service->identifier()][$collectionId] = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -562,104 +681,132 @@ class Manager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve entity for specific user and collection
|
* Get message delta/changes
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @since 2025.05.01
|
||||||
* @param string $userId user identifier
|
|
||||||
* @param string $providerId provider identifier
|
|
||||||
* @param string|int $serviceId service identifier
|
|
||||||
* @param string|int $collectionId collection identifier
|
|
||||||
* @param array<string|int> $identifiers entity identifiers
|
|
||||||
*
|
*
|
||||||
* @return array<string|int,IEntityBase>
|
* @param string $tenantId Tenant identifier
|
||||||
|
* @param string|null $userId User identifier for context
|
||||||
|
* @param SourceSelector $sources Message sources with signatures
|
||||||
|
*
|
||||||
|
* @return array<string, array<string|int, array<string|int, array>>> Delta grouped by provider/service/collection
|
||||||
*/
|
*/
|
||||||
public function entityFetch(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, array $identifiers): array {
|
public function entityDelta(string $tenantId, string $userId, SourceSelector $sources): array {
|
||||||
// retrieve services
|
// confirm that sources are provided
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
if ($sources === null) {
|
||||||
// retrieve collection
|
$sources = new SourceSelector([]);
|
||||||
return $service->entityFetch($collectionId, ...$identifiers);
|
}
|
||||||
|
// retrieve providers
|
||||||
|
$providers = $this->providerList($tenantId, $userId, $sources);
|
||||||
|
$providersRequested = $sources->identifiers();
|
||||||
|
$providersUnavailable = array_diff($providersRequested, array_keys($providers));
|
||||||
|
// initialize response with unavailable providers
|
||||||
|
$responseData = array_fill_keys($providersUnavailable, false);
|
||||||
|
// iterate through available providers
|
||||||
|
foreach ($providers as $provider) {
|
||||||
|
$serviceSelector = $sources[$provider->identifier()];
|
||||||
|
$servicesRequested = $serviceSelector instanceof ServiceSelector ? $serviceSelector->identifiers() : [];
|
||||||
|
/** @var ServiceBaseInterface[] $services */
|
||||||
|
$services = $provider->serviceList($tenantId, $userId, $servicesRequested);
|
||||||
|
$servicesUnavailable = array_diff($servicesRequested, array_keys($services));
|
||||||
|
if ($servicesUnavailable !== []) {
|
||||||
|
$responseData[$provider->identifier()] = array_fill_keys($servicesUnavailable, false);
|
||||||
|
}
|
||||||
|
// iterate through available services
|
||||||
|
foreach ($services as $service) {
|
||||||
|
$collectionSelector = $serviceSelector[$service->identifier()];
|
||||||
|
$collectionsRequested = $collectionSelector instanceof CollectionSelector ? $collectionSelector->identifiers() : [];
|
||||||
|
if ($collectionsRequested === []) {
|
||||||
|
$responseData[$provider->identifier()][$service->identifier()] = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
foreach ($collectionsRequested as $collection) {
|
||||||
|
$entitySelector = $collectionSelector[$collection] ?? null;
|
||||||
|
$responseData[$provider->identifier()][$service->identifier()][$collection] = $service->entityDelta($collection, $entitySelector);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $responseData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new entity in a specific collection for a specific user
|
* Create a new entity in a collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @param string $tenantId tenant identifier
|
||||||
* @param string $userId user identifier
|
* @param string $userId user identifier
|
||||||
* @param string $providerId provider identifier
|
* @param string $providerId provider identifier
|
||||||
* @param string|int $serviceId service identifier
|
* @param string|int $serviceId service identifier
|
||||||
* @param string|int $collectionId collection identifier
|
* @param string|int $collectionId collection identifier
|
||||||
* @param IEntityBase|array $entity entity to create
|
* @param EntityMutableInterface|array $entity entity to create
|
||||||
* @param array $options additional options for creation
|
* @param array $options additional options
|
||||||
*
|
*
|
||||||
* @return IEntityBase
|
* @return EntityBaseInterface
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function entityCreate(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, IEntityBase|array $entity, array $options = []): IEntityBase {
|
public function entityCreate(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, EntityMutableInterface|array $object, array $options = []): EntityBaseInterface {
|
||||||
// retrieve service
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports entity creation
|
// Check if service supports entity creation
|
||||||
if (!($service instanceof IServiceEntityMutable)) {
|
if (!($service instanceof ServiceEntityMutableInterface)) {
|
||||||
throw new InvalidArgumentException("Service does not support entity mutations");
|
throw new InvalidArgumentException("Service does not support entity mutations");
|
||||||
}
|
}
|
||||||
if (!$service->capable(IServiceEntityMutable::CAPABILITY_ENTITY_CREATE)) {
|
if (!$service->capable(ServiceEntityMutableInterface::CAPABILITY_ENTITY_CREATE)) {
|
||||||
throw new InvalidArgumentException("Service is not capable of creating entities");
|
throw new InvalidArgumentException("Service is not capable of creating entities");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_array($entity)) {
|
if (is_array($object)) {
|
||||||
$entityInstance = $service->entityFresh();
|
$entity = $service->entityFresh();
|
||||||
$entityInstance->jsonDeserialize($entity);
|
$entity->getProperties()->jsonDeserialize($object);
|
||||||
} else {
|
} else {
|
||||||
$entityInstance = $entity;
|
$entity = $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create entity
|
return $service->entityCreate($collectionId, $entity, $options);
|
||||||
return $service->entityCreate($collectionId, $entityInstance, $options);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Modify an existing entity in a collection for a specific user
|
* Modify an existing entity in a collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @param string $tenantId tenant identifier
|
||||||
* @param string $userId user identifier
|
* @param string $userId user identifier
|
||||||
* @param string $providerId provider identifier
|
* @param string|int $providerId provider identifier
|
||||||
* @param string|int $serviceId service identifier
|
* @param string|int $serviceId service identifier
|
||||||
* @param string|int $collectionId collection identifier
|
* @param string|int $collectionId collection identifier
|
||||||
* @param string|int $identifier entity identifier
|
* @param string|int $identifier entity identifier
|
||||||
* @param IEntityBase|array $entity entity with modifications
|
* @param EntityBaseInterface|array $entity entity with modifications
|
||||||
*
|
*
|
||||||
* @return IEntityBase
|
* @return EntityBaseInterface
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function entityModify(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, string|int $identifier, IEntityBase|array $entity): IEntityBase {
|
public function entityUpdate(string $tenantId, string $userId, string|int $providerId, string|int $serviceId, string|int $collectionId, string|int $identifier, EntityBaseInterface|array $object): EntityBaseInterface {
|
||||||
// retrieve service
|
// retrieve service
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports entity modification
|
// Check if service supports entity creation
|
||||||
if (!($service instanceof IServiceEntityMutable)) {
|
if (!($service instanceof ServiceEntityMutableInterface)) {
|
||||||
throw new InvalidArgumentException("Service does not support entity mutations");
|
throw new InvalidArgumentException("Service does not support entity mutations");
|
||||||
}
|
}
|
||||||
if (!$service->capable(IServiceEntityMutable::CAPABILITY_ENTITY_MODIFY)) {
|
if (!$service->capable(ServiceEntityMutableInterface::CAPABILITY_ENTITY_CREATE)) {
|
||||||
throw new InvalidArgumentException("Service is not capable of modifying entities");
|
throw new InvalidArgumentException("Service is not capable of creating entities");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_array($entity)) {
|
if (is_array($object)) {
|
||||||
$entityInstance = $service->entityFresh();
|
$entity = $service->entityFresh();
|
||||||
$entityInstance->jsonDeserialize($entity);
|
$entity->getProperties()->jsonDeserialize($object);
|
||||||
} else {
|
} else {
|
||||||
$entityInstance = $entity;
|
$entity = $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Modify entity
|
return $service->entityUpdate($collectionId, $identifier, $entity);
|
||||||
return $service->entityModify($collectionId, $identifier, $entityInstance);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete an entity from a collection for a specific user
|
* Destroy an entity from a collection
|
||||||
*
|
*
|
||||||
* @param string $tenantId tenant identifier
|
* @param string $tenantId tenant identifier
|
||||||
* @param string $userId user identifier
|
* @param string $userId user identifier
|
||||||
* @param string $providerId provider identifier
|
* @param string|int $providerId provider identifier
|
||||||
* @param string|int $serviceId service identifier
|
* @param string|int $serviceId service identifier
|
||||||
* @param string|int $collectionId collection identifier
|
* @param string|int $collectionId collection identifier
|
||||||
* @param string|int $identifier entity identifier
|
* @param string|int $identifier entity identifier
|
||||||
@@ -667,19 +814,16 @@ class Manager {
|
|||||||
* @return bool
|
* @return bool
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function entityDestroy(string $tenantId, string $userId, string $providerId, string|int $serviceId, string|int $collectionId, string|int $identifier): bool {
|
public function entityDelete(string $tenantId, string $userId, string|int $providerId, string|int $serviceId, string|int $collectionId, string|int $identifier): bool {
|
||||||
// retrieve service
|
|
||||||
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
$service = $this->serviceFetch($tenantId, $userId, $providerId, $serviceId);
|
||||||
|
|
||||||
// Check if service supports entity destruction
|
if (!($service instanceof ServiceEntityMutableInterface)) {
|
||||||
if (!($service instanceof IServiceEntityMutable)) {
|
throw new InvalidArgumentException('Service does not support entity destruction');
|
||||||
throw new InvalidArgumentException("Service does not support entity mutations");
|
|
||||||
}
|
|
||||||
if (!$service->capable(IServiceEntityMutable::CAPABILITY_ENTITY_DESTROY)) {
|
|
||||||
throw new InvalidArgumentException("Service is not capable of destroying entities");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Destroy entity and cast result to bool
|
$entity = $service->entityDelete($collectionId, $identifier);
|
||||||
return (bool)$service->entityDestroy($collectionId, $identifier);
|
|
||||||
|
return $entity !== null;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user