- Depend on the
com.twitter.finatra:finatra-http
library. - We also recommend depending on
com.twitter.finatra:finatra-slf4j
andch.qos.logback:logback-classic
to choose Logback as your slf4j implementation. - See the finatra-hello-world example.
- See also todo list example.
Routes are defined inside a Controller and are comprised of:
- an HTTP method
- a matching pattern
- and a callback function.
When Finatra receives an HTTP request, it will scan all registered controllers (in the order they are added) and dispatch the request to the first matching route starting from the top of each controller.
The following route will return "hi" when an HTTP GET of / is received.
class Example extends Controller {
get("/") { request: Request =>
"hi"
}
}
- Follow REST conventions if possible. When deciding which routes to group into a particular controller, group routes related to a single resource into one controller. The per-route stats in StatsFilter work best when this convention is followed.
class GroupsController extends Controller {
get("/groups/:id") { ... }
post("/groups") { ... }
delete("/groups/:id") { ... }
}
yields the following stats:
route/groups_id/GET/...
route/groups/POST/...
route/groups_id/DELETE/...
Alternatively, each route can be assigned a name which will then be used to create stat names.
class GroupsController extends Controller {
get("/groups/:id", name = "group_by_id") { ... }
post("/groups", name = "create_group") { ... }
delete("/groups/:id", name = "delete_group") { ... }
}
which will yield the stats:
route/group_by_id/GET/...
route/create_group/POST/...
route/delete_group/DELETE/...
Route patterns may include named parameters:
get("/users/:id") { request: Request =>
"You looked up " + request.params("id")
}
Note: Query params and path params are both stored in the "params" field of the request. If a path parameter and a query param have the same name, the path param always wins. Therefore, ensure your path param names do not collide with a query param name that you plan to read.
Routes can also contain the wildcard pattern. The wildcard can only appear once at the end of a pattern, and it will capture all text in its place. For example
get("/files/:*") { request: Request =>
request.params("*")
}
For a GET of /files/abc/123/foo.txt
the endpoint will return abc/123/foo.txt
Any path starting with /admin/finatra/
will be exposed only on the Server's admin port. All TwitterServer based services get an HTTP admin interface for exposing internal endpoints such as stats. The admin endpoint should never be exposed outside your DMZ).
get("/admin/finatra/users/") { request: Request =>
userDatabase.getAllUsers(
request.params("cursor"))
}
Regular expressions are no longer allowed in string defined paths. Note: We are planning to support regular expression based paths in a future release.
Each route has a callback which is executed when the route matches a request. Callbacks require explicit input types and Finatra will then try to convert the incoming request into the specified input type. Finatra supports two request types:
- Finagle HTTP Request
- A custom
case class
Request
This is a com.twitter.finagle.http.Request which contains common HTTP attributes.
Custom requests allow declarative request parsing with support for type conversions, default values, and validations.
For example suppose you wanted to parse a GET request with three query params -- max
, startDate
, and verbose
:
http://foo.com/users?max=10&start_date=2014-05-30TZ&verbose=true
This can be parsed with the following case class
:
case class UsersRequest(
@Max(100) @QueryParam max: Int,
@PastDate @QueryParam startDate: Option[DateTime],
@QueryParam verbose: Boolean = false)
The custom UsersRequest
is then used as the callbacks input type:
get("/users") { request: UsersRequest =>
request
}
The following integration test shows how this endpoint will now respond.
server.httpGet(
path = "/users?start_date=2013&max=10&verbose=true",
andExpect = Ok,
withJsonBody = """
{
"start_date": "2013-01-01T00:00:00.000Z",
"max": 10,
"verbose": true
}
""")
server.httpGet(
path = "/users?max=10",
andExpect = Ok,
withJsonBody = """
{
"max": 10,
"verbose": false
}
""")
server.httpGet(
path = "/users?max=10&verbose=true",
andExpect = Ok,
withJsonBody = """
{
"max": 10,
"verbose": true
}
""")
server.httpGet(
path = "/users?verbose=5",
andExpect = BadRequest,
withJsonBody = """
{
"errors": [
"max is a required field",
"verbose's value '5' is not a valid boolean"]
]
}
""")
Notes:
-
The
case class
field names should match the request parameters names.- A PropertyNamingStrategy can be configured to handle common name substitutions (e.g. snake_case or camelCase). By default, snake_case is used (defaults are set in
FinatraJacksonModule
). - Use backticks when special characters are involved (e.g. @Header
user-agent
: String) - @JsonProperty can also be used to specify the JSON field name
- A PropertyNamingStrategy can be configured to handle common name substitutions (e.g. snake_case or camelCase). By default, snake_case is used (defaults are set in
-
Non optional fields without default values are required. If required fields are missing, a
CaseClassMappingException
is thrown. Normally, the default ExceptionMapper (included inExceptionMapperModule
) turns this exception into a HTTP 400 BadRequest with a JSON errors array (however this behavior can be customized). -
The following field annotations specify where to parse the field out of the request
- Request Fields
@RouteParam
@QueryParam
- Note: Ensure that route param names do not collide with QueryParam names. Otherwise, a QueryParam could end up parsing a route param
@Header
@Cookie
- Request Fields
-
Other
@RequestInject
: Injects the Finagle Http Request
Note: HTTP requests with a content-type of application/json, are similarly parsed (but "Request Field" annotations are ignored). See JSON section below.
Finatra will convert your route callbacks return type into a Future[Response]
using the following rules:
- If you return a
Future[Response]
, then no conversion will be performed - A non
Future
return value will be converted into aFuture
using a Finatra providedFuturePool
(see Future Conversion section for more details) Some[T]
will be converted into a HTTP 200 OKNone
will be converted into a HTTP 404 NotFound- Non-response classes will be converted into a HTTP 200 OK
All Controllers have a protected "response" field they can use to build responses. For example:
get("/foo") { request: Request =>
response.
ok.
header("a", "b").
json("""
{
"name": "Bob",
"age": 19
}
""")
}
get("/foo") { request: Request =>
response.
status(999).
body(bytes)
}
get("/redirect") { request: Request =>
response.
temporaryRedirect.
location("/foo/123")
}
post("/users") { request: FormPostRequest =>
response.created.
location("123").
html(
TestUserView(request.age, request.name))
}
Responses can be embedded inside exceptions with .toException
. You can throw the exception to terminate control flow, or wrap it inside a Future.exception
to return a failed Future
. However, instead of directly returning error responses in this manner, a better convention is to handle application-specific exceptions in an ExceptionMapper
.
get("/NotFound") { request: Request =>
response.notFound("abc not found").toFutureException
}
get("/ServerError") { request: Request =>
response.internalServerError.toFutureException
}
get("/ServiceUnavailable") { request: Request =>
// can throw a raw exception too
throw response.serviceUnavailable.toException
}
ResponseBuilder has a "location" method.
post("/users") { request: Request =>
response.created.
location("http").
html(
TestUserView(request.age, request.name))
}
which can be used:
- if the URI starts with "http" or "/" then the URI is placed in the Location header unchanged.
response.location("123")
will get turned into the correct full URL in the HttpResponseFilter (e.g.http://host.com/users/123
)
Or to obtain the request full path URL as follows:
RequestUtils.pathUrl(request)
Callbacks that do not return a Future will have their return values wrapped in a ConstFuture. If your non-future result calls a blocking method, you must avoid blocking the Finagle request by wrapping your blocking operation in a FuturePool e.g.
import com.twitter.finatra.utils.FuturePools
class MyController extends Controller {
private val futurePool = FuturePools.unboundedPool("CallbackConverter")
get("/") { request: Request =>
futurePool {
blockingCall()
}
}
}
The Finatra convention is to create a Scala object with a name ending in "Main" as so
object MyServerMain extends MyServer
class MyServer extends HttpServer {...}
This allows your server to be instantiated multiple times in tests without worrying about static state persisting across test runs in the same JVM. MyServerMain
is then the static object which contains the runnable main method.
Documentation coming soon. See example.
Finatra improves on the already excellent jackson-module-scala. JSON support is provided in the finatra-jackson library, which can be used outside of Finatra HTTP as a replacement for jackson-scala-module or jerkson.
- Usable outside of Finatra.
FinatraObjectMapper
which provides additional Scala friendly methods not found inScalaObjectMapper
.- Guice module for injecting
FinatraObjectMapper
(with support for customization e.g. snake_case vs camelCase). - Custom
case class
deserializer which overcomes limitations in jackson-scala-module. - Support for
case class
validations which accumulate errors (without failing fast) during json parsing.
Integration with Finatra HTTP routing to support binding and validation of query params, route params, and headers.
- Utils for comparing json in tests.
- Experimental support for iterator based json stream parsing.
The default configuration of Jackson is provided by the FinatraObjectMapper
.
The following Jackson integrations are provided by default:
- Joda Module
- Scala Module
- LongKeyDeserializer: Allow deserializing maps with long keys.
- Wrapped Value Serializer
- Duration Millis Serializer
- Improved DateTime Deserializer
- Improved
case class
Deserializer: See details below.
To override defaults or provide other config options, specify your own module (usually extending FinatraJacksonModule).
class Server extends HttpServer {
override def jacksonModule = CustomJacksonModule
...
}
object CustomJacksonModule extends FinatraJacksonModule {
override val additionalJacksonModules = Seq(
new SimpleModule {
addSerializer(LocalDateParser)
})
override val serializationInclusion = Include.NON_EMPTY
override val propertyNamingStrategy = CamelCasePropertyNamingStrategy
override def additionalMapperConfiguration(mapper: ObjectMapper) {
mapper.configure(Feature.WRITE_NUMBERS_AS_STRINGS, true)
}
}
Finatra provides a custom case class
deserializer which overcomes limitations in jackson-scala-module:
- Throw a JsonException when 'non Option' fields are missing in the incoming json
- Use default values when fields are missing in the incoming json
- Properly deserialize a Seq[Long] (see FasterXML/jackson-module-scala#62)
- Support "wrapped values" using
WrappedValue
(this is needed since jackson-scala-module does not support@JsonCreator
) - Support for accumulating JSON parsing errors (instead of failing fast).
- Support for field and method level validations which also accumulate errors.
If a custom case class
is used as a route callback's input type, Finatra will parse the request body into the custom request. Similar to declaratively parsing a GET request (described above), Finatra will perform validations and return a 400 BadRequest with a list of the accumulated errors (in JSON format).
Suppose you wanted to handle POST's of the following JSON (representing a group of tweet ids):
{
"name": "EarlyTweets",
"description": "Some of the earliest tweets on Twitter.",
"tweetIds": [20, 22, 24],
"dates": {
"start": "1",
"end": "2"
}
}
Then you'd create the following case classes
case class GroupRequest(
@NotEmpty name: String,
description: Option[String],
tweetIds: Set[Long],
dates: Dates) {
@MethodValidation
def validateName = {
ValidationResult(
name.startsWith("grp-"),
"name must start with 'grp-'")
}
}
case class Dates(
@PastTime start: DateTime,
@PastTime end: DateTime)
We provide a simple validation framework inspired by JSR-330. Our framework integrates with our custom case class
deserializer to efficiently apply per field validations as request parsing is performed. The following validations are included, and additional validations can be provided:
- CountryCode
- FutureTime
- PastTime
- Max
- Min
- NotEmpty
- OneOf
- Range
- Size
- TimeGranularity
- UUID
- MethodValidation
Can be used for:
- Non-generic validations -- a
MethodValidation
can be used instead of defining a reusable annotation and validator. - Cross-field validations (e.g.
startDate
beforeendDate
)
See the implementation of the GroupRequest
above for an example of using MethodValidation
.
See also: CommonMethodValidations
Use Java Enums for representing enumerations since they integrate well with Jackson's ObjectMapper and now have exhaustiveness checking as of Scala 2.10. The following Jackson annotations may be useful when working with Enums:
- @JsonCreator: Useful on a custom fromString method
- @JsonValue: Useful to place on an overridden toString method
The Finatra framework internally uses Guice extensively, and it's also availble for service writers if they choose to use it. For projects not wishing to use Guice, please see [FinatraWithoutGuice].
Finatra's Guice integration usually starts with Controllers as the root objects in the object graph. As such, controllers are added to Finatra's router by type as such:
class Server extends HttpServer {
override def configureHttp(router: HttpRouter) {
val controller = new Controller(...)
router.add[MyController]
}
}
Controllers are then annotated with JSR-330 inject annotations (Note: The somewhat strange syntax required to add an annotation to a Scala class constructor:
class MyController @Inject()(
dao: GroupsDAO,
service: FooService)
extends Controller
We provide a TwitterModule base class which extends the capabilities of the excellent scala-guice-module.
- Twitter Util Flags can be defined inside modules. This allows various reusable modules that require external configuration to be composed in a server.
- Prefer using a
@Provider
method over using the bind dsl. - Usually modules are Scala objects since the modules contain no state and usage of the module is less verbose.
- Always remember to add
@Singleton
to your provides method if desired. - Usually, modules are only required for creating classes that you don't control. Otherwise, you would simply add the JSR inject annotations directly to the class. For example, suppose you need to create an
ThirdPartyFoo
class which comes from a thirdparty jar. You could create the following Guice module to construct a singletonThirdPartyFoo
class which is created with a key provided through a command line flag.
object MyModule1 extends TwitterModule {
val key = flag("key", "defaultkey", "The key to use.")
@Singleton
@Provides
def providesThirdPartyFoo: ThirdPartyFoo = {
new ThirdPartyFoo(key())
}
}
A server is then started with a list of immutable Guice modules:
class Server extends HttpServer {
override val modules = Seq(
MyModule1,
MyModule2)
...
}
Guice supports custom scopes in addition to the most common Singleton and Unscoped. Request scopes are often used to allow injecting classes that change depending on the incoming request (e.g. the authenticated User). Finatra provides an implementation of a request scope that works across Finagle non-blocking threads (Guice's included request scope implementation uses ThreadLocal's which will not work with Finagle Futures).
Note: Fields added to the Finagle request scope will remain present in threads launched from Finagle future pools
First add a jar dependency on inject-request-scope
Then define a module
import com.myapp.User
import com.twitter.finatra.requestscope.RequestScopeBinding
import com.twitter.inject.TwitterModule
object UserModule
extends TwitterModule
with RequestScopeBinding {
override def configure() {
bindRequestScope[User]
}
}
Then define a Filter to seed the User into the Finatra Request Scope
class UserFilter @Inject()(
requestScope: FinagleRequestScope)
extends SimpleFilter[Request, Response] {
override def apply(request: Request, service: Service[Request, Response]): Future[Response] = {
val userId = parseUserId(request.cookie)
val user = User(userId)
requestScope.seed[User](user)
service(request)
}
}
Next add the FinagleRequestScopeFilter
filter to your server before the UserFilter
(shown below w/ other common filters in a recommended filter order):
class Server extends HttpServer {
override def configureHttp(router: HttpRouter) {
router.
filter[FinagleRequestScopeFilter].
filter[UserFilter].
add[MyController1]
}
}
Then inject a User or a Provider[User]
wherever you need to access the request scope user. Note, Provider[User]
must be used when injecting into a Singleton class.
import javax.inject.Provider
class MyController @Inject()(
dao: GroupsDAO,
user: Provider[User])
extends Controller {
get("/") { request: Request =>
"The incoming user has id " + user.get.id
}
}
- The server's injector is available as a protected method in
HttpServer
, but it's use should be avoided except for calling warmup classes, and for extending the Finatra framework. - Avoid
@Named
annotations in favor of specific Binding Annotations. If building with Maven, simply place your Java annotations in src/main/java for cross-compilation with your Scala code.
class NonGuiceServer extends HttpServer {
val key = flag("key", "123", "The Key to use")
override def configureHttp(router: HttpRouter) {
val keyValue = key()
val controller = new Controller(keyValue, ...)
router.add(controller)
}
}
Per controller filters are added as such:
class Server extends HttpServer {
override configureHttp(router: HttpRouter) {
router.
add[MyFilter, MyController]
}
}
A common filter order is as follows:
class Server extends HttpServer {
override configureHttp(router: HttpRouter) {
router.
filter[AccessLoggingFilter[Request]].
filter[StatsFilter].
filter[ExceptionMappingFilter[Request]].
filter[LoggingMDCFilter].
filter[FinagleRequestScopeFilter].
filter[UserFilter].
filter[HttpResponseFilter[Request]].
add[MyController1].
add[MyController2]
}
}
Finatra composes some commonly used filters into com.twitter.finatra.http.filters.CommonFilters
. CommonFilters
can be added in the same manner as any other filter, e.g.,
class Server extends HttpServer {
override configureHttp(router: HttpRouter) {
router.
filter[CommonFilters].
filter[LoggingMDCFilter].
filter[FinagleRequestScopeFilter].
filter[UserFilter].
add[MyController1].
add[MyController2]
}
}
Finatra server provides a warmup method that's called before the server's external HTTP port is bound and the /health port responds with OK. Often classes or routes will be called to warmup the JVM before traffic is routed to the server.
class DoEverythingServer extends HttpServer {
...
override def warmup() {
injector.instance[MyWarmupHandler].warmup()
}
}
import com.twitter.finagle.http.Status._
import com.twitter.finatra.http.routing.HttpWarmup
import com.twitter.finatra.httpclient.RequestBuilder._
import com.twitter.finatra.utils.Handler
import com.twitter.finatra.utils.ResponseUtils.expectOkResponse
import javax.inject.Inject
class MyWarmupHandler @Inject()(
httpWarmup: HttpWarmup) {
override def warmup() = {
httpWarmup.send(
request = get("/ok"),
responseCallback = expectOkResponse(_, "ok"))
}
}
A StatsReceiver can be injected anywhere in your application
class MyService @Inject()(
stats: StatsReceiver)
To override the default binding for StatsReceiver see GuiceApp#statsModule
Flag values can be injected in classes (and provider methods), by using the @Flag annotation:
class MyService @Inject()(
@Flag("key") key: String) {
}
class MyModule extends TwitterModule {
@Provider
@Singleton
def providesFoo(@Flag("key") key: String) = {
new Foo(key)
}
}
- If a flag is defined in a module, dereference that flag directly within that module (instead of using @Flag)
object MyModule1 extends TwitterModule {
val key = flag("key", "defaultkey", "The key to use.")
@Singleton
@Provides
def providesThirdPartyFoo: ThirdPartyFoo = {
new ThirdPartyFoo(key())
}
}
Some deployment environments may make it difficult to set command line flags. If this is the case, Finatra HTTP's core flags can be set from code. For example, instead of setting the "maxRequestSize" flag, you can override the following method in your server.
class TweetsEndpointServer extends HttpServer {
override val defaultMaxRequestSize = 10.megabytes
override def configureHttp(router: HttpRouter) {
...
}
}
Mustache templates must be placed in src/main/resources/templates
.
@Mustache("foo")
case class FooView(
name: String)
get("/foo") { request: Request =>
FooView("abc")
}
get("/foo") { request: Request =>
response.notFound(
FooView("abc"))
}
get("/foo") { request: Request =>
response.ok.view(
"foo.mustache",
FooClass("abc"))
}
Finatra's file server support is meant for internal apps only. Do not use the fileserver for production apps requiring a robust high performance file serving solution.
By default, files are served from the classpath. You can use the flag -doc.root
to customize the classpath root.
To serve files from the local file system, use the flag -local.doc.root
. Note that setting Java System Property -Denv=env
is no longer required nor supported. Setting the -local.doc.root
flag will trigger the same localFileMode
behavior.
Also note that it is an error to attempt to set both the -doc.root
and the -local.doc.root
flags. Either do nothing to load resources from the base of the classpath, configure a classpath "namespace" by setting the -doc.root
or load files from a local filesystem location specified by the -local.doc.root
flag.
get("/file") { request: Request =>
response.ok.file("/file123.txt")
}
get("/:*") { request: Request =>
response.ok.fileOrIndex(
request.params("*"),
"index.html")
}
Filters are code that runs before any request is dispatched to a particular Controller. They can modify the incoming request as well as the outbound response. A great example is our own LoggingFilter:
class DurationLoggingFilter
extends SimpleFilter[FinagleRequest, FinagleResponse]
with Logging {
def apply(
request: FinagleRequest,
service: Service[FinagleRequest, FinagleResponse]
) = {
val start = System.currentTimeMillis()
service(request) map { response =>
val end = System.currentTimeMillis()
val duration = end - start
info(request.method + " " + request.uri + " " + response.statusCode + " " + duration + " ms")
response
}
}
}
You can register these inside the HttpServer like so:
class MyServer extends HttpServer {
override def configureHttp(router: HttpRouter) {
router.
filter[DurationLoggingFilter].
add[MyController]
}
}
Cookies, like Headers, are read from request and set via render:
get("/") { request =>
val loggedIn = request.cookie("loggedIn").getOrElse("false")
response.ok.
plain("logged in?:" + loggedIn)
}
get("/") { request =>
response.ok.
plain("hi").
cookie("loggedIn", "true")
}
Advanced cookies are supported by creating and configuring Cookie objects:
get("/") { request =>
val c = DefaultCookie("Biz", "Baz")
c.setSecure(true)
response.ok.
plain("get:path").
cookie(c)
}
See the Cookie class for more details.
See MultiParamsTest.
One common and simple test is to check that the service can start up and report itself as healthy. This checks the correctness of the Guice dependency graph, catching errors that can otherwise cause the service to fail to start.
- Startup tests should mimic production as close as possible. As such:
- avoid using
@Bind
and "override modules" in startup tests. - set the Guice
stage
toPRODUCTION
so that all singletons will be eagerly created at startup (integration/feature tests run inState.DEVELOPMENT
by default). - prevent Finagle clients from making outbound connections during startup tests by setting Dtab entries to
nil!
.
- avoid using
For example:
import com.google.inject.Stage
import com.twitter.finatra.http.test.EmbeddedHttpServer
import com.twitter.inject.server.FeatureTest
class MyServiceStartupTests extends FeatureTest {
val server = EmbeddedHttpServer(
stage = Stage.PRODUCTION,
twitterServer = new SampleApiServer,
extraArgs = Seq(
"-dtab.add=/srv => /$/nil",
"-dtab.add=/srv# => /$/nil"))
"SampleApiServer" should {
"startup" in {
server.assertHealthy()
}
}
}
See TestInjector
Finatra uses the slf4j for framework logging.
From the slf4j documentation:
"The Simple Logging Facade for Java serves as a simple facade or abstraction for various logging frameworks, such as java.util.logging, Logback and log4j. SLF4J allows the end-user to plug in the desired logging framework at deployment time."
Note that slf4j is an interface that requires an actual logging implementation. If you are familiar with Log4j, this concept will be familiar as it separates the logging api interface from implementation allowing you to pick an appropriate implementation.
With that, when you are using slf4j you should ensure that you do not end-up with multiple implementations on your classpath, e.g., you should not have multiple slf4j bindings and/or a java.util.logging implementation, etc on your classpath as these are all competing implementations and classpath order is non-deterministic.
While there are several scala-wrappers for slf4j, Finatra uses and exposes some additional features on top of the excellent grizzled-slf4j project.
The main logging utility is the com.twitter.inject.Logging trait which can be mixed into any object or class:
class MyClass extends Logging {
def foo() = {
info("Calculating...")
"bar"
}
}
We highly recommend using Logback as an slf4j binding. If you choose to use Logback, include jar dependencies on ch.qos.logback:logback-classic
and com.twitter.finatra:finatra-slf4j
which will give a Logback slf4j implementation (logback-classic
) and bridges from the 3 most popular jvm logging libraries (via finatra-slf4j
):
- log4j
- commons-logging
java.util.logging
: There is a performance penalty for intercepting jul log messages, so make sure to also include the Slf4jBridgeModule in your servers list of Guice modules, as this will install the SLF4JBridgeHandler which mitigates most of the performance penalty. e.g.,
class Server extends HttpServer {
override val modules = Seq(
Slf4jBridgeModule,
...)
...
}
See logback.xml and logback-test.xml in finatra-hello-world project.
MDC Filters
Place the LoggingMDCFilter filter before any other filters which will add entries or expect MDC entries to be present.
- Avoid
private[this]
unless you are in a hotspot identified during profiling. - Avoid using custom flags for server locations in Finagle clients. Instead, use the Finagle provided
resolverMap
.
Many Finatra utilities are provided as conversions which add methods to common Scala and Finagle classes. They can be found in the com.twitter.finatra.conversions
package in com.twitter.finatra:finatra-utils
. Currently, the best documentation are the unit tests showing their usage.
A barebones httpclient built on finagle-http
is included in the finatra-httpclient
project. Stay tuned for further documentation and examples.
Finatra's HTTP server is built on top of several reusable traits.
One of these traits is com.twitter.inject.App
which provides the integration between Guice and com.twitter.app.App
. com.twitter.inject.App
can be used standalone to create command line apps which may also reuse your Guice modules defined in other libraries.
See SampleGuiceApp and SampleAppIntegrationTest
You no longer need to return a Future
from controller routes (however, always return a Future
if you already have one).
###Add Request type to controller callbacks
//v1
get("/foo") { request =>
get("/foo") { _ =>
//v2
import com.twitter.finagle.http.Request
get("/foo") { request: Request =>
Change "render" to "response" and specify the HTTP status as the first method after response (e.g. ok, created, notFound, etc)
//v1
render.json(ret)
//v2
response.ok.json(ret)
Route params are now stored in request.params (which allows us to reuse finagle.http.Request
without defining our own).
//v1
request.routeParams("q")
//v2
request.params("q")
To continue using "Java Util Logging", add a jar dependency on slf4j-jdk14
. Otherwise, we recommend using Logback by adding jar dependencies on ch.qos.logback:logback-classic
and com.twitter.finatra:finatra-slf4j
.
//v1
log.info("hello")
//v2
info("hello")
ExceptionMappers map exceptions to responses. It needs to implement the following trait:
trait ExceptionMapper[T <: Throwable] {
def toResponse(request: Request, throwable: T): Response
}
which says it will handle T
-typed exceptions. The request that triggered the exception is also provided as an argument. You can make use of exception mapping by adding the ExceptionMappingFilter
to your com.twitter.finatra.routing.HttpRouter
, e.g.,
router.filter[ExceptionMappingFilter[Request]]
The ExceptionMappingFilter
takes an ExceptionManager
which is provided by default in the ExceptionMapperModule
. You can override the default module if necessary by overriding the value in the HttpServer
.
To replicate v1 functionality around notFound
and error
, you could do the following:
//v1
notFound { request => ... }
error { request => ... }
//v2 notFound: Create a filter and add it before your controller:
@Singleton
class NotFoundFilter @Inject()(
response: ResponseBuilder)
extends SimpleFilter[Request, Response] {
def apply(request: Request, service: Service[Request, Response]): Future[Response] = {
service(request) map { origResponse =>
if (origResponse.status == Status.NotFound)
response.notFound("bar")
else
origResponse
}
}
}
//v2 error: Write an exception mapper and register it with HttpRouter
@Singleton
class ArithmeticExceptionMapper @Inject()(
response: ResponseBuilder)
extends ExceptionMapper[ArithmeticException] {
override def toResponse(request: Request, e: ArithmeticException): Response = {
response.internalServerError("whoops, divide by zero!")
}
}
router.
filter[NotFoundFilter].
filter[ExceptionMappingFilter[Request]].
exceptionMapper[ArithmeticExceptionMapper]
Override the configure method and add your controllers there.
Note: Flag parsing that used to be in App's constructor should be moved into the configureHttp
method.
//v2
class Server extends HttpServer {
override def configureHttp(router: HttpRouter) {
val controller1 = new Controller1(...)
val controller2 = new Controller2(...)
router.
commonFilter[CommonFilters].
commonFilter[NotFoundFilter]. // if needed (see above section on Error Handling)
add(controller1).
add(controller2)
}
}
- Web resources (html/js) go in
src/main/webapp
- Mustache templates now go in
src/main/resources/templates
To serve static files, you now need explicit routes:
get("/:*") { request: Request =>
response.ok.file(
request.params("*"))
}
If you have an "index" page (e.g. index.html) add the following route.
get("/:*") { request: Request =>
response.ok.fileOrIndex(
filePath = request.params("*"),
indexPath = "index.html")
Global flags are no longer used for standard server configuration. Instead:
//v2
-log.output=twitter-server.log
-http.port=:8080
-admin.port=:8081
- In v1,
SpecHelper
andMockApp
are used. - In v2, we provide a common way to run blackbox and whitebox integration tests against a locally running server:
- Simple Example
- More Powerful Example [TODO]
- Render a route from another route.
- Controller notFound and error handler methods.