HTTP Ingress

Verbs annotated with ftl:ingress will be exposed via HTTP (http is the default ingress type). These endpoints will then be available on one of our default ingress ports (local development defaults to http://localhost:8891).

The following will be available at http://localhost:8891/http/users/123/posts?postId=456.

type GetRequestPathParams struct {
	UserID string `json:"userId"`
}

type GetRequestQueryParams struct {
	PostID string `json:"postId"`
}

type GetResponse struct {
	Message string `json:"msg"`
}

//ftl:ingress GET /http/users/{userId}/posts
func Get(ctx context.Context, req builtin.HttpRequest[ftl.Unit, GetRequestPathParams, GetRequestQueryParams]) (builtin.HttpResponse[GetResponse, ErrorResponse], error) {
  // ...
}

Because the example above only has a single path parameter it can be simplified by just using a scalar such as string or int64 as the path parameter type:


//ftl:ingress GET /http/users/{userId}/posts
func Get(ctx context.Context, req builtin.HttpRequest[ftl.Unit, int64, GetRequestQueryParams]) (builtin.HttpResponse[GetResponse, ErrorResponse], error) {
  // ...
}

NOTE! The req and resp types of HTTP ingress verbs must be builtin.HttpRequest and builtin.HttpResponse respectively. These types provide the necessary fields for HTTP ingress (headers, statusCode, etc.)

You will need to import ftl/builtin.

Key points:

  • ingress verbs will be automatically exported by default.

Field mapping

The HttpRequest request object takes 3 type parameters, the body, the path parameters and the query parameters.

Given the following request verb:


type PostBody struct{
	Title string               `json:"title"`
	Content string             `json:"content"`
	Tag ftl.Option[string]     `json:"tag"`
}
type PostPathParams struct {
	UserID string             `json:"userId"`
	PostID string             `json:"postId"`
}

type PostQueryParams struct {
	Publish boolean `json:"publish"`
}

//ftl:ingress http PUT /users/{userId}/posts/{postId}
func Get(ctx context.Context, req builtin.HttpRequest[PostBody, PostPathParams, PostQueryParams]) (builtin.HttpResponse[GetResponse, string], error) {
	return builtin.HttpResponse[GetResponse, string]{
		Headers: map[string][]string{"Get": {"Header from FTL"}},
		Body: ftl.Some(GetResponse{
			Message: fmt.Sprintf("UserID: %s, PostID: %s, Tag: %s", req.pathParameters.UserID, req.pathParameters.PostID, req.Body.Tag.Default("none")),
		}),
	}, nil
}

The rules for how each element is mapped are slightly different, as they have a different structure:

  • The body is mapped directly to the body of the request, generally as a JSON object. Scalars are also supported, as well as []byte to get the raw body. If they type is any then it will be assumed to be JSON and mapped to the appropriate types based on the JSON structure.
  • The path parameters can be mapped directly to an object with field names corresponding to the name of the path parameter. If there is only a single path parameter it can be injected directly as a scalar. They can also be injected as a map[string]string.
  • The path parameters can also be mapped directly to an object with field names corresponding to the name of the path parameter. They can also be injected directly as a map[string]string, or map[string][]string for multiple values.

Optional fields

Optional fields are represented by the ftl.Option type. The Option type is a wrapper around the actual type and can be Some or None. In the example above, the Tag field is optional.

curl -i http://localhost:8891/users/123/posts/456

Because the tag query parameter is not provided, the response will be:

{
  "msg": "UserID: 123, PostID: 456, Tag: none"
}

Casing

Field names use lowerCamelCase by default. You can override this by using the json tag.

SumTypes

Given the following request verb:

//ftl:enum export
type SumType interface {
	tag()
}

type A string

func (A) tag() {}

type B []string

func (B) tag() {}

//ftl:ingress http POST /typeenum
func TypeEnum(ctx context.Context, req builtin.HttpRequest[SumType, ftl.Unit, ftl.Unit]) (builtin.HttpResponse[SumType, string], error) {
	return builtin.HttpResponse[SumType, string]{Body: ftl.Some(req.Body)}, nil
}

The following curl request will map the SumType name and value to the req.Body:

curl -X POST "http://localhost:8891/typeenum" \
     -H "Content-Type: application/json" \
     --data '{"name": "A", "value": "sample"}'

The response will be:

{
  "name": "A",
  "value": "sample"
}

Encoding query params as JSON

Complex query params can also be encoded as JSON using the @json query parameter. For example:

{"tag":"ftl"} url-encoded is %7B%22tag%22%3A%22ftl%22%7D

curl -i http://localhost:8891/users/123/posts/456?@json=%7B%22tag%22%3A%22ftl%22%7D

Kotlin uses the @Ingress annotation to define HTTP endpoints. These endpoints will be exposed on the default ingress port (local development defaults to http://localhost:8891).

import xyz.block.ftl.Ingress
import xyz.block.ftl.Option

// Simple GET endpoint with path and query parameters
@Ingress("GET /users/{userId}/posts")
fun getPost(request: Request): Response {
    val userId = request.pathParams["userId"]
    val postId = request.queryParams["postId"]
    return Response.ok(Post(userId, postId))
}

// POST endpoint with request body
@Ingress("POST /users/{userId}/posts")
fun createPost(request: Request): Response {
    val userId = request.pathParams["userId"]
    val body = request.body<PostBody>()
    return Response.created(Post(userId, body.title))
}

// Request body data class
data class PostBody(
    val title: String,
    val content: String,
    val tag: Option<String>  // Optional field using Option type
)

// Response data class
data class Post(
    val userId: String,
    val title: String
)

Key features:

  • The @Ingress annotation takes a string parameter combining the HTTP method and path
  • Path parameters are accessed via request.pathParams
  • Query parameters are accessed via request.queryParams
  • Request bodies can be automatically deserialized using request.body<T>()
  • Optional fields are represented using the Option<T> type
  • Response helpers like Response.ok() and Response.created() for common status codes

JVM Languages use the JAX-RS annotations to define HTTP endpoints. The following example shows how to define an HTTP endpoint in Java. As the underling implementation is based on Quarkus it is also possible to use the Quarkus extensions to the JAX-RS annotations:

In general the difference between the Quarkus annotation and the standard JAX-RS ones is that the Quarkus parameters infer the parameter name from the method parameter name, while the JAX-RS ones require the parameter name to be explicitly defined.


import java.util.List;

import jakarta.ws.rs.DELETE;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.POST;
import jakarta.ws.rs.PUT;
import jakarta.ws.rs.Path;

import jakarta.ws.rs.QueryParam; // JAX-RS annotation to get the query parameter
import org.jboss.resteasy.reactive.RestPath; // Quarkus annotation to get the path parameter

@Path("/")
public class TestHTTP {

    @GET
    @Path("/http/users/{userId}/posts")
    public String get(@RestPath String userId, @QueryParam("postId") String post) {
        //...
    }

}

Under the hood these HTTP invocations are being mapped to verbs that take a builtin.HttpRequest and return a builtin.HttpResponse. This is not exposed directly to the user, but is instead mapped directly to JAX-RS annotations.