Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

TDR-3326 Assign references to file/directories #705

Merged
merged 12 commits into from
Oct 12, 2023
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,7 @@ libraryDependencies ++= Seq(
"io.circe" %% "circe-optics" % "0.15.0",
"io.circe" %% "circe-generic" % circeVersion,
"io.circe" %% "circe-generic-extras" % "0.14.3",
"com.softwaremill.sttp.client3" %% "core" % "3.9.0",
"uk.gov.nationalarchives" %% "consignment-api-db" % "0.1.36",
"org.postgresql" % "postgresql" % "42.6.0",
"com.typesafe.slick" %% "slick" % "3.4.0",
Expand Down
8 changes: 8 additions & 0 deletions src/main/resources/application.base.conf
Original file line number Diff line number Diff line change
Expand Up @@ -33,8 +33,16 @@ fileUpload {

featureAccessBlock {
http4s = ${BLOCK_HTTP4S}
assignFileReferences = ${BLOCK_ASSIGN_FILE_REFERENCES}
}

referenceGenerator {
referenceGeneratorUrl = ${REFERENCE_GENERATOR_URL}
referenceLimit = ${REFERENCE_GENERATOR_LIMIT}
}

thanhz marked this conversation as resolved.
Show resolved Hide resolved
environment = ${ENVIRONMENT}

akka.http {
server {
idle-timeout = 180 s
Expand Down
8 changes: 8 additions & 0 deletions src/main/resources/application.conf
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,16 @@ frontend = {

featureAccessBlock {
http4s = true
assignFileReferences = true
TomJKing marked this conversation as resolved.
Show resolved Hide resolved
}

referenceGenerator {
referenceGeneratorUrl = "https://q59yzinibd.execute-api.eu-west-2.amazonaws.com"
TomJKing marked this conversation as resolved.
Show resolved Hide resolved
referenceLimit = 500
}

environment = "intg"

consignmentapi = {
useIamAuth = false
db {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import com.typesafe.scalalogging.Logger
import sangria.execution._
import sangria.marshalling.ResultMarshaller
import slick.jdbc.JdbcBackend
import sttp.client3.SimpleHttpClient
import uk.gov.nationalarchives.tdr.api.auth.AuthorisationException
import uk.gov.nationalarchives.tdr.api.consignmentstatevalidation.ConsignmentStateException
import uk.gov.nationalarchives.tdr.api.db.repository._
Expand Down Expand Up @@ -83,14 +84,15 @@ trait GraphQLServerBase {
new FFIDMetadataService(ffidMetadataRepository, ffidMetadataMatchesRepository, timeSource, uuidSource)
val fileStatusService = new FileStatusService(fileStatusRepository)
val displayPropertiesService = new DisplayPropertiesService(displayPropertiesRepository)
val referenceGeneratorService = new ReferenceGeneratorService(config, SimpleHttpClient())
val fileService = new FileService(
fileRepository,
consignmentRepository,
customMetadataPropertiesRepository,
ffidMetadataService,
antivirusMetadataService,
fileStatusService,
fileMetadataService,
referenceGeneratorService,
new CurrentTimeSource,
uuidSource,
config
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,12 +28,12 @@ import scala.math.min

class FileService(
fileRepository: FileRepository,
consignmentRepository: ConsignmentRepository,
customMetadataPropertiesRepository: CustomMetadataPropertiesRepository,
ffidMetadataService: FFIDMetadataService,
avMetadataService: AntivirusMetadataService,
TomJKing marked this conversation as resolved.
Show resolved Hide resolved
fileStatusService: FileStatusService,
fileMetadataService: FileMetadataService,
referenceGeneratorService: ReferenceGeneratorService,
timeSource: TimeSource,
uuidSource: UUIDSource,
config: Config
Expand All @@ -43,6 +43,7 @@ class FileService(

private val fileUploadBatchSize: Int = config.getInt("fileUpload.batchSize")
private val filePageMaxLimit: Int = config.getInt("pagination.filesMaxLimit")
private val referenceGeneratorFeatureBlock: Boolean = config.getBoolean("featureAccessBlock.assignFileReferences")

def addFile(addFileAndMetadataInput: AddFileAndMetadataInput, userId: UUID): Future[List[FileMatches]] = {
val now = Timestamp.from(timeSource.now)
Expand All @@ -53,10 +54,26 @@ class FileService(

val row: (UUID, String, String) => FilemetadataRow = FilemetadataRow(uuidSource.uuid, _, _, now, userId, _)
val rows: Future[List[Rows]] = customMetadataPropertiesRepository.getCustomMetadataValuesWithDefault.map(filePropertyValue => {
((allEmptyDirectoryNodes ++ allFileNodes) map { case (path, treeNode) =>
val allNodes = allEmptyDirectoryNodes ++ allFileNodes
val allNodesWithReference = if (referenceGeneratorFeatureBlock) {
allNodes.keys.zip(allNodes.values.zip(List.fill(allNodes.size)(None))).toMap
} else {
val generatedReferences = referenceGeneratorService.getReferences(allNodes.size)
allNodes.keys.zip(allNodes.values.zip(generatedReferences.map(Some(_)))).toMap
}
(allNodesWithReference map { case (path, (treeNode, reference)) =>
val parentId = treeNode.parentPath.map(path => allFileNodes.getOrElse(path, allEmptyDirectoryNodes(path)).id)
val fileId = treeNode.id
val fileRow = FileRow(fileId, consignmentId, userId, now, filetype = Some(treeNode.treeNodeType), filename = Some(treeNode.name), parentid = parentId)
val fileRow = FileRow(
fileId,
consignmentId,
userId,
now,
filetype = Some(treeNode.treeNodeType),
filename = Some(treeNode.name),
parentid = parentId,
filereference = reference
)

val commonMetadataRows = List(
row(fileId, path, ClientSideOriginalFilepath),
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
package uk.gov.nationalarchives.tdr.api.service

import com.typesafe.config.Config
import com.typesafe.scalalogging.Logger
import io.circe.parser._
import org.apache.http.client.HttpResponseException
import sttp.client3.{Response, SimpleHttpClient, UriContext, basicRequest}
import uk.gov.nationalarchives.tdr.api.service.ReferenceGeneratorService.reference

import scala.annotation.tailrec

class ReferenceGeneratorService(config: Config, client: SimpleHttpClient) {
val logger: Logger = Logger("ReferenceGeneratorService")
val environment: String = config.getString("environment")
thanhz marked this conversation as resolved.
Show resolved Hide resolved
val refGeneratorUrl: String = config.getString("referenceGenerator.referenceGeneratorUrl")
val refGeneratorLimit: Int = config.getInt("referenceGenerator.referenceLimit")

def getReferences(numberOfRefs: Int): List[reference] = {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would abstract out some of this code from the getReferences method, to make it a little clearer, ie so that it is doing one thing.

Some thing like this:

...
  def getReferences(numberOfRefs: Int): List[reference] = {
    @tailrec
    def fetchReferences(numberOfRefs: Int, acc: List[reference]): List[reference] = {
      if (numberOfRefs <= 0) acc
      else {
        val batchSize = Math.min(numberOfRefs, refGeneratorLimit)
        fetchReferences(numberOfRefs - batchSize, acc ++ sendRequest(batchSize))
      }
    }

    fetchReferences(numberOfRefs, Nil)
  }

  private def sendRequest(numberOfRefs: Int): List[reference] = {
    val response = client.send(basicRequest.get(uri"$refGeneratorUrl/$environment/counter?numberofrefs=$numberOfRefs"))

    try {
      response.body match {
        case Left(body) =>
          val message = "Failed to get references from reference generator api"
          logger.error(s"${response.code} $message:\n$body")
          throw new HttpResponseException(response.code.code, message)
        case Right(body) =>
          val references = parse(body).getOrElse(null)
          val listOfReferences = references.as[List[reference]].getOrElse(null)
          logger.info(s"2xx response to GET:\n$listOfReferences")
          listOfReferences
      }
    } finally {
      client.close()
    }
  }
...

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is much cleaner. I've made the changes.


def processResponse(response: Response[Either[String, reference]]): List[reference] = {
try {
response.body match {
case Left(body) =>
val message = "Failed to get references from reference generator api"
logger.error(s"${response.code} $message:\n$body")
throw new HttpResponseException(response.code.code, message)
case Right(body) =>
val references = parse(body).getOrElse(null)
val listOfReferences = references.as[List[reference]].getOrElse(null)
logger.info(s"2xx response to GET:\n$listOfReferences")
listOfReferences
}
} finally {
client.close()
}
}

@tailrec
def fetchReferences(numberOfRefs: Int, acc: List[reference]): List[reference] = {
if (numberOfRefs <= 0) acc
else {
val batchSize = Math.min(numberOfRefs, refGeneratorLimit)
val response: Response[Either[String, reference]] = client.send(basicRequest.get(uri"$refGeneratorUrl/$environment/counter?numberofrefs=$batchSize"))
fetchReferences(numberOfRefs - batchSize, acc ++ processResponse(response))
}
}

if (numberOfRefs > refGeneratorLimit) {
fetchReferences(numberOfRefs, Nil)
} else {
val response: Response[Either[String, reference]] = client.send(basicRequest.get(uri"$refGeneratorUrl/$environment/counter?numberofrefs=$numberOfRefs"))
processResponse(response)
}
}
}

object ReferenceGeneratorService {
type reference = String
thanhz marked this conversation as resolved.
Show resolved Hide resolved
}
8 changes: 8 additions & 0 deletions src/test/resources/application.conf
Original file line number Diff line number Diff line change
Expand Up @@ -36,4 +36,12 @@ fileUpload {

featureAccessBlock {
http4s = true
assignFileReferences = true
}

referenceGenerator {
referenceGeneratorUrl = "https://dummy-reference-url.com"
referenceLimit = 2
}

environment = "intg"
Loading