All Downloads are FREE. Search and download functionalities are using the official Maven repository.

harness.docker.DockerCommands.scala Maven / Gradle / Ivy

There is a newer version: 3.1.14
Show newest version
package harness.docker

import cats.data.NonEmptyList
import cats.syntax.either.*
import cats.syntax.parallel.*
import cats.syntax.traverse.*
import harness.core.*
import harness.zio.*
import java.time.ZonedDateTime
import java.time.format.DateTimeFormatter
import scala.util.Try
import zio.*
import zio.json.*

object DockerCommands {

  private val portMappingRegex = ":::(\\d+)->(\\d+)/[a-z]+".r

  final case class Port(PublicPort: Int, PrivatePort: Int) {
    override def toString: String = s"$PublicPort->$PrivatePort"
  }
  implicit val portsDecoder: JsonDecoder[List[Port]] =
    JsonDecoder.string.map {
      _.split(", ").toList
        .filter(_.nonEmpty)
        .collect { case portMappingRegex(publicPort, privatePort) => Port(publicPort.toInt, privatePort.toInt) }
    }

  private implicit val formatter: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss Z z")
  implicit val zdtDecoder: JsonDecoder[ZonedDateTime] =
    JsonDecoder.string.mapOrFail { string =>
      Try { ZonedDateTime.parse(string, formatter) }.toEither.leftMap(_.getMessage)
    }

  final case class PsContainer(
      ID: String,
      Names: String, // TODO (KR) :
      Image: String,
      Command: String,
      CreatedAt: ZonedDateTime,
      Ports: List[Port],
      // Labels: Map[String, String],
      State: String,
      Status: String,
  )
  object PsContainer {
    implicit val jsonCodec: JsonDecoder[PsContainer] = DeriveJsonDecoder.gen[PsContainer]
  }

  val getContainers: HRIO[DockerNeedsSudo & Logger, List[PsContainer]] =
    for {
      _ <- Logger.log.info("Getting docker containers")
      dockerNeedsSudo <- DockerNeedsSudo.value
      dockerPsCommand = Sys.Command("docker", "ps", "-a", "--format", "{{json .}}").sudoIf(dockerNeedsSudo)
      sysString <- Sys.executeString0(dockerPsCommand).map(_.replace("\\u003e", ">"))
      containers <- ZIO.eitherNelToInternalDefects(sysString.split('\n').toList.filter(_.nonEmpty).parTraverse(_.fromJson[PsContainer].toEitherNel))
      _ <- Logger.log.debug(s"Containers:${containers.map { c => s"\n  - ${c.Names} (${c.State}) : ${c.Ports.mkString(", ")}" }.mkString}")
    } yield containers

  def stopAndRemoveContainer(container: PsContainer): HRIO[DockerNeedsSudo & Logger, Unit] =
    DockerNeedsSudo.value.flatMap { dockerNeedsSudo =>
      val stop =
        Logger.log.important(s"Stopping container '${container.Names}'") *>
          Sys.executeString0(Sys.Command("docker", "stop", container.Names).sudoIf(dockerNeedsSudo))
      val rm =
        Logger.log.important(s"Removing container '${container.Names}'") *>
          Sys.executeString0(Sys.Command("docker", "rm", container.Names).sudoIf(dockerNeedsSudo)).unit

      Logger.log.info(s"Found existing container '${container.Names}'") *>
        (container.State match {
          case "running" | "created" => stop *> rm
          case "exited"              => rm
          case state                 => ZIO.fail(HError.SystemFailure(s"Unable to act on unknown state '$state' for container '${container.Names}'"))
        })
    }

  private def generateDockerComposeFile(containers: List[DockerContainer]): HRIO[FileSystem & Logger & Scope, Unit] =
    for {
      path <- Path("docker-compose.yml")
      _ <- ZIO.fail(HError.UserError("'docker-compose.yml' already exists")).whenZIO(path.exists)
      yamlString =
        IndentedString
          .inline(
            "version: '3'",
            "services:",
            IndentedString.indented(
              containers.map(_.toYamlString),
            ),
          )
          .toString("  ") + "\n"
      _ <- Logger.log.debug(s"Yaml string:\n\n$yamlString")
      _ <- path.writeString(yamlString).withFinalizer(_ => path.delete.orDie)
    } yield ()

  def runDockerCompose(command: String, containers: List[DockerContainer]): HRIO[DockerAppName & DockerNeedsSudo & FileSystem & Logger, Unit] =
    ZIO.scoped {
      for {
        dockerAppName <- DockerAppName.value
        _ <- Logger.log.info(s"Running docker compose for '$dockerAppName'")
        _ <- generateDockerComposeFile(containers)
        dockerNeedsSudo <- DockerNeedsSudo.value
        _ <- Sys.execute0(Sys.Command("docker", "compose", command, "-d").sudoIf(dockerNeedsSudo))
      } yield ()
    }

  def runContainer(container: DockerContainer): HRIO[DockerNeedsSudo & FileSystem & Logger, Unit] =
    for {
      _ <- Logger.log.info(s"Running docker container '${container.name}'")
      dockerNeedsSudo <- DockerNeedsSudo.value
      _ <- ZIO.foreachDiscard(container.volumeMappings) { mapping =>
        Path(mapping.host).flatMap { path => path.mkdirs.unlessZIO(path.exists) }
      }
      cmd = Sys.Command(
        NonEmptyList(
          "docker",
          List(
            List("run", "--name", container.name),
            container.hostName.toList.flatMap { hostName => List("--hostname", hostName) },
            container.envVars.flatMap(_.toArgs),
            container.portMappings.flatMap(_.toArgs("-p")),
            container.volumeMappings.flatMap(_.toArgs("-v")),
            List("-d", s"${container.image}:${container.imageVersion}"),
          ).flatten,
        ),
      )
      _ <- Sys.executeString0(cmd.sudoIf(dockerNeedsSudo))
    } yield ()

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy