au.com.dius.pact.server.Server.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of pact-jvm-server_2.10 Show documentation
Show all versions of pact-jvm-server_2.10 Show documentation
Pact server
===========
The pact server is a stand-alone interactions recorder and verifier, aimed at clients that are non-JVM or non-Ruby based.
The pact client for that platform will need to be implemented, but it only be responsible for generating the `JSON`
interactions, running the tests and communicating with the server.
The server implements a `JSON` `REST` Admin API with the following endpoints.
/ -> For diagnostics, currently returns a list of ports of the running mock servers.
/create -> For initialising a test server and submitting the JSON interactions. It returns a port
/complete -> For finalising and verifying the interactions with the server. It writes the `JSON` pact file to disk.
## Running the server
$ sbt pact-jvm-server/run
By default will run on port `29999` but a port number can be optionally supplied.
## Life cycle
The following actions are expected to occur
* The client calls `/create` to initialise a server with the expected `JSON` interactions and state
* The admin server will start a mock server on a random port and return the port number in the response
* The client will execute its interaction tests against the mock server with the supplied port
* Once finished, the client will call `/complete' on the Admin API, posting the port number
* The pact server will verify the interactions and write the `JSON` `pact` file to disk under `/target`
* The mock server running on the supplied port will be shutdown.
## Endpoints
### /create
The client will need `POST` to `/create` the generated `JSON` interactions, also providing a state as a query parameter.
For example:
POST http://localhost:29999/create?state=NoUsers '{ "provider": { "name": "Animal_Service"}, ... }'
This will create a new running mock service provider on a randomly generated port. The port will be returned in the
`201` response:
{ "port" : 34423 }
### /complete
Once the client has finished running its tests against the mock server on the supplied port (in this example port
`34423`) the client will need to `POST` to `/complete` the port number of the mock server that was used.
For example:
POST http://localhost:29999/complete '{ "port" : 34423 }'
This will cause the Pact server to verify the interactions, shutdown the mock server running on that port and writing
the pact `JSON` file to disk under the `target` directory.
### /
The `/` endpoint is for diagnostics and to check that the pact server is running. It will return all the currently
running mock servers port numbers.
For example:
GET http://localhost:29999/
'{ "ports": [23443,43232] }'
The newest version!
package au.com.dius.pact.server
import au.com.dius.pact.model._
import ch.qos.logback.classic.Level
import org.slf4j.{Logger, LoggerFactory}
import scala.collection.JavaConversions
object ListServers {
def apply(oldState: ServerState): Result = {
val body = OptionalBody.body("{\"ports\": [" + oldState.keySet.mkString(", ") + "]}")
Result(new Response(200, JavaConversions.mapAsJavaMap(Map("Content-Type" -> "application/json")), body), oldState)
}
}
case class Result(response: Response, newState: ServerState)
case class Config(port: Int = 29999,
host: String = "localhost",
daemon: Boolean = false,
portLowerBound: Int = 20000,
portUpperBound: Int = 40000,
debug: Boolean = false,
pactVersion: Int = 2)
object Server extends App {
val parser = new scopt.OptionParser[Config]("pact-jvm-server") {
arg[Int]("port") optional() action { (x, c) => c.copy(port = x) } text("port to run on (defaults to 29999)")
help("help") text("prints this usage text")
opt[String]('h', "host") action { (x, c) => c.copy(host = x) } text("host to bind to (defaults to localhost)")
opt[Int]('l', "mock-port-lower") action { (x, c) => c.copy(portLowerBound = x) } text("lower bound to allocate mock ports (defaults to 20000)")
opt[Int]('u', "mock-port-upper") action { (x, c) => c.copy(portUpperBound = x) } text("upper bound to allocate mock ports (defaults to 40000)")
opt[Unit]('d', "daemon") action { (_, c) => c.copy(daemon = true) } text("run as a daemon process")
opt[Unit]("debug") action { (_, c) => c.copy(debug = true) } text("run with debug logging")
opt[Int]('v', "pact-version") action { (x, c) => c.copy(pactVersion = x) } text("pact version to generate for (2 or 3)")
}
parser.parse(args, Config()) match {
case Some(config) =>
val logger = LoggerFactory.getLogger(Logger.ROOT_LOGGER_NAME).asInstanceOf[ch.qos.logback.classic.Logger]
if (config.debug) {
logger.setLevel(Level.DEBUG)
} else {
logger.setLevel(Level.INFO)
}
val server = _root_.unfiltered.netty.Server.http(config.port, config.host)
.handler(RequestHandler(new ServerStateStore(), config))
println(s"starting unfiltered app at ${config.host} on port ${config.port}")
server.start()
if (!config.daemon) {
readLine("press enter to stop server:\n")
server.stop()
}
case None =>
parser.showUsage
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy