Skip to content

Commit

Permalink
Merge pull request #4 from janschultecom/bumb-versions
Browse files Browse the repository at this point in the history
Update cats, cats-effect and http4s versions
  • Loading branch information
rossabaker authored Feb 22, 2019
2 parents 591fe92 + fd25aeb commit 5c9b9d0
Show file tree
Hide file tree
Showing 5 changed files with 95 additions and 95 deletions.
6 changes: 1 addition & 5 deletions core/build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -21,11 +21,7 @@ enablePlugins(MetadataPlugin, ScalaTestPlugin)
libraryDependencies ++= Seq(
dependencies.simulacrum.core,
"org.typelevel" %% "cats-core" % V.cats,
"org.typelevel" %% "cats-effect" % V.catsEffect,
"org.typelevel" %% "cats-free" % V.cats,
"io.verizon.knobs" %% "core" % V.knobs,
"io.verizon.journal" %% "core" % V.journal,
"org.typelevel" %% "cats-kernel-laws" % V.cats % "test"
"org.typelevel" %% "cats-free" % V.cats
)

addCompilerPlugin(dependencies.kindprojector.plugin)
Expand Down
7 changes: 3 additions & 4 deletions http4s/build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -20,15 +20,14 @@ enablePlugins(MetadataPlugin, ScalaTestPlugin)

libraryDependencies ++= Seq(
dependencies.simulacrum.core,
"io.verizon.knobs" %% "core" % V.knobs,
"io.verizon.journal" %% "core" % V.journal,
"io.argonaut" %% "argonaut" % V.argonaut,
"io.argonaut" %% "argonaut-cats" % V.argonaut,
"org.http4s" %% "http4s-argonaut" % V.http4s,
"org.http4s" %% "http4s-blaze-client" % V.http4s,
"io.verizon.knobs" %% "core" % V.knobs,
"io.verizon.journal" %% "core" % V.journal,
"com.whisk" %% "docker-testkit-scalatest" % V.dockerit % "test",
"com.whisk" %% "docker-testkit-impl-spotify" % V.dockerit % "test",
"org.typelevel" %% "cats-kernel-laws" % V.cats % "test"
"com.whisk" %% "docker-testkit-impl-spotify" % V.dockerit % "test"
)

addCompilerPlugin(dependencies.kindprojector.plugin)
Expand Down
44 changes: 19 additions & 25 deletions http4s/src/main/scala/platypus/http4s/Http4sVault.scala
Original file line number Diff line number Diff line change
Expand Up @@ -57,58 +57,52 @@ final class Http4sVaultClient(authToken: Token,

val addCreds: Request[IO] => Request[IO] = _.putHeaders(Header("X-Vault-Token", authToken.value))

def req[T: DecodeJson](req: IO[Request[IO]]): IO[T] =
client.fetch(req.map(addCreds)){
def req[T: DecodeJson](req: Request[IO]): IO[T] =
client.fetch(addCreds(req)){
case Ok(resp) => resp.as(FlatMap[IO], jsonOf[IO, T])
case resp =>
(for {
r <- req
body <- resp.as[String]
} yield {
val msg = s"unexpected status: ${resp.status} from request: ${r.pathInfo}, msg: ${body}"
resp.as[String].flatMap(body => {
val msg = s"unexpected status: ${resp.status} from request: ${req.pathInfo}, msg: ${body}"
IO.raiseError(new RuntimeException(msg))
}).flatMap(identity)
})
}

def reqVoid(req: IO[Request[IO]]): IO[Unit] =
client.fetch(req.map(addCreds)) {
def reqVoid(req: Request[IO]): IO[Unit] =
client.fetch(addCreds(req)) {
case NoContent(_) => IO.pure(())
case resp =>
(for {
r <- req
body <- resp.as[String]
} yield {
val msg = s"unexpected status: ${resp.status} from request: ${r.pathInfo}, msg: ${body}"
resp.as[String].flatMap(body => {
val msg = s"unexpected status: ${resp.status} from request: ${req.pathInfo}, msg: ${body}"
IO.raiseError(new RuntimeException(msg))
}).flatMap(identity)
})
}

def isInitialized: IO[Boolean] =
req[Initialized](IO.pure(Request(GET, v1 / "sys" / "init"))).map(_.init)
req[Initialized](Request(GET, v1 / "sys" / "init")).map(_.init)

def initialize(init: Initialization): IO[InitialCreds] =
req[InitialCreds](Request(PUT, v1 / "sys" / "init").withBody(init.asJson))
req[InitialCreds](Request(PUT, v1 / "sys" / "init").withEntity(init.asJson))

def unseal(key: MasterKey): IO[SealStatus] =
req[SealStatus](Request(PUT, v1 / "sys" / "unseal").withBody(jsonUnseal(key)))
req[SealStatus](Request(PUT, v1 / "sys" / "unseal").withEntity(jsonUnseal(key)))

def sealStatus: IO[SealStatus] =
req[SealStatus](IO.pure(Request(GET, v1 / "sys" / "seal-status")))
req[SealStatus](Request(GET, v1 / "sys" / "seal-status"))

def seal: IO[Unit] =
req[String](IO.pure(Request(GET, v1 / "sys" / "init"))).void
req[String](Request(GET, v1 / "sys" / "init")).void

def createPolicy(cp: CreatePolicy): IO[Unit] =
reqVoid(Request(POST, v1 / "sys" / "policy" / cp.name).withBody(cp.asJson))
reqVoid(Request(POST, v1 / "sys" / "policy" / cp.name).withEntity(cp.asJson))

def deletePolicy(name: String): IO[Unit] =
reqVoid(IO.pure(Request(DELETE, v1 / "sys" / "policy" / name)))
reqVoid(Request(DELETE, v1 / "sys" / "policy" / name))

def getMounts: IO[SortedMap[String, Mount]] =
req[SortedMap[String, Mount]](IO.pure(Request(GET, uri = v1 / "sys" / "mounts")))
req[SortedMap[String, Mount]](Request(GET, uri = v1 / "sys" / "mounts"))

def createToken(ct: CreateToken): IO[Token] =
req[argonaut.Json](Request(POST, v1 / "auth" / "token" / "create").withBody(ct.asJson)).flatMap { json =>
req[argonaut.Json](Request(POST, v1 / "auth" / "token" / "create").withEntity(ct.asJson)).flatMap { json =>
val clientToken = for {
cursor <- Some(json.cursor): Option[Cursor]
auth <- cursor.downField("auth")
Expand Down
126 changes: 69 additions & 57 deletions http4s/src/test/scala/platypus/http4s/Http4sVaultSpec.scala
Original file line number Diff line number Diff line change
@@ -1,19 +1,19 @@
//: ----------------------------------------------------------------------------
//: Copyright (C) 2017 Verizon. All Rights Reserved.
//:
//: Licensed under the Apache License, Version 2.0 (the "License");
//: you may not use this file except in compliance with the License.
//: You may obtain a copy of the License at
//:
//: http://www.apache.org/licenses/LICENSE-2.0
//:
//: Unless required by applicable law or agreed to in writing, software
//: distributed under the License is distributed on an "AS IS" BASIS,
//: WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//: See the License for the specific language governing permissions and
//: limitations under the License.
//:
//: ----------------------------------------------------------------------------
//: ----------------------------------------------------------------------------
//: Copyright (C) 2017 Verizon. All Rights Reserved.
//:
//: Licensed under the Apache License, Version 2.0 (the "License");
//: you may not use this file except in compliance with the License.
//: You may obtain a copy of the License at
//:
//: http://www.apache.org/licenses/LICENSE-2.0
//:
//: Unless required by applicable law or agreed to in writing, software
//: distributed under the License is distributed on an "AS IS" BASIS,
//: WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//: See the License for the specific language governing permissions and
//: limitations under the License.
//:
//: ----------------------------------------------------------------------------
package platypus
package http4s

Expand All @@ -25,13 +25,16 @@ import org.http4s.client.blaze._
import org.scalatest.{FlatSpec, Matchers}

import scala.concurrent.duration._
import scala.concurrent.ExecutionContext.Implicits.global
import cats.effect._
import org.http4s._
import org.http4s.client.Client

class Http4sVaultSpec extends FlatSpec
with DockerTestKit
with Matchers
with DockerVaultService
with http4s.Json
{
with DockerTestKit
with Matchers
with DockerVaultService
with http4s.Json {
override val PullImagesTimeout = 20.minutes
override val StartContainersTimeout = 1.minute
override val StopContainersTimeout = 1.minute
Expand All @@ -44,64 +47,73 @@ class Http4sVaultSpec extends FlatSpec

var masterKey: MasterKey = _
var rootToken: RootToken = _
var interp: Http4sVaultClient = _

type Interpreter = Client[IO] => Http4sVaultClient

var interp: Interpreter = _

val baseUrl: Uri = vaultHost getOrElse Uri.uri("http://0.0.0.0:8200")

val client = Http1Client[IO]().unsafeRunSync()
implicit val cs: ContextShift[IO] = IO.contextShift(global)
implicit val timer: Timer[IO] = IO.timer(global)

val resource: Resource[IO, Client[IO]] = BlazeClientBuilder[IO](global).resource

val token = Token("asdf")
interp = new Http4sVaultClient(token, baseUrl, client)
interp = client => new Http4sVaultClient(token, baseUrl, client)

def withClient[A](op : VaultOpF[A])(interpreter:Interpreter):IO[A] = resource.use( client => op.foldMap(interpreter(client)))

if (sys.env.get("BUILDKITE").isEmpty) {
behavior of "vault"

it should "not be initialized" in {
VaultOp.isInitialized.foldMap(interp).unsafeRunSync() should be (false)
withClient(VaultOp.isInitialized)(interp).unsafeRunSync() should be(false)
}

it should "initialize" in {
val result = VaultOp.initialize(1,1).foldMap(interp).unsafeRunSync()
result.keys.size should be (1)
val result = withClient(VaultOp.initialize(1, 1))(interp).unsafeRunSync()
result.keys.size should be(1)
this.masterKey = result.keys(0)
this.rootToken = result.rootToken
this.interp = new Http4sVaultClient(Token(rootToken.value), baseUrl, client)
this.interp = client => new Http4sVaultClient(Token(rootToken.value), baseUrl, client)
}

it should "be initialized now" in {
VaultOp.isInitialized.foldMap(interp).unsafeRunSync() should be (true)
withClient(VaultOp.isInitialized)(interp).unsafeRunSync() should be(true)
}

it should "be sealed at startup" in {
val sealStatus = VaultOp.sealStatus.foldMap(interp).unsafeRunSync()
sealStatus.`sealed` should be (true)
sealStatus.total should be (1)
sealStatus.progress should be (0)
sealStatus.quorum should be (1)
val sealStatus = withClient(VaultOp.sealStatus)(interp).unsafeRunSync()
sealStatus.`sealed` should be(true)
sealStatus.total should be(1)
sealStatus.progress should be(0)
sealStatus.quorum should be(1)
}

it should "be unsealable" in {
val sealStatus = VaultOp.unseal(this.masterKey).foldMap(interp).unsafeRunSync()
sealStatus.`sealed` should be (false)
sealStatus.total should be (1)
sealStatus.progress should be (0)
sealStatus.quorum should be (1)
val sealStatus = withClient(VaultOp.unseal(this.masterKey))(interp).unsafeRunSync()
sealStatus.`sealed` should be(false)
sealStatus.total should be(1)
sealStatus.progress should be(0)
sealStatus.quorum should be(1)
}

it should "be unsealed after unseal" in {
val sealStatus = VaultOp.sealStatus.foldMap(interp).unsafeRunSync()
sealStatus.`sealed` should be (false)
sealStatus.total should be (1)
sealStatus.progress should be (0)
sealStatus.quorum should be (1)
val sealStatus = withClient(VaultOp.sealStatus)(interp).unsafeRunSync()
sealStatus.`sealed` should be(false)
sealStatus.total should be(1)
sealStatus.progress should be(0)
sealStatus.quorum should be(1)
}

it should "have cubbyhole, secret, sys mounted" in {
val mounts = VaultOp.getMounts.foldMap(interp).attempt.unsafeRunSync()
mounts.toOption.get.size should be (4)
mounts.toOption.get.contains("cubbyhole/") should be (true)
mounts.toOption.get.contains("secret/") should be (true)
mounts.toOption.get.contains("identity/") should be (true)
mounts.toOption.get.contains("sys/") should be (true)
val mounts = withClient(VaultOp.getMounts)(interp).attempt.unsafeRunSync()
mounts.toOption.get.size should be(4)
mounts.toOption.get.contains("cubbyhole/") should be(true)
mounts.toOption.get.contains("secret/") should be(true)
mounts.toOption.get.contains("identity/") should be(true)
mounts.toOption.get.contains("sys/") should be(true)
}

// This is how platypus writes policies. It provides a good test case for us.
Expand All @@ -123,24 +135,24 @@ class Http4sVaultSpec extends FlatSpec
)

it should "write policies" in {
VaultOp.createPolicy(cp.name, cp.rules).foldMap(interp).unsafeRunSync() should be (())
withClient(VaultOp.createPolicy(cp.name, cp.rules))(interp).unsafeRunSync() should be(())
}

it should "delete policies" in {
VaultOp.deletePolicy(cp.name).foldMap(interp).unsafeRunSync() should be (())
withClient(VaultOp.deletePolicy(cp.name))(interp).unsafeRunSync() should be(())
}

it should "encode policies correctly" in {
cp.asJson.field("policy") should be (Some(jString("""{"path":{"sys/*":{"policy":"deny"},"auth/token/revoke-self":{"policy":"write"},"example/qa/mysql/creds/howdy":{"capabilities":["read"]},"example/qa/cassandra/creds/howdy":{"capabilities":["read"]}}}""")))
cp.asJson.field("policy") should be(Some(jString("""{"path":{"sys/*":{"policy":"deny"},"auth/token/revoke-self":{"policy":"write"},"example/qa/mysql/creds/howdy":{"capabilities":["read"]},"example/qa/cassandra/creds/howdy":{"capabilities":["read"]}}}""")))
}

it should "create tokens" in {
val token2 = VaultOp.createToken(
val token2 = withClient(VaultOp.createToken(
policies = Some(List("default")),
ttl = Some(1.minute)
).foldMap(interp).unsafeRunSync()
val interp2 = new Http4sVaultClient(token2, baseUrl, client)
VaultOp.isInitialized.foldMap(interp2).unsafeRunSync() should be (true)
))(interp).unsafeRunSync()
val interp2 : Interpreter = client => new Http4sVaultClient(token2, baseUrl, client)
withClient(VaultOp.isInitialized)(interp2).unsafeRunSync() should be(true)
}
}
}
7 changes: 3 additions & 4 deletions project/versions.scala
Original file line number Diff line number Diff line change
Expand Up @@ -16,13 +16,12 @@
//: ----------------------------------------------------------------------------

object V {
val http4sArgonaut61 = "0.16.0"
val http4s = "0.18.9"
val http4s = "0.20.0-M4"
val journal = "3.0.18"
val knobs = "6.0.33"
val dockerit = "0.9.8"
val argonaut = "6.2.1"

val catsEffect = "0.10.1"
val cats = "1.1.0"
val catsEffect = "1.1.0"
val cats = "1.5.0"
}

0 comments on commit 5c9b9d0

Please sign in to comment.