harness.docker.DockerCommands.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of harness-docker_3 Show documentation
Show all versions of harness-docker_3 Show documentation
Miscellaneous libraries/utilities for Scala.
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