diff --git a/core/src/main/resources/reference.conf b/core/src/main/resources/reference.conf index 34d70f8ba..b25a4a042 100644 --- a/core/src/main/resources/reference.conf +++ b/core/src/main/resources/reference.conf @@ -98,6 +98,14 @@ bodyReadTimeout = 500 millis } + debug { + http { + enable = false + logHeaders = true + logBody = false + redactHeaders = [] + } + } enableDefaultRedirect = false preTerminationPeriod = 10 seconds diff --git a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Config.scala b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Config.scala index e09577b8c..ab5e06cca 100644 --- a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Config.scala +++ b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Config.scala @@ -41,7 +41,8 @@ case class Config[+SinkConfig]( enableDefaultRedirect: Boolean, redirectDomains: Set[String], preTerminationPeriod: FiniteDuration, - license: Config.License + license: Config.License, + debug: Config.Debug.Debug ) object Config { @@ -167,6 +168,11 @@ object Config { accept: Boolean ) + object Debug { + case class Http(enable: Boolean, logHeaders: Boolean, logBody: Boolean, redactHeaders: List[String]) + case class Debug(http: Http) + } + implicit def decoder[SinkConfig: Decoder]: Decoder[Config[SinkConfig]] = { implicit val license: Decoder[License] = { val truthy = Set("true", "yes", "on", "1") @@ -199,6 +205,8 @@ object Config { implicit val hsts = deriveDecoder[HSTS] implicit val telemetry = deriveDecoder[Telemetry] implicit val networking = deriveDecoder[Networking] + implicit val http = deriveDecoder[Debug.Http] + implicit val debug = deriveDecoder[Debug.Debug] implicit val sinkConfig = newDecoder[SinkConfig].or(legacyDecoder[SinkConfig]) implicit val streams = deriveDecoder[Streams[SinkConfig]] diff --git a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/HttpServer.scala b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/HttpServer.scala index 83b154247..dc26b180d 100644 --- a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/HttpServer.scala +++ b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/HttpServer.scala @@ -19,7 +19,8 @@ import org.http4s.{HttpApp, HttpRoutes} import org.http4s.blaze.server.BlazeServerBuilder import org.http4s.headers.`Strict-Transport-Security` import org.http4s.server.Server -import org.http4s.server.middleware.{HSTS, Metrics} +import org.http4s.server.middleware.{HSTS, Logger => LoggerMiddleware, Metrics, Timeout} +import org.typelevel.ci.CIString import org.typelevel.log4cats.Logger import org.typelevel.log4cats.slf4j.Slf4jLogger @@ -36,11 +37,12 @@ object HttpServer { secure: Boolean, hsts: Config.HSTS, networking: Config.Networking, - metricsConfig: Config.Metrics + metricsConfig: Config.Metrics, + debugHttp: Config.Debug.Http ): Resource[F, Server] = for { withMetricsMiddleware <- createMetricsMiddleware(routes, metricsConfig) - server <- buildBlazeServer[F](withMetricsMiddleware, port, secure, hsts, networking) + server <- buildBlazeServer[F](withMetricsMiddleware, port, secure, hsts, networking, debugHttp) } yield server private def createMetricsMiddleware[F[_]: Async]( @@ -67,17 +69,33 @@ object HttpServer { HSTS(routes, `Strict-Transport-Security`.unsafeFromDuration(hsts.maxAge)) else routes + private def loggerMiddleware[F[_]: Async](routes: HttpApp[F], config: Config.Debug.Http): HttpApp[F] = + if (config.enable) { + LoggerMiddleware.httpApp[F]( + logHeaders = config.logHeaders, + logBody = config.logBody, + redactHeadersWhen = config.redactHeaders.map(CIString(_)).contains(_), + logAction = Some((msg: String) => Logger[F].debug(msg)) + )(routes) + } else routes + + private def timeoutMiddleware[F[_]: Async](routes: HttpApp[F], networking: Config.Networking): HttpApp[F] = + Timeout.httpApp[F](timeout = networking.responseHeaderTimeout)(routes) + private def buildBlazeServer[F[_]: Async]( routes: HttpRoutes[F], port: Int, secure: Boolean, hsts: Config.HSTS, - networking: Config.Networking + networking: Config.Networking, + debugHttp: Config.Debug.Http ): Resource[F, Server] = Resource.eval(Logger[F].info("Building blaze server")) >> BlazeServerBuilder[F] .bindSocketAddress(new InetSocketAddress(port)) - .withHttpApp(hstsMiddleware(hsts, routes.orNotFound)) + .withHttpApp( + loggerMiddleware(timeoutMiddleware(hstsMiddleware(hsts, routes.orNotFound), networking), debugHttp) + ) .withIdleTimeout(networking.idleTimeout) .withMaxConnections(networking.maxConnections) .withResponseHeaderTimeout(networking.responseHeaderTimeout) diff --git a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Run.scala b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Run.scala index 24765102f..4553955ff 100644 --- a/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Run.scala +++ b/core/src/main/scala/com.snowplowanalytics.snowplow.collector.core/Run.scala @@ -105,7 +105,8 @@ object Run { config.ssl.enable, config.hsts, config.networking, - config.monitoring.metrics + config.monitoring.metrics, + config.debug.http ) _ <- withGracefulShutdown(config.preTerminationPeriod)(httpServer) httpClient <- BlazeClientBuilder[F].resource diff --git a/core/src/test/scala/com.snowplowanalytics.snowplow.collector.core/TestUtils.scala b/core/src/test/scala/com.snowplowanalytics.snowplow.collector.core/TestUtils.scala index 92fd420f2..aade5a76e 100644 --- a/core/src/test/scala/com.snowplowanalytics.snowplow.collector.core/TestUtils.scala +++ b/core/src/test/scala/com.snowplowanalytics.snowplow.collector.core/TestUtils.scala @@ -123,6 +123,9 @@ object TestUtils { 2.second, 500.millis ), + debug = Debug.Debug( + http = Debug.Http(enable = false, logHeaders = true, logBody = false, redactHeaders = List.empty) + ), enableDefaultRedirect = false, redirectDomains = Set.empty[String], preTerminationPeriod = 10.seconds,