From 62fc300474980f483131dc8d36db8c2527ccb9cf Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Mon, 9 Oct 2023 11:47:36 +0200 Subject: [PATCH 1/6] Ensure file errors are written correctly --- .../delta/plugins/storage/files/Files.scala | 325 +++++++++++------- .../test/resources/files/errors/blank-id.json | 5 + .../files/routes/FilesRoutesSpec.scala | 140 ++++++-- .../nexus/delta/sdk/JsonLdValue.scala | 19 + .../sdk/directives/ResponseToJsonLd.scala | 9 +- .../delta/sdk/error/AuthTokenError.scala | 8 - 6 files changed, 345 insertions(+), 161 deletions(-) create mode 100644 delta/plugins/storage/src/test/resources/files/errors/blank-id.json diff --git a/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala b/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala index 48a69042ad..912b1fd738 100644 --- a/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala +++ b/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala @@ -54,27 +54,7 @@ import java.util.UUID /** * Operations for handling files */ -final class Files( - formDataExtractor: FormDataExtractor, - log: FilesLog, - aclCheck: AclCheck, - fetchContext: FetchContext[FileRejection], - storages: Storages, - storagesStatistics: StoragesStatistics, - remoteDiskStorageClient: RemoteDiskStorageClient, - config: StorageTypeConfig -)(implicit - uuidF: UUIDF, - system: ClassicActorSystem -) { - - implicit private val kamonComponent: KamonMetricComponent = KamonMetricComponent(entityType.value) - - // format: off - private val testStorageRef = ResourceRef.Revision(iri"http://localhost/test", 1) - private val testStorageType = StorageType.DiskStorage - private val testAttributes = FileAttributes(UUID.randomUUID(), "http://localhost", Uri.Path.Empty, "", None, 0, ComputedDigest(DigestAlgorithm.default, "value"), Client) - // format: on +trait Files { /** * Create a new file where the id is self generated @@ -90,16 +70,7 @@ final class Files( storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- generateId(pc) - _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) - } yield res - }.span("createFile") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Create a new file with the provided id @@ -118,16 +89,7 @@ final class Files( storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- expandIri(id, pc) - _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) - } yield res - }.span("createFile") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Create a new file linking where the id is self generated @@ -149,13 +111,7 @@ final class Files( filename: Option[String], mediaType: Option[ContentType], path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- generateId(pc) - res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) - } yield res - }.span("createLink") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Create a new file linking it from an existing file in a storage @@ -180,13 +136,7 @@ final class Files( filename: Option[String], mediaType: Option[ContentType], path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- expandIri(id, pc) - res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) - } yield res - }.span("createLink") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Update an existing file @@ -208,16 +158,7 @@ final class Files( projectRef: ProjectRef, rev: Int, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) - } yield res - }.span("updateFile") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Update a new file linking it from an existing file in a storage @@ -245,20 +186,7 @@ final class Files( mediaType: Option[ContentType], path: Uri.Path, rev: Int - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - resolvedFilename <- IO.fromOption(filename.orElse(path.lastSegment), InvalidFileLink(iri)) - description <- FileDescription(resolvedFilename, mediaType) - attributes <- LinkFile(storage, remoteDiskStorageClient, config) - .apply(path, description) - .mapError(LinkRejection(iri, storage.id, _)) - res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) - } yield res - }.span("updateLink") + )(implicit caller: Caller): IO[FileRejection, FileResource] /** * Add a tag to an existing file @@ -280,13 +208,7 @@ final class Files( tag: UserTag, tagRev: Int, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(TagFile(iri, projectRef, tagRev, tag, rev, subject)) - } yield res - }.span("tagFile") + )(implicit subject: Subject): IO[FileRejection, FileResource] /** * Delete a tag on an existing file. @@ -305,13 +227,7 @@ final class Files( projectRef: ProjectRef, tag: UserTag, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(DeleteFileTag(iri, projectRef, tag, rev, subject)) - } yield res - }.span("deleteFileTag") + )(implicit subject: Subject): IO[FileRejection, FileResource] /** * Deprecate an existing file @@ -327,13 +243,7 @@ final class Files( id: IdSegment, projectRef: ProjectRef, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(DeprecateFile(iri, projectRef, rev, subject)) - } yield res - }.span("deprecateFile") + )(implicit subject: Subject): IO[FileRejection, FileResource] /** * Fetch the last version of a file content @@ -343,7 +253,196 @@ final class Files( * @param project * the project where the storage belongs */ - def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] = { + def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] + + /** + * Fetch the last version of a file + * + * @param id + * the identifier that will be expanded to the Iri of the file with its optional rev/tag + * @param project + * the project where the storage belongs + */ + def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] + + /** + * Starts a stream that attempts to update file attributes asynchronously for linked files in remote storages + * + * @param offset + * the offset to start from + * @return + */ + private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] + + private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] +} + +final private class FilesImpl( + formDataExtractor: FormDataExtractor, + log: FilesLog, + aclCheck: AclCheck, + fetchContext: FetchContext[FileRejection], + storages: Storages, + storagesStatistics: StoragesStatistics, + remoteDiskStorageClient: RemoteDiskStorageClient, + config: StorageTypeConfig +)(implicit + uuidF: UUIDF, + system: ClassicActorSystem +) extends Files { + + implicit private val kamonComponent: KamonMetricComponent = KamonMetricComponent(entityType.value) + + private val logger: Logger = Logger[Files] + + // format: off + private val testStorageRef = ResourceRef.Revision(iri"http://localhost/test", 1) + private val testStorageType = StorageType.DiskStorage + private val testAttributes = FileAttributes(UUID.randomUUID(), "http://localhost", Uri.Path.Empty, "", None, 0, ComputedDigest(DigestAlgorithm.default, "value"), Client) + // format: on + + override def create( + storageId: Option[IdSegment], + projectRef: ProjectRef, + entity: HttpEntity + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- generateId(pc) + _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) + } yield res + }.span("createFile") + + override def create( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + entity: HttpEntity + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- expandIri(id, pc) + _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) + } yield res + }.span("createFile") + + override def createLink( + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- generateId(pc) + res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) + } yield res + }.span("createLink") + + override def createLink( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- expandIri(id, pc) + res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) + } yield res + }.span("createLink") + + override def update( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + rev: Int, + entity: HttpEntity + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) + } yield res + }.span("updateFile") + + override def updateLink( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path, + rev: Int + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + resolvedFilename <- IO.fromOption(filename.orElse(path.lastSegment), InvalidFileLink(iri)) + description <- FileDescription(resolvedFilename, mediaType) + attributes <- LinkFile(storage, remoteDiskStorageClient, config) + .apply(path, description) + .mapError(LinkRejection(iri, storage.id, _)) + res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) + } yield res + }.span("updateLink") + + override def tag( + id: IdSegment, + projectRef: ProjectRef, + tag: UserTag, + tagRev: Int, + rev: Int + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(TagFile(iri, projectRef, tagRev, tag, rev, subject)) + } yield res + }.span("tagFile") + + override def deleteTag( + id: IdSegment, + projectRef: ProjectRef, + tag: UserTag, + rev: Int + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(DeleteFileTag(iri, projectRef, tag, rev, subject)) + } yield res + }.span("deleteFileTag") + + override def deprecate( + id: IdSegment, + projectRef: ProjectRef, + rev: Int + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(DeprecateFile(iri, projectRef, rev, subject)) + } yield res + }.span("deprecateFile") + + override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit + caller: Caller + ): IO[FileRejection, FileResponse] = { for { file <- fetch(id, project) attributes = file.value.attributes @@ -358,15 +457,7 @@ final class Files( } yield FileResponse(attributes.filename, mediaType, attributes.bytes, s) }.span("fetchFileContent") - /** - * Fetch the last version of a file - * - * @param id - * the identifier that will be expanded to the Iri of the file with its optional rev/tag - * @param project - * the project where the storage belongs - */ - def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = { + override def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = { for { pc <- fetchContext.onRead(project) iri <- expandIri(id.value, pc) @@ -448,13 +539,7 @@ final class Files( private def generateId(pc: ProjectContext)(implicit uuidF: UUIDF): UIO[Iri] = uuidF().map(uuid => pc.base.iri / uuid.toString) - /** - * Starts a stream that attempts to update file attributes asynchronously for linked files in remote storages - * @param offset - * the offset to start from - * @return - */ - private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] = { + override private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] = { for { // The stream will start only if remote storage is enabled retryStrategy <- Stream.iterable(config.remoteDisk).map { c => @@ -515,7 +600,7 @@ final class Files( } yield stream } - private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] = + override private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] = for { f <- log.stateOr(project, iri, FileNotFound(iri, project)) storage <- storages @@ -543,8 +628,6 @@ final class Files( object Files { - private val logger: Logger = Logger[Files] - /** * The file entity type. */ @@ -719,7 +802,7 @@ object Files { as: ActorSystem[Nothing] ): Files = { implicit val classicAs: ClassicActorSystem = as.classicSystem - new Files( + new FilesImpl( FormDataExtractor(config.mediaTypeDetector), ScopedEventLog(definition, config.eventLog, xas), aclCheck, diff --git a/delta/plugins/storage/src/test/resources/files/errors/blank-id.json b/delta/plugins/storage/src/test/resources/files/errors/blank-id.json new file mode 100644 index 0000000000..66a095e9d3 --- /dev/null +++ b/delta/plugins/storage/src/test/resources/files/errors/blank-id.json @@ -0,0 +1,5 @@ +{ + "@context" : "https://bluebrain.github.io/nexus/contexts/error.json", + "@type" : "BlankResourceId", + "reason" : "Resource identifier cannot be blank." +} diff --git a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala index 2d8cb6a800..bf38eea8a9 100644 --- a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala +++ b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala @@ -5,38 +5,42 @@ import akka.http.scaladsl.model.ContentTypes.`text/plain(UTF-8)` import akka.http.scaladsl.model.MediaRanges._ import akka.http.scaladsl.model.MediaTypes.`text/html` import akka.http.scaladsl.model.headers.{Accept, Location, OAuth2BearerToken, RawHeader} -import akka.http.scaladsl.model.{StatusCodes, Uri} +import akka.http.scaladsl.model._ import akka.http.scaladsl.server.Route import ch.epfl.bluebrain.nexus.delta.kernel.http.MediaTypeDetectorConfig import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.model.Digest.ComputedDigest import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.model.{FileAttributes, FileRejection} import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.routes.FilesRoutesSpec.fileMetadata -import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.{contexts => fileContexts, permissions, FileFixtures, Files, FilesConfig} +import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.{contexts => fileContexts, permissions, FileFixtures, FileResource, Files, FilesConfig} import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.model.{StorageRejection, StorageStatEntry, StorageType} import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.operations.remote.client.RemoteDiskStorageClient -import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.{contexts => storageContexts, permissions => storagesPermissions, StorageFixtures, Storages, StoragesConfig, StoragesStatistics} +import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.{contexts => storageContexts, permissions => storagesPermissions, StorageFixtures, Storages, StoragesConfig} import ch.epfl.bluebrain.nexus.delta.rdf.IriOrBNode.Iri import ch.epfl.bluebrain.nexus.delta.rdf.RdfMediaTypes.`application/ld+json` import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary.{contexts, nxv} import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.context.{ContextValue, RemoteContextResolution} +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.encoder.JsonLdEncoder import ch.epfl.bluebrain.nexus.delta.sdk.IndexingAction import ch.epfl.bluebrain.nexus.delta.sdk.acls.AclSimpleCheck import ch.epfl.bluebrain.nexus.delta.sdk.acls.model.AclAddress import ch.epfl.bluebrain.nexus.delta.sdk.auth.{AuthTokenProvider, Credentials} -import ch.epfl.bluebrain.nexus.delta.sdk.directives.DeltaSchemeDirectives +import ch.epfl.bluebrain.nexus.delta.sdk.directives.{DeltaSchemeDirectives, FileResponse} import ch.epfl.bluebrain.nexus.delta.sdk.http.HttpClient import ch.epfl.bluebrain.nexus.delta.sdk.identities.IdentitiesDummy import ch.epfl.bluebrain.nexus.delta.sdk.identities.model.{Caller, ServiceAccount} import ch.epfl.bluebrain.nexus.delta.sdk.implicits._ -import ch.epfl.bluebrain.nexus.delta.sdk.model.{BaseUri, ResourceUris} +import ch.epfl.bluebrain.nexus.delta.sdk.marshalling.HttpResponseFields +import ch.epfl.bluebrain.nexus.delta.sdk.model.{BaseUri, IdSegment, IdSegmentRef, ResourceUris} import ch.epfl.bluebrain.nexus.delta.sdk.permissions.Permissions.events import ch.epfl.bluebrain.nexus.delta.sdk.permissions.model.Permission import ch.epfl.bluebrain.nexus.delta.sdk.projects.FetchContextDummy import ch.epfl.bluebrain.nexus.delta.sdk.resolvers.ResolverContextResolution +import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection import ch.epfl.bluebrain.nexus.delta.sdk.utils.{BaseRouteSpec, RouteFixtures} import ch.epfl.bluebrain.nexus.delta.sourcing.model.Identity.{Anonymous, Authenticated, Group, Subject, User} -import ch.epfl.bluebrain.nexus.delta.sourcing.model.{Label, ProjectRef, ResourceRef} +import ch.epfl.bluebrain.nexus.delta.sourcing.model.{Label, ProjectRef, ResourceRef, Tag} +import ch.epfl.bluebrain.nexus.delta.sourcing.offset.Offset import ch.epfl.bluebrain.nexus.testkit._ import ch.epfl.bluebrain.nexus.testkit.bio.IOFromMap import io.circe.Json @@ -53,8 +57,6 @@ class FilesRoutesSpec import akka.actor.typed.scaladsl.adapter._ implicit val typedSystem: typed.ActorSystem[Nothing] = system.toTyped val httpClient: HttpClient = HttpClient()(httpClientConfig, system, s) - val authTokenProvider: AuthTokenProvider = AuthTokenProvider.anonymousForTest - val remoteDiskStorageClient = new RemoteDiskStorageClient(httpClient, authTokenProvider, Credentials.Anonymous) // TODO: sort out how we handle this in tests implicit override def rcr: RemoteContextResolution = @@ -70,7 +72,6 @@ class FilesRoutesSpec implicit private val caller: Caller = Caller(alice, Set(alice, Anonymous, Authenticated(realm), Group("group", realm))) - private val identities = IdentitiesDummy(caller) private val asAlice = addCredentials(OAuth2BearerToken("alice")) @@ -94,9 +95,6 @@ class FilesRoutesSpec private val stCfg = config.copy(disk = config.disk.copy(defaultMaxFileSize = 1000, allowedVolumes = Set(path))) - private val storagesStatistics: StoragesStatistics = - (_, _) => IO.pure { StorageStatEntry(0, 0) } - private val aclCheck = AclSimpleCheck().accepted lazy val storages: Storages = Storages( fetchContext.mapRejection(StorageRejection.ProjectContextRejection), @@ -107,21 +105,100 @@ class FilesRoutesSpec StoragesConfig(eventLogConfig, pagination, stCfg), ServiceAccount(User("nexus-sa", Label.unsafe("sa"))) ).accepted - lazy val files: Files = Files( - fetchContext.mapRejection(FileRejection.ProjectContextRejection), - aclCheck, - storages, - storagesStatistics, - xas, - config, - FilesConfig(eventLogConfig, MediaTypeDetectorConfig.Empty), - remoteDiskStorageClient - ) - private val groupDirectives = - DeltaSchemeDirectives(fetchContext, ioFromMap(uuid -> projectRef.organization), ioFromMap(uuid -> projectRef)) - private lazy val routes = - Route.seal(FilesRoutes(stCfg, identities, aclCheck, files, groupDirectives, IndexingAction.noop)) + private def createFiles() = { + Files( + fetchContext.mapRejection(FileRejection.ProjectContextRejection), + aclCheck, + storages, + (_, _) => + IO.pure { + StorageStatEntry(0, 0) + }, + xas, + config, + FilesConfig(eventLogConfig, MediaTypeDetectorConfig.Empty), + new RemoteDiskStorageClient(httpClient, AuthTokenProvider.anonymousForTest, Credentials.Anonymous) + ) + } + + private lazy val routes = + createRoutes(createFiles()) + + private def filesWithErrorInFileContents[E: JsonLdEncoder: HttpResponseFields](error: E): Files = { + new Files { + override def create(storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit + caller: Caller + ): IO[FileRejection, FileResource] = ??? + override def create(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)( + implicit caller: Caller + ): IO[FileRejection, FileResource] = ??? + override def createLink( + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path + )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def createLink( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path + )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def update( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + rev: Int, + entity: HttpEntity + )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def updateLink( + id: IdSegment, + storageId: Option[IdSegment], + projectRef: ProjectRef, + filename: Option[String], + mediaType: Option[ContentType], + path: Uri.Path, + rev: Int + )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def tag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, tagRev: Int, rev: Int)(implicit + subject: Subject + ): IO[FileRejection, FileResource] = ??? + override def deleteTag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, rev: Int)(implicit + subject: Subject + ): IO[FileRejection, FileResource] = ??? + override def deprecate(id: IdSegment, projectRef: ProjectRef, rev: Int)(implicit + subject: Subject + ): IO[FileRejection, FileResource] = ??? + override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit + caller: Caller + ): IO[FileRejection, FileResponse] = IO.pure( + FileResponse( + "file.name", + ContentTypes.`text/plain(UTF-8)`, + 1024, + IO.raiseError(error) + ) + ) + override def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = ??? + override private[files] def attributesUpdateStream(offset: Offset) = ??? + override private[files] def updateAttributes(iri: Iri, project: ProjectRef) = ??? + } + } + + private def createRoutes(files: Files) = Route.seal( + FilesRoutes( + stCfg, + IdentitiesDummy(caller), + aclCheck, + files, + DeltaSchemeDirectives(fetchContext, ioFromMap(uuid -> projectRef.organization), ioFromMap(uuid -> projectRef)), + IndexingAction.noop + ) + ) private val diskIdRev = ResourceRef.Revision(dId, 1) private val s3IdRev = ResourceRef.Revision(s3Id, 2) @@ -473,6 +550,17 @@ class FilesRoutesSpec response.header[Location].value.uri shouldEqual Uri("https://bbp.epfl.ch/nexus/web/org/project/resources/file1") } } + + "fail when auth fails" in { + val mockRoutes = createRoutes(filesWithErrorInFileContents[ResourceRejection](ResourceRejection.BlankResourceId)) + Get("/v1/files/org/proj/file1") ~> Accept(`*/*`) ~> mockRoutes ~> check { + contentType.value shouldEqual `application/ld+json`.value + response.asJson shouldEqual jsonContentOf( + "/files/errors/blank-id.json" + ) + status shouldEqual StatusCodes.BadRequest + } + } } } diff --git a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/JsonLdValue.scala b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/JsonLdValue.scala index d1277fa9d3..aacffff40b 100644 --- a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/JsonLdValue.scala +++ b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/JsonLdValue.scala @@ -1,6 +1,11 @@ package ch.epfl.bluebrain.nexus.delta.sdk +import ch.epfl.bluebrain.nexus.delta.rdf.RdfError +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.api.{JsonLdApi, JsonLdOptions} +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.{CompactedJsonLd, ExpandedJsonLd} +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.context.{ContextValue, RemoteContextResolution} import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.encoder.JsonLdEncoder +import monix.bio.IO /** * A definition of a value that can be converted to JSONLD @@ -24,4 +29,18 @@ object JsonLdValue { override val value: A = v override val encoder: JsonLdEncoder[A] = implicitly[JsonLdEncoder[A]] } + + implicit val jsonLdEncoder: JsonLdEncoder[JsonLdValue] = { + new JsonLdEncoder[JsonLdValue] { + override def context(value: JsonLdValue): ContextValue = value.encoder.context(value.value) + override def expand( + value: JsonLdValue + )(implicit opts: JsonLdOptions, api: JsonLdApi, rcr: RemoteContextResolution): IO[RdfError, ExpandedJsonLd] = + value.encoder.expand(value.value) + override def compact( + value: JsonLdValue + )(implicit opts: JsonLdOptions, api: JsonLdApi, rcr: RemoteContextResolution): IO[RdfError, CompactedJsonLd] = + value.encoder.compact(value.value) + } + } } diff --git a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLd.scala b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLd.scala index 6dfde427e9..1b28a3a957 100644 --- a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLd.scala +++ b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLd.scala @@ -106,12 +106,9 @@ object ResponseToJsonLd extends FileBytesInstances { override def apply(statusOverride: Option[StatusCode]): Route = { val flattened = io.flatMap { fr => fr.content.attempt.map(_.map { s => fr.metadata -> s }) }.attempt onSuccess(flattened.runToFuture) { - case Left(complete: Complete[E]) => emit(complete) - case Left(reject: Reject[E]) => emit(reject) - case Right(Left(c)) => - implicit val valueEncoder = c.value.encoder - emit(c.value.value) - + case Left(complete: Complete[E]) => emit(complete) + case Left(reject: Reject[E]) => emit(reject) + case Right(Left(c)) => emit(c) case Right(Right((metadata, content))) => headerValueByType(Accept) { accept => if (accept.mediaRanges.exists(_.matches(metadata.contentType.mediaType))) { diff --git a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/error/AuthTokenError.scala b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/error/AuthTokenError.scala index 2f1bd3e3f8..6ef19a4c2e 100644 --- a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/error/AuthTokenError.scala +++ b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/error/AuthTokenError.scala @@ -27,12 +27,6 @@ object AuthTokenError { final case class AuthTokenNotFoundInResponse(failure: DecodingFailure) extends AuthTokenError(s"Auth token not found in auth response: ${failure.reason}") - /** - * Signals that the expiry was missing from the authentication response - */ - final case class ExpiryNotFoundInResponse(failure: DecodingFailure) - extends AuthTokenError(s"Expiry not found in auth response: ${failure.reason}") - /** * Signals that the realm specified for authentication is deprecated */ @@ -45,8 +39,6 @@ object AuthTokenError { JsonObject(keywords.tpe := "AuthTokenHttpError", "reason" := r.reason) case AuthTokenNotFoundInResponse(r) => JsonObject(keywords.tpe -> "AuthTokenNotFoundInResponse".asJson, "reason" := r.message) - case ExpiryNotFoundInResponse(r) => - JsonObject(keywords.tpe -> "ExpiryNotFoundInResponse".asJson, "reason" := r.message) case r: RealmIsDeprecated => JsonObject(keywords.tpe := "RealmIsDeprecated", "reason" := r.getMessage) } From a5a7be7b392ca9207d3bd121831a76a4abf6369d Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Mon, 9 Oct 2023 14:46:59 +0200 Subject: [PATCH 2/6] deal with scalafmt --- .../files/routes/FilesRoutesSpec.scala | 79 ++++++------------- 1 file changed, 22 insertions(+), 57 deletions(-) diff --git a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala index bf38eea8a9..70c360fa53 100644 --- a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala +++ b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala @@ -125,69 +125,34 @@ class FilesRoutesSpec private lazy val routes = createRoutes(createFiles()) + // format: off private def filesWithErrorInFileContents[E: JsonLdEncoder: HttpResponseFields](error: E): Files = { new Files { - override def create(storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit - caller: Caller - ): IO[FileRejection, FileResource] = ??? - override def create(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)( - implicit caller: Caller - ): IO[FileRejection, FileResource] = ??? - override def createLink( - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def createLink( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def update( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - rev: Int, - entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def updateLink( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path, - rev: Int - )(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def tag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, tagRev: Int, rev: Int)(implicit - subject: Subject - ): IO[FileRejection, FileResource] = ??? - override def deleteTag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, rev: Int)(implicit - subject: Subject - ): IO[FileRejection, FileResource] = ??? - override def deprecate(id: IdSegment, projectRef: ProjectRef, rev: Int)(implicit - subject: Subject - ): IO[FileRejection, FileResource] = ??? - override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit - caller: Caller - ): IO[FileRejection, FileResponse] = IO.pure( - FileResponse( - "file.name", - ContentTypes.`text/plain(UTF-8)`, - 1024, - IO.raiseError(error) + override def create(storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def create(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def createLink(storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def createLink(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def update(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, rev: Int, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def updateLink(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path, rev: Int)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? + override def tag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, tagRev: Int, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? + override def deleteTag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? + override def deprecate(id: IdSegment, projectRef: ProjectRef, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? + override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] = { + IO.pure( + FileResponse( + "file.name", + ContentTypes.`text/plain(UTF-8)`, + 1024, + IO.raiseError(error) + ) ) - ) + } override def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = ??? - override private[files] def attributesUpdateStream(offset: Offset) = ??? - override private[files] def updateAttributes(iri: Iri, project: ProjectRef) = ??? + override private[files] def attributesUpdateStream(offset: Offset) = ??? + override private[files] def updateAttributes(iri: Iri, project: ProjectRef) = ??? } } + // format: on private def createRoutes(files: Files) = Route.seal( FilesRoutes( From ec3e9eda7892fa42cf2b1a56709345f988c8faad Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Tue, 10 Oct 2023 00:06:59 +0200 Subject: [PATCH 3/6] Add unit test for ResponseToJsonLd --- .../test/resources/directives/blank-id.json | 5 ++ .../sdk/directives/ResponseToJsonLdSpec.scala | 89 +++++++++++++++++++ 2 files changed, 94 insertions(+) create mode 100644 delta/sdk/src/test/resources/directives/blank-id.json create mode 100644 delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala diff --git a/delta/sdk/src/test/resources/directives/blank-id.json b/delta/sdk/src/test/resources/directives/blank-id.json new file mode 100644 index 0000000000..66a095e9d3 --- /dev/null +++ b/delta/sdk/src/test/resources/directives/blank-id.json @@ -0,0 +1,5 @@ +{ + "@context" : "https://bluebrain.github.io/nexus/contexts/error.json", + "@type" : "BlankResourceId", + "reason" : "Resource identifier cannot be blank." +} diff --git a/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala new file mode 100644 index 0000000000..7a39c8cfbc --- /dev/null +++ b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala @@ -0,0 +1,89 @@ +package ch.epfl.bluebrain.nexus.delta.sdk.directives + +import akka.http.scaladsl.model.ContentTypes.`text/plain(UTF-8)` +import akka.http.scaladsl.model.MediaRanges.`*/*` +import akka.http.scaladsl.model.headers.Accept +import akka.http.scaladsl.model.{ContentType, StatusCodes} +import akka.http.scaladsl.server.RouteConcatenation +import akka.stream.scaladsl.Source +import akka.util.ByteString +import ch.epfl.bluebrain.nexus.delta.rdf.RdfMediaTypes.`application/ld+json` +import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary.contexts +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.context.RemoteContextResolution +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.encoder.JsonLdEncoder +import ch.epfl.bluebrain.nexus.delta.rdf.syntax.JsonSyntax +import ch.epfl.bluebrain.nexus.delta.rdf.utils.JsonKeyOrdering +import ch.epfl.bluebrain.nexus.delta.sdk.directives.DeltaDirectives._ +import ch.epfl.bluebrain.nexus.delta.sdk.marshalling.HttpResponseFields +import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection +import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection.BlankResourceId +import ch.epfl.bluebrain.nexus.delta.sdk.utils.RouteHelpers +import ch.epfl.bluebrain.nexus.delta.sdk.{AkkaSource, SimpleRejection, SimpleResource} +import ch.epfl.bluebrain.nexus.testkit.ShouldMatchers.convertToAnyShouldWrapper +import ch.epfl.bluebrain.nexus.testkit.TestHelpers.jsonContentOf +import monix.bio.IO +import monix.execution.Scheduler + +class ResponseToJsonLdSpec extends RouteHelpers with JsonSyntax with RouteConcatenation { + + implicit val s: Scheduler = Scheduler.global + implicit val rcr: RemoteContextResolution = + RemoteContextResolution.fixed( + SimpleResource.contextIri -> SimpleResource.context, + SimpleRejection.contextIri -> SimpleRejection.context, + contexts.error -> jsonContentOf("/contexts/error.json").topContextValueOrEmpty + ) + implicit val jo: JsonKeyOrdering = JsonKeyOrdering.default() + + private def responseWithSourceError[E: JsonLdEncoder: HttpResponseFields](error: E) = { + IO.pure( + responseWith( + `text/plain(UTF-8)`, + IO.raiseError(error) + ) + ) + } + + private val expectedBlankIdErrorResponse = jsonContentOf( + "/directives/blank-id.json" + ) + + private val FileContents = "hello" + + private def fileSourceOfString(value: String) = { + IO.pure(Source.single(ByteString(value))) + } + + private def responseWith[E: JsonLdEncoder: HttpResponseFields]( + contentType: ContentType, + contents: IO[E, AkkaSource] + ) = { + FileResponse( + "file.name", + contentType, + 1024, + contents + ) + } + + "ResponseToJsonLd file handling" should { + + "Return the contents of a file" in { + Get() ~> Accept(`*/*`) ~> emit( + IO.pure(responseWith(`text/plain(UTF-8)`, fileSourceOfString(FileContents))) + ) ~> check { + status shouldEqual StatusCodes.OK + contentType shouldEqual `text/plain(UTF-8)` + response.asString shouldEqual FileContents + } + } + + "Return an error from a file content IO" in { + Get() ~> emit(responseWithSourceError[ResourceRejection](BlankResourceId)) ~> check { + status shouldEqual StatusCodes.BadRequest + contentType.mediaType shouldEqual `application/ld+json` + response.asJson shouldEqual expectedBlankIdErrorResponse + } + } + } +} From b8398757cab40048d49b29eaf04bec5438f689fe Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Tue, 10 Oct 2023 09:34:51 +0200 Subject: [PATCH 4/6] Remove unnecessary test changes --- .../delta/plugins/storage/files/Files.scala | 325 +++++++----------- .../test/resources/files/errors/blank-id.json | 5 - .../files/routes/FilesRoutesSpec.scala | 105 ++---- 3 files changed, 147 insertions(+), 288 deletions(-) delete mode 100644 delta/plugins/storage/src/test/resources/files/errors/blank-id.json diff --git a/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala b/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala index 912b1fd738..48a69042ad 100644 --- a/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala +++ b/delta/plugins/storage/src/main/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/Files.scala @@ -54,7 +54,27 @@ import java.util.UUID /** * Operations for handling files */ -trait Files { +final class Files( + formDataExtractor: FormDataExtractor, + log: FilesLog, + aclCheck: AclCheck, + fetchContext: FetchContext[FileRejection], + storages: Storages, + storagesStatistics: StoragesStatistics, + remoteDiskStorageClient: RemoteDiskStorageClient, + config: StorageTypeConfig +)(implicit + uuidF: UUIDF, + system: ClassicActorSystem +) { + + implicit private val kamonComponent: KamonMetricComponent = KamonMetricComponent(entityType.value) + + // format: off + private val testStorageRef = ResourceRef.Revision(iri"http://localhost/test", 1) + private val testStorageType = StorageType.DiskStorage + private val testAttributes = FileAttributes(UUID.randomUUID(), "http://localhost", Uri.Path.Empty, "", None, 0, ComputedDigest(DigestAlgorithm.default, "value"), Client) + // format: on /** * Create a new file where the id is self generated @@ -70,7 +90,16 @@ trait Files { storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- generateId(pc) + _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) + } yield res + }.span("createFile") /** * Create a new file with the provided id @@ -89,7 +118,16 @@ trait Files { storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- expandIri(id, pc) + _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) + } yield res + }.span("createFile") /** * Create a new file linking where the id is self generated @@ -111,7 +149,13 @@ trait Files { filename: Option[String], mediaType: Option[ContentType], path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- generateId(pc) + res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) + } yield res + }.span("createLink") /** * Create a new file linking it from an existing file in a storage @@ -136,7 +180,13 @@ trait Files { filename: Option[String], mediaType: Option[ContentType], path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onCreate(projectRef) + iri <- expandIri(id, pc) + res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) + } yield res + }.span("createLink") /** * Update an existing file @@ -158,7 +208,16 @@ trait Files { projectRef: ProjectRef, rev: Int, entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + attributes <- extractFileAttributes(iri, entity, storage) + res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) + } yield res + }.span("updateFile") /** * Update a new file linking it from an existing file in a storage @@ -186,7 +245,20 @@ trait Files { mediaType: Option[ContentType], path: Uri.Path, rev: Int - )(implicit caller: Caller): IO[FileRejection, FileResource] + )(implicit caller: Caller): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) + (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) + resolvedFilename <- IO.fromOption(filename.orElse(path.lastSegment), InvalidFileLink(iri)) + description <- FileDescription(resolvedFilename, mediaType) + attributes <- LinkFile(storage, remoteDiskStorageClient, config) + .apply(path, description) + .mapError(LinkRejection(iri, storage.id, _)) + res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) + } yield res + }.span("updateLink") /** * Add a tag to an existing file @@ -208,7 +280,13 @@ trait Files { tag: UserTag, tagRev: Int, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(TagFile(iri, projectRef, tagRev, tag, rev, subject)) + } yield res + }.span("tagFile") /** * Delete a tag on an existing file. @@ -227,7 +305,13 @@ trait Files { projectRef: ProjectRef, tag: UserTag, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(DeleteFileTag(iri, projectRef, tag, rev, subject)) + } yield res + }.span("deleteFileTag") /** * Deprecate an existing file @@ -243,7 +327,13 @@ trait Files { id: IdSegment, projectRef: ProjectRef, rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] + )(implicit subject: Subject): IO[FileRejection, FileResource] = { + for { + pc <- fetchContext.onModify(projectRef) + iri <- expandIri(id, pc) + res <- eval(DeprecateFile(iri, projectRef, rev, subject)) + } yield res + }.span("deprecateFile") /** * Fetch the last version of a file content @@ -253,196 +343,7 @@ trait Files { * @param project * the project where the storage belongs */ - def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] - - /** - * Fetch the last version of a file - * - * @param id - * the identifier that will be expanded to the Iri of the file with its optional rev/tag - * @param project - * the project where the storage belongs - */ - def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] - - /** - * Starts a stream that attempts to update file attributes asynchronously for linked files in remote storages - * - * @param offset - * the offset to start from - * @return - */ - private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] - - private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] -} - -final private class FilesImpl( - formDataExtractor: FormDataExtractor, - log: FilesLog, - aclCheck: AclCheck, - fetchContext: FetchContext[FileRejection], - storages: Storages, - storagesStatistics: StoragesStatistics, - remoteDiskStorageClient: RemoteDiskStorageClient, - config: StorageTypeConfig -)(implicit - uuidF: UUIDF, - system: ClassicActorSystem -) extends Files { - - implicit private val kamonComponent: KamonMetricComponent = KamonMetricComponent(entityType.value) - - private val logger: Logger = Logger[Files] - - // format: off - private val testStorageRef = ResourceRef.Revision(iri"http://localhost/test", 1) - private val testStorageType = StorageType.DiskStorage - private val testAttributes = FileAttributes(UUID.randomUUID(), "http://localhost", Uri.Path.Empty, "", None, 0, ComputedDigest(DigestAlgorithm.default, "value"), Client) - // format: on - - override def create( - storageId: Option[IdSegment], - projectRef: ProjectRef, - entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- generateId(pc) - _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) - } yield res - }.span("createFile") - - override def create( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- expandIri(id, pc) - _ <- test(CreateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(CreateFile(iri, projectRef, storageRef, storage.tpe, attributes, caller.subject)) - } yield res - }.span("createFile") - - override def createLink( - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- generateId(pc) - res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) - } yield res - }.span("createLink") - - override def createLink( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onCreate(projectRef) - iri <- expandIri(id, pc) - res <- createLink(iri, projectRef, pc, storageId, filename, mediaType, path) - } yield res - }.span("createLink") - - override def update( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - rev: Int, - entity: HttpEntity - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - attributes <- extractFileAttributes(iri, entity, storage) - res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) - } yield res - }.span("updateFile") - - override def updateLink( - id: IdSegment, - storageId: Option[IdSegment], - projectRef: ProjectRef, - filename: Option[String], - mediaType: Option[ContentType], - path: Uri.Path, - rev: Int - )(implicit caller: Caller): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - _ <- test(UpdateFile(iri, projectRef, testStorageRef, testStorageType, testAttributes, rev, caller.subject)) - (storageRef, storage) <- fetchActiveStorage(storageId, projectRef, pc) - resolvedFilename <- IO.fromOption(filename.orElse(path.lastSegment), InvalidFileLink(iri)) - description <- FileDescription(resolvedFilename, mediaType) - attributes <- LinkFile(storage, remoteDiskStorageClient, config) - .apply(path, description) - .mapError(LinkRejection(iri, storage.id, _)) - res <- eval(UpdateFile(iri, projectRef, storageRef, storage.tpe, attributes, rev, caller.subject)) - } yield res - }.span("updateLink") - - override def tag( - id: IdSegment, - projectRef: ProjectRef, - tag: UserTag, - tagRev: Int, - rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(TagFile(iri, projectRef, tagRev, tag, rev, subject)) - } yield res - }.span("tagFile") - - override def deleteTag( - id: IdSegment, - projectRef: ProjectRef, - tag: UserTag, - rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(DeleteFileTag(iri, projectRef, tag, rev, subject)) - } yield res - }.span("deleteFileTag") - - override def deprecate( - id: IdSegment, - projectRef: ProjectRef, - rev: Int - )(implicit subject: Subject): IO[FileRejection, FileResource] = { - for { - pc <- fetchContext.onModify(projectRef) - iri <- expandIri(id, pc) - res <- eval(DeprecateFile(iri, projectRef, rev, subject)) - } yield res - }.span("deprecateFile") - - override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit - caller: Caller - ): IO[FileRejection, FileResponse] = { + def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] = { for { file <- fetch(id, project) attributes = file.value.attributes @@ -457,7 +358,15 @@ final private class FilesImpl( } yield FileResponse(attributes.filename, mediaType, attributes.bytes, s) }.span("fetchFileContent") - override def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = { + /** + * Fetch the last version of a file + * + * @param id + * the identifier that will be expanded to the Iri of the file with its optional rev/tag + * @param project + * the project where the storage belongs + */ + def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = { for { pc <- fetchContext.onRead(project) iri <- expandIri(id.value, pc) @@ -539,7 +448,13 @@ final private class FilesImpl( private def generateId(pc: ProjectContext)(implicit uuidF: UUIDF): UIO[Iri] = uuidF().map(uuid => pc.base.iri / uuid.toString) - override private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] = { + /** + * Starts a stream that attempts to update file attributes asynchronously for linked files in remote storages + * @param offset + * the offset to start from + * @return + */ + private[files] def attributesUpdateStream(offset: Offset): ElemStream[Unit] = { for { // The stream will start only if remote storage is enabled retryStrategy <- Stream.iterable(config.remoteDisk).map { c => @@ -600,7 +515,7 @@ final private class FilesImpl( } yield stream } - override private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] = + private[files] def updateAttributes(iri: Iri, project: ProjectRef): IO[FileRejection, Unit] = for { f <- log.stateOr(project, iri, FileNotFound(iri, project)) storage <- storages @@ -628,6 +543,8 @@ final private class FilesImpl( object Files { + private val logger: Logger = Logger[Files] + /** * The file entity type. */ @@ -802,7 +719,7 @@ object Files { as: ActorSystem[Nothing] ): Files = { implicit val classicAs: ClassicActorSystem = as.classicSystem - new FilesImpl( + new Files( FormDataExtractor(config.mediaTypeDetector), ScopedEventLog(definition, config.eventLog, xas), aclCheck, diff --git a/delta/plugins/storage/src/test/resources/files/errors/blank-id.json b/delta/plugins/storage/src/test/resources/files/errors/blank-id.json deleted file mode 100644 index 66a095e9d3..0000000000 --- a/delta/plugins/storage/src/test/resources/files/errors/blank-id.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "@context" : "https://bluebrain.github.io/nexus/contexts/error.json", - "@type" : "BlankResourceId", - "reason" : "Resource identifier cannot be blank." -} diff --git a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala index 70c360fa53..2d8cb6a800 100644 --- a/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala +++ b/delta/plugins/storage/src/test/scala/ch/epfl/bluebrain/nexus/delta/plugins/storage/files/routes/FilesRoutesSpec.scala @@ -5,42 +5,38 @@ import akka.http.scaladsl.model.ContentTypes.`text/plain(UTF-8)` import akka.http.scaladsl.model.MediaRanges._ import akka.http.scaladsl.model.MediaTypes.`text/html` import akka.http.scaladsl.model.headers.{Accept, Location, OAuth2BearerToken, RawHeader} -import akka.http.scaladsl.model._ +import akka.http.scaladsl.model.{StatusCodes, Uri} import akka.http.scaladsl.server.Route import ch.epfl.bluebrain.nexus.delta.kernel.http.MediaTypeDetectorConfig import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.model.Digest.ComputedDigest import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.model.{FileAttributes, FileRejection} import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.routes.FilesRoutesSpec.fileMetadata -import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.{contexts => fileContexts, permissions, FileFixtures, FileResource, Files, FilesConfig} +import ch.epfl.bluebrain.nexus.delta.plugins.storage.files.{contexts => fileContexts, permissions, FileFixtures, Files, FilesConfig} import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.model.{StorageRejection, StorageStatEntry, StorageType} import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.operations.remote.client.RemoteDiskStorageClient -import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.{contexts => storageContexts, permissions => storagesPermissions, StorageFixtures, Storages, StoragesConfig} +import ch.epfl.bluebrain.nexus.delta.plugins.storage.storages.{contexts => storageContexts, permissions => storagesPermissions, StorageFixtures, Storages, StoragesConfig, StoragesStatistics} import ch.epfl.bluebrain.nexus.delta.rdf.IriOrBNode.Iri import ch.epfl.bluebrain.nexus.delta.rdf.RdfMediaTypes.`application/ld+json` import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary.{contexts, nxv} import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.context.{ContextValue, RemoteContextResolution} -import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.encoder.JsonLdEncoder import ch.epfl.bluebrain.nexus.delta.sdk.IndexingAction import ch.epfl.bluebrain.nexus.delta.sdk.acls.AclSimpleCheck import ch.epfl.bluebrain.nexus.delta.sdk.acls.model.AclAddress import ch.epfl.bluebrain.nexus.delta.sdk.auth.{AuthTokenProvider, Credentials} -import ch.epfl.bluebrain.nexus.delta.sdk.directives.{DeltaSchemeDirectives, FileResponse} +import ch.epfl.bluebrain.nexus.delta.sdk.directives.DeltaSchemeDirectives import ch.epfl.bluebrain.nexus.delta.sdk.http.HttpClient import ch.epfl.bluebrain.nexus.delta.sdk.identities.IdentitiesDummy import ch.epfl.bluebrain.nexus.delta.sdk.identities.model.{Caller, ServiceAccount} import ch.epfl.bluebrain.nexus.delta.sdk.implicits._ -import ch.epfl.bluebrain.nexus.delta.sdk.marshalling.HttpResponseFields -import ch.epfl.bluebrain.nexus.delta.sdk.model.{BaseUri, IdSegment, IdSegmentRef, ResourceUris} +import ch.epfl.bluebrain.nexus.delta.sdk.model.{BaseUri, ResourceUris} import ch.epfl.bluebrain.nexus.delta.sdk.permissions.Permissions.events import ch.epfl.bluebrain.nexus.delta.sdk.permissions.model.Permission import ch.epfl.bluebrain.nexus.delta.sdk.projects.FetchContextDummy import ch.epfl.bluebrain.nexus.delta.sdk.resolvers.ResolverContextResolution -import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection import ch.epfl.bluebrain.nexus.delta.sdk.utils.{BaseRouteSpec, RouteFixtures} import ch.epfl.bluebrain.nexus.delta.sourcing.model.Identity.{Anonymous, Authenticated, Group, Subject, User} -import ch.epfl.bluebrain.nexus.delta.sourcing.model.{Label, ProjectRef, ResourceRef, Tag} -import ch.epfl.bluebrain.nexus.delta.sourcing.offset.Offset +import ch.epfl.bluebrain.nexus.delta.sourcing.model.{Label, ProjectRef, ResourceRef} import ch.epfl.bluebrain.nexus.testkit._ import ch.epfl.bluebrain.nexus.testkit.bio.IOFromMap import io.circe.Json @@ -57,6 +53,8 @@ class FilesRoutesSpec import akka.actor.typed.scaladsl.adapter._ implicit val typedSystem: typed.ActorSystem[Nothing] = system.toTyped val httpClient: HttpClient = HttpClient()(httpClientConfig, system, s) + val authTokenProvider: AuthTokenProvider = AuthTokenProvider.anonymousForTest + val remoteDiskStorageClient = new RemoteDiskStorageClient(httpClient, authTokenProvider, Credentials.Anonymous) // TODO: sort out how we handle this in tests implicit override def rcr: RemoteContextResolution = @@ -72,6 +70,7 @@ class FilesRoutesSpec implicit private val caller: Caller = Caller(alice, Set(alice, Anonymous, Authenticated(realm), Group("group", realm))) + private val identities = IdentitiesDummy(caller) private val asAlice = addCredentials(OAuth2BearerToken("alice")) @@ -95,6 +94,9 @@ class FilesRoutesSpec private val stCfg = config.copy(disk = config.disk.copy(defaultMaxFileSize = 1000, allowedVolumes = Set(path))) + private val storagesStatistics: StoragesStatistics = + (_, _) => IO.pure { StorageStatEntry(0, 0) } + private val aclCheck = AclSimpleCheck().accepted lazy val storages: Storages = Storages( fetchContext.mapRejection(StorageRejection.ProjectContextRejection), @@ -105,65 +107,21 @@ class FilesRoutesSpec StoragesConfig(eventLogConfig, pagination, stCfg), ServiceAccount(User("nexus-sa", Label.unsafe("sa"))) ).accepted - - private def createFiles() = { - Files( - fetchContext.mapRejection(FileRejection.ProjectContextRejection), - aclCheck, - storages, - (_, _) => - IO.pure { - StorageStatEntry(0, 0) - }, - xas, - config, - FilesConfig(eventLogConfig, MediaTypeDetectorConfig.Empty), - new RemoteDiskStorageClient(httpClient, AuthTokenProvider.anonymousForTest, Credentials.Anonymous) - ) - } - - private lazy val routes = - createRoutes(createFiles()) - - // format: off - private def filesWithErrorInFileContents[E: JsonLdEncoder: HttpResponseFields](error: E): Files = { - new Files { - override def create(storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def create(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def createLink(storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def createLink(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def update(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, rev: Int, entity: HttpEntity)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def updateLink(id: IdSegment, storageId: Option[IdSegment], projectRef: ProjectRef, filename: Option[String], mediaType: Option[ContentType], path: Uri.Path, rev: Int)(implicit caller: Caller): IO[FileRejection, FileResource] = ??? - override def tag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, tagRev: Int, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? - override def deleteTag(id: IdSegment, projectRef: ProjectRef, tag: Tag.UserTag, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? - override def deprecate(id: IdSegment, projectRef: ProjectRef, rev: Int)(implicit subject: Subject): IO[FileRejection, FileResource] = ??? - override def fetchContent(id: IdSegmentRef, project: ProjectRef)(implicit caller: Caller): IO[FileRejection, FileResponse] = { - IO.pure( - FileResponse( - "file.name", - ContentTypes.`text/plain(UTF-8)`, - 1024, - IO.raiseError(error) - ) - ) - } - override def fetch(id: IdSegmentRef, project: ProjectRef): IO[FileRejection, FileResource] = ??? - override private[files] def attributesUpdateStream(offset: Offset) = ??? - override private[files] def updateAttributes(iri: Iri, project: ProjectRef) = ??? - } - } - // format: on - - private def createRoutes(files: Files) = Route.seal( - FilesRoutes( - stCfg, - IdentitiesDummy(caller), - aclCheck, - files, - DeltaSchemeDirectives(fetchContext, ioFromMap(uuid -> projectRef.organization), ioFromMap(uuid -> projectRef)), - IndexingAction.noop - ) + lazy val files: Files = Files( + fetchContext.mapRejection(FileRejection.ProjectContextRejection), + aclCheck, + storages, + storagesStatistics, + xas, + config, + FilesConfig(eventLogConfig, MediaTypeDetectorConfig.Empty), + remoteDiskStorageClient ) + private val groupDirectives = + DeltaSchemeDirectives(fetchContext, ioFromMap(uuid -> projectRef.organization), ioFromMap(uuid -> projectRef)) + + private lazy val routes = + Route.seal(FilesRoutes(stCfg, identities, aclCheck, files, groupDirectives, IndexingAction.noop)) private val diskIdRev = ResourceRef.Revision(dId, 1) private val s3IdRev = ResourceRef.Revision(s3Id, 2) @@ -515,17 +473,6 @@ class FilesRoutesSpec response.header[Location].value.uri shouldEqual Uri("https://bbp.epfl.ch/nexus/web/org/project/resources/file1") } } - - "fail when auth fails" in { - val mockRoutes = createRoutes(filesWithErrorInFileContents[ResourceRejection](ResourceRejection.BlankResourceId)) - Get("/v1/files/org/proj/file1") ~> Accept(`*/*`) ~> mockRoutes ~> check { - contentType.value shouldEqual `application/ld+json`.value - response.asJson shouldEqual jsonContentOf( - "/files/errors/blank-id.json" - ) - status shouldEqual StatusCodes.BadRequest - } - } } } From 9ddfd5b1860a79d2df78e5b25b8e104404034799 Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Tue, 10 Oct 2023 09:57:03 +0200 Subject: [PATCH 5/6] tidy up test --- .../sdk/directives/ResponseToJsonLdSpec.scala | 32 +++++++++++-------- 1 file changed, 18 insertions(+), 14 deletions(-) diff --git a/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala index 7a39c8cfbc..05c8b40b1a 100644 --- a/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala +++ b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/directives/ResponseToJsonLdSpec.scala @@ -36,11 +36,9 @@ class ResponseToJsonLdSpec extends RouteHelpers with JsonSyntax with RouteConcat implicit val jo: JsonKeyOrdering = JsonKeyOrdering.default() private def responseWithSourceError[E: JsonLdEncoder: HttpResponseFields](error: E) = { - IO.pure( - responseWith( - `text/plain(UTF-8)`, - IO.raiseError(error) - ) + responseWith( + `text/plain(UTF-8)`, + IO.raiseError(error) ) } @@ -58,19 +56,25 @@ class ResponseToJsonLdSpec extends RouteHelpers with JsonSyntax with RouteConcat contentType: ContentType, contents: IO[E, AkkaSource] ) = { - FileResponse( - "file.name", - contentType, - 1024, - contents + IO.pure( + FileResponse( + "file.name", + contentType, + 1024, + contents + ) ) } + private def request = { + Get() ~> Accept(`*/*`) + } + "ResponseToJsonLd file handling" should { "Return the contents of a file" in { - Get() ~> Accept(`*/*`) ~> emit( - IO.pure(responseWith(`text/plain(UTF-8)`, fileSourceOfString(FileContents))) + request ~> emit( + responseWith(`text/plain(UTF-8)`, fileSourceOfString(FileContents)) ) ~> check { status shouldEqual StatusCodes.OK contentType shouldEqual `text/plain(UTF-8)` @@ -79,8 +83,8 @@ class ResponseToJsonLdSpec extends RouteHelpers with JsonSyntax with RouteConcat } "Return an error from a file content IO" in { - Get() ~> emit(responseWithSourceError[ResourceRejection](BlankResourceId)) ~> check { - status shouldEqual StatusCodes.BadRequest + request ~> emit(responseWithSourceError[ResourceRejection](BlankResourceId)) ~> check { + status shouldEqual StatusCodes.BadRequest // BlankResourceId is supposed to result in BadRequest contentType.mediaType shouldEqual `application/ld+json` response.asJson shouldEqual expectedBlankIdErrorResponse } From 33ef89b08ffd781a1f8f8b7919282ef855030046 Mon Sep 17 00:00:00 2001 From: Daniel Bell Date: Tue, 10 Oct 2023 10:41:16 +0200 Subject: [PATCH 6/6] add CatsResponseToJsonLdSpec, fix in that too --- .../delta/sdk/ce/CatsResponseToJsonLd.scala | 9 +- .../sdk/ce/CatsResponseToJsonLdSpec.scala | 97 +++++++++++++++++++ 2 files changed, 100 insertions(+), 6 deletions(-) create mode 100644 delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLdSpec.scala diff --git a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLd.scala b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLd.scala index fcbbde44b2..de2f33fd3d 100644 --- a/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLd.scala +++ b/delta/sdk/src/main/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLd.scala @@ -116,12 +116,9 @@ object CatsResponseToJsonLd extends FileBytesInstances { } onSuccess(flattened.unsafeToFuture()) { - case Left(complete: Complete[E]) => emit(complete) - case Left(reject: Reject[E]) => emit(reject) - case Right(Left(c)) => - implicit val valueEncoder = c.value.encoder - emit(c.value.value) - + case Left(complete: Complete[E]) => emit(complete) + case Left(reject: Reject[E]) => emit(reject) + case Right(Left(c)) => emit(c) case Right(Right((metadata, content))) => headerValueByType(Accept) { accept => if (accept.mediaRanges.exists(_.matches(metadata.contentType.mediaType))) { diff --git a/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLdSpec.scala b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLdSpec.scala new file mode 100644 index 0000000000..6d2758a72e --- /dev/null +++ b/delta/sdk/src/test/scala/ch/epfl/bluebrain/nexus/delta/sdk/ce/CatsResponseToJsonLdSpec.scala @@ -0,0 +1,97 @@ +package ch.epfl.bluebrain.nexus.delta.sdk.ce + +import akka.http.scaladsl.model.ContentTypes.`text/plain(UTF-8)` +import akka.http.scaladsl.model.MediaRanges.`*/*` +import akka.http.scaladsl.model.headers.Accept +import akka.http.scaladsl.model.{ContentType, StatusCodes} +import akka.http.scaladsl.server.RouteConcatenation +import akka.stream.scaladsl.Source +import akka.util.ByteString +import ch.epfl.bluebrain.nexus.delta.rdf.RdfMediaTypes.`application/ld+json` +import ch.epfl.bluebrain.nexus.delta.rdf.Vocabulary.contexts +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.context.RemoteContextResolution +import ch.epfl.bluebrain.nexus.delta.rdf.jsonld.encoder.JsonLdEncoder +import ch.epfl.bluebrain.nexus.delta.rdf.syntax.JsonSyntax +import ch.epfl.bluebrain.nexus.delta.rdf.utils.JsonKeyOrdering +import ch.epfl.bluebrain.nexus.delta.sdk.ce.DeltaDirectives._ +import ch.epfl.bluebrain.nexus.delta.sdk.directives.FileResponse +import ch.epfl.bluebrain.nexus.delta.sdk.marshalling.HttpResponseFields +import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection +import ch.epfl.bluebrain.nexus.delta.sdk.resources.model.ResourceRejection.BlankResourceId +import ch.epfl.bluebrain.nexus.delta.sdk.utils.RouteHelpers +import ch.epfl.bluebrain.nexus.delta.sdk.{AkkaSource, SimpleRejection, SimpleResource} +import ch.epfl.bluebrain.nexus.testkit.ShouldMatchers.convertToAnyShouldWrapper +import ch.epfl.bluebrain.nexus.testkit.TestHelpers.jsonContentOf +import monix.bio.IO +import cats.effect.{IO => CatsIO} +import monix.execution.Scheduler + +class CatsResponseToJsonLdSpec extends RouteHelpers with JsonSyntax with RouteConcatenation { + + implicit val s: Scheduler = Scheduler.global + implicit val rcr: RemoteContextResolution = + RemoteContextResolution.fixed( + SimpleResource.contextIri -> SimpleResource.context, + SimpleRejection.contextIri -> SimpleRejection.context, + contexts.error -> jsonContentOf("/contexts/error.json").topContextValueOrEmpty + ) + implicit val jo: JsonKeyOrdering = JsonKeyOrdering.default() + + private def responseWithSourceError[E: JsonLdEncoder: HttpResponseFields](error: E) = { + responseWith( + `text/plain(UTF-8)`, + IO.raiseError(error) + ) + } + + private val expectedBlankIdErrorResponse = jsonContentOf( + "/directives/blank-id.json" + ) + + private val FileContents = "hello" + + private def fileSourceOfString(value: String) = { + IO.pure(Source.single(ByteString(value))) + } + + private def responseWith[E: JsonLdEncoder: HttpResponseFields]( + contentType: ContentType, + contents: IO[E, AkkaSource] + ) = { + CatsIO.pure( + Right( + FileResponse( + "file.name", + contentType, + 1024, + contents + ) + ) + ) + } + + private def request = { + Get() ~> Accept(`*/*`) + } + + "ResponseToJsonLd file handling" should { + + "Return the contents of a file" in { + request ~> emit( + responseWith(`text/plain(UTF-8)`, fileSourceOfString(FileContents)) + ) ~> check { + status shouldEqual StatusCodes.OK + contentType shouldEqual `text/plain(UTF-8)` + response.asString shouldEqual FileContents + } + } + + "Return an error from a file content IO" in { + request ~> emit(responseWithSourceError[ResourceRejection](BlankResourceId)) ~> check { + status shouldEqual StatusCodes.BadRequest // BlankResourceId is supposed to result in BadRequest + contentType.mediaType shouldEqual `application/ld+json` + response.asJson shouldEqual expectedBlankIdErrorResponse + } + } + } +}