From e31d1df5e74edcd93084997bfa71fdd6e7ad2e1d Mon Sep 17 00:00:00 2001 From: Pawel Batko Date: Wed, 20 Oct 2021 08:38:31 +0200 Subject: [PATCH 1/2] [DPP-620] Adop self-service error codes in time service CHANGELOG_BEGIN CHANGELOG_END --- .../platform/apiserver/ApiServices.scala | 5 ++- .../apiserver/services/ApiTimeService.scala | 42 +++++++++++++------ 2 files changed, 34 insertions(+), 13 deletions(-) diff --git a/ledger/participant-integration-api/src/main/scala/platform/apiserver/ApiServices.scala b/ledger/participant-integration-api/src/main/scala/platform/apiserver/ApiServices.scala index d3ccb7ce8593..847836c4eb9a 100644 --- a/ledger/participant-integration-api/src/main/scala/platform/apiserver/ApiServices.scala +++ b/ledger/participant-integration-api/src/main/scala/platform/apiserver/ApiServices.scala @@ -169,7 +169,10 @@ private[daml] object ApiServices { val apiTimeServiceOpt = optTimeServiceBackend.map(tsb => - new TimeServiceAuthorization(ApiTimeService.create(ledgerId, tsb), authorizer) + new TimeServiceAuthorization( + ApiTimeService.create(ledgerId, tsb, errorsVersionsSwitcher), + authorizer, + ) ) val writeServiceBackedApiServices = intitializeWriteServiceBackedApiServices( diff --git a/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala b/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala index e176ff2ad766..ebe954c392ae 100644 --- a/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala +++ b/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala @@ -7,7 +7,11 @@ import akka.NotUsed import akka.stream.Materializer import akka.stream.scaladsl.Source import com.daml.api.util.TimestampConversion._ -import com.daml.error.{ContextualizedErrorLogger, DamlContextualizedErrorLogger} +import com.daml.error.{ + ContextualizedErrorLogger, + DamlContextualizedErrorLogger, + ErrorCodesVersionSwitcher, +} import com.daml.grpc.adapter.ExecutionSequencerFactory import com.daml.ledger.api.domain.LedgerId import com.daml.ledger.api.v1.testing.time_service.TimeServiceGrpc.TimeService @@ -17,8 +21,7 @@ import com.daml.platform.akkastreams.dispatcher.SignalDispatcher import com.daml.platform.api.grpc.GrpcApiService import com.daml.platform.apiserver.TimeServiceBackend import com.daml.platform.server.api.ValidationLogger -import com.daml.platform.server.api.validation.ErrorFactories -import com.daml.platform.server.api.validation.FieldValidations._ +import com.daml.platform.server.api.validation.{ErrorFactories, FieldValidations} import com.google.protobuf.empty.Empty import io.grpc.{ServerServiceDefinition, StatusRuntimeException} import scalaz.syntax.tag._ @@ -29,6 +32,7 @@ import scala.concurrent.{ExecutionContext, Future} private[apiserver] final class ApiTimeService private ( val ledgerId: LedgerId, backend: TimeServiceBackend, + errorCodesVersionSwitcher: ErrorCodesVersionSwitcher, )(implicit protected val mat: Materializer, protected val esf: ExecutionSequencerFactory, @@ -45,10 +49,17 @@ private[apiserver] final class ApiTimeService private ( s"${getClass.getSimpleName} initialized with ledger ID ${ledgerId.unwrap}, start time ${backend.getCurrentTime}" ) + private val errorFactories = ErrorFactories(errorCodesVersionSwitcher) + private val fieldValidation = FieldValidations(errorFactories) + private val dispatcher = SignalDispatcher[Instant]() - override protected def getTimeSource(request: GetTimeRequest): Source[GetTimeResponse, NotUsed] = - matchLedgerId(ledgerId)(LedgerId(request.ledgerId)).fold( + override protected def getTimeSource( + request: GetTimeRequest + ): Source[GetTimeResponse, NotUsed] = { + val validated = + fieldValidation.matchLedgerId(ledgerId = ledgerId)(received = LedgerId(request.ledgerId)) + validated.fold( t => Source.failed(ValidationLogger.logFailureWithContext(request, t)), { ledgerId => logger.info(s"Received request for time with ledger ID $ledgerId") @@ -66,6 +77,7 @@ private[apiserver] final class ApiTimeService private ( .via(logger.logErrorsOnStream) }, ) + } @SuppressWarnings(Array("org.wartremover.warts.JavaSerializable")) override def setTime(request: SetTimeRequest): Future[Empty] = { @@ -80,7 +92,7 @@ private[apiserver] final class ApiTimeService private ( if (success) Right(requestedTime) else Left( - ErrorFactories.invalidArgument(None)( + errorFactories.invalidArgument(None)( s"current_time mismatch. Provided: $expectedTime. Actual: ${backend.getCurrentTime}" ) ) @@ -88,15 +100,17 @@ private[apiserver] final class ApiTimeService private ( } val result = for { - _ <- matchLedgerId(ledgerId)(LedgerId(request.ledgerId)) - expectedTime <- requirePresence(request.currentTime, "current_time").map(toInstant) - requestedTime <- requirePresence(request.newTime, "new_time").map(toInstant) + _ <- fieldValidation.matchLedgerId(ledgerId)(LedgerId(request.ledgerId)) + expectedTime <- fieldValidation + .requirePresence(request.currentTime, "current_time") + .map(toInstant) + requestedTime <- fieldValidation.requirePresence(request.newTime, "new_time").map(toInstant) _ <- { if (!requestedTime.isBefore(expectedTime)) Right(()) else Left( - ErrorFactories.invalidArgument(None)( + errorFactories.invalidArgument(None)( s"new_time [$requestedTime] is before current_time [$expectedTime]. Setting time backwards is not allowed." ) ) @@ -131,11 +145,15 @@ private[apiserver] final class ApiTimeService private ( } private[apiserver] object ApiTimeService { - def create(ledgerId: LedgerId, backend: TimeServiceBackend)(implicit + def create( + ledgerId: LedgerId, + backend: TimeServiceBackend, + errorCodesVersionSwitcher: ErrorCodesVersionSwitcher, + )(implicit mat: Materializer, esf: ExecutionSequencerFactory, executionContext: ExecutionContext, loggingContext: LoggingContext, ): TimeService with GrpcApiService = - new ApiTimeService(ledgerId, backend) + new ApiTimeService(ledgerId, backend, errorCodesVersionSwitcher) } From ec105968750db64227b4d53f6da3a37472265f08 Mon Sep 17 00:00:00 2001 From: Pawel Batko Date: Wed, 20 Oct 2021 13:24:03 +0200 Subject: [PATCH 2/2] TV review --- .../apiserver/services/ApiTimeService.scala | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala b/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala index ebe954c392ae..1c66ef5580fe 100644 --- a/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala +++ b/ledger/participant-integration-api/src/main/scala/platform/apiserver/services/ApiTimeService.scala @@ -45,20 +45,20 @@ private[apiserver] final class ApiTimeService private ( private implicit val contextualizedErrorLogger: ContextualizedErrorLogger = new DamlContextualizedErrorLogger(logger, loggingContext, None) + private val errorFactories = ErrorFactories(errorCodesVersionSwitcher) + private val fieldValidations = FieldValidations(errorFactories) + logger.debug( s"${getClass.getSimpleName} initialized with ledger ID ${ledgerId.unwrap}, start time ${backend.getCurrentTime}" ) - private val errorFactories = ErrorFactories(errorCodesVersionSwitcher) - private val fieldValidation = FieldValidations(errorFactories) - private val dispatcher = SignalDispatcher[Instant]() override protected def getTimeSource( request: GetTimeRequest ): Source[GetTimeResponse, NotUsed] = { val validated = - fieldValidation.matchLedgerId(ledgerId = ledgerId)(received = LedgerId(request.ledgerId)) + fieldValidations.matchLedgerId(ledgerId)(LedgerId(request.ledgerId)) validated.fold( t => Source.failed(ValidationLogger.logFailureWithContext(request, t)), { ledgerId => @@ -100,11 +100,11 @@ private[apiserver] final class ApiTimeService private ( } val result = for { - _ <- fieldValidation.matchLedgerId(ledgerId)(LedgerId(request.ledgerId)) - expectedTime <- fieldValidation + _ <- fieldValidations.matchLedgerId(ledgerId)(LedgerId(request.ledgerId)) + expectedTime <- fieldValidations .requirePresence(request.currentTime, "current_time") .map(toInstant) - requestedTime <- fieldValidation.requirePresence(request.newTime, "new_time").map(toInstant) + requestedTime <- fieldValidations.requirePresence(request.newTime, "new_time").map(toInstant) _ <- { if (!requestedTime.isBefore(expectedTime)) Right(())