diff --git a/backend/controller/ingress/ingress_integration_test.go b/backend/controller/ingress/ingress_integration_test.go
index 52767e6371..2d6831e096 100644
--- a/backend/controller/ingress/ingress_integration_test.go
+++ b/backend/controller/ingress/ingress_integration_test.go
@@ -5,6 +5,7 @@ package ingress_test
import (
"net/http"
"os"
+ "strings"
"testing"
"github.com/alecthomas/assert/v2"
@@ -15,146 +16,161 @@ import (
func TestHttpIngress(t *testing.T) {
in.Run(t,
+ in.WithLanguages("go", "java"),
in.CopyModule("httpingress"),
in.Deploy("httpingress"),
- in.HttpCall(http.MethodGet, "/users/123/posts/456", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Get"])
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
+ in.SubTests(
+ in.SubTest{Name: "GetWithPathParams", Action: in.HttpCall(http.MethodGet, "/users/123/posts/456", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Get"])
+ expectContentType(t, resp, "application/json;charset=utf-8")
- message, ok := resp.JsonBody["msg"].(string)
- assert.True(t, ok, "msg is not a string: %s", repr.String(resp.JsonBody))
- assert.Equal(t, "UserID: 123, PostID: 456", message)
+ message, ok := resp.JsonBody["msg"].(string)
+ assert.True(t, ok, "msg is not a string: %s", repr.String(resp.JsonBody))
+ assert.Equal(t, "UserID: 123, PostID: 456", message)
- nested, ok := resp.JsonBody["nested"].(map[string]any)
- assert.True(t, ok, "nested is not a map: %s", repr.String(resp.JsonBody))
- goodStuff, ok := nested["good_stuff"].(string)
- assert.True(t, ok, "good_stuff is not a string: %s", repr.String(resp.JsonBody))
- assert.Equal(t, "This is good stuff", goodStuff)
- }),
- in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"userId": 123, "postId": 345}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 201, resp.Status)
- assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Post"])
- success, ok := resp.JsonBody["success"].(bool)
- assert.True(t, ok, "success is not a bool: %s", repr.String(resp.JsonBody))
- assert.True(t, success)
- }),
- // contains aliased field
- in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 201, resp.Status)
- }),
- in.HttpCall(http.MethodPut, "/users/123", nil, in.JsonData(t, in.Obj{"postId": "346"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Put"])
- assert.Equal(t, map[string]any{}, resp.JsonBody)
- }),
- in.HttpCall(http.MethodDelete, "/users/123", nil, in.JsonData(t, in.Obj{}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Delete"])
- assert.Equal(t, map[string]any{}, resp.JsonBody)
- }),
+ nested, ok := resp.JsonBody["nested"].(map[string]any)
+ assert.True(t, ok, "nested is not a map: %s", repr.String(resp.JsonBody))
+ goodStuff, ok := nested["good_stuff"].(string)
+ assert.True(t, ok, "good_stuff is not a string: %s", repr.String(resp.JsonBody))
+ assert.Equal(t, "This is good stuff", goodStuff)
+ })},
+ in.SubTest{Name: "PostUsers", Action: in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"userId": 123, "postId": 345}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 201, resp.Status)
+ assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Post"])
+ success, ok := resp.JsonBody["success"].(bool)
+ assert.True(t, ok, "success is not a bool: %s", repr.String(resp.JsonBody))
+ assert.True(t, success)
+ })},
+ // contains aliased field
+ in.SubTest{Name: "PostUsersAliased", Action: in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 201, resp.Status)
+ })},
+ in.SubTest{Name: "PutUsers", Action: in.HttpCall(http.MethodPut, "/users/123", nil, in.JsonData(t, in.Obj{"postId": "346"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Put"])
+ assert.Equal(t, map[string]any{}, resp.JsonBody)
+ })},
+ in.SubTest{Name: "DeleteUsers", Action: in.HttpCall(http.MethodDelete, "/users/123", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, []string{"Header from FTL"}, resp.Headers["Delete"])
+ assert.Equal(t, map[string]any{}, resp.JsonBody)
+ })},
- in.HttpCall(http.MethodGet, "/queryparams?foo=bar", nil, in.JsonData(t, in.Obj{}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, "bar", string(resp.BodyBytes))
- }),
+ in.SubTest{Name: "GetQueryParams", Action: in.HttpCall(http.MethodGet, "/queryparams?foo=bar", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, "bar", string(resp.BodyBytes))
+ })},
- in.HttpCall(http.MethodGet, "/queryparams", nil, in.JsonData(t, in.Obj{}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, "No value", string(resp.BodyBytes))
- }),
+ in.SubTest{Name: "GetMissingQueryParams", Action: in.HttpCall(http.MethodGet, "/queryparams", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, "No value", string(resp.BodyBytes))
+ })},
- in.HttpCall(http.MethodGet, "/html", nil, in.JsonData(t, in.Obj{}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"text/html; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, "
HTML Page From FTL 🚀!
", string(resp.BodyBytes))
- }),
+ in.SubTest{Name: "GetHTML", Action: in.HttpCall(http.MethodGet, "/html", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "text/html;charset=utf-8")
+ assert.Equal(t, "HTML Page From FTL 🚀!
", string(resp.BodyBytes))
+ })},
- in.HttpCall(http.MethodPost, "/bytes", nil, []byte("Hello, World!"), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/octet-stream"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("Hello, World!"), resp.BodyBytes)
- }),
+ in.SubTest{Name: "PostBytes", Action: in.HttpCall(http.MethodPost, "/bytes", nil, []byte("Hello, World!"), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/octet-stream")
+ assert.Equal(t, []byte("Hello, World!"), resp.BodyBytes)
+ })},
- in.HttpCall(http.MethodGet, "/empty", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, nil, resp.Headers["Content-Type"])
- assert.Equal(t, nil, resp.BodyBytes)
- }),
-
- in.HttpCall(http.MethodPost, "/string", nil, []byte("Hello, World!"), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"text/plain; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("Hello, World!"), resp.BodyBytes)
- }),
+ in.SubTest{Name: "GetEmpty", Action: in.HttpCall(http.MethodGet, "/empty", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ assert.Equal(t, nil, resp.Headers["Content-Type"])
+ assert.Equal(t, nil, resp.BodyBytes)
+ })},
- in.HttpCall(http.MethodPost, "/int", nil, []byte("1234"), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"text/plain; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("1234"), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/float", nil, []byte("1234.56789"), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"text/plain; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("1234.56789"), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/bool", nil, []byte("true"), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"text/plain; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("true"), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodGet, "/error", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 500, resp.Status)
- assert.Equal(t, []string{"text/plain; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, []byte("Error from FTL"), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/array/string", nil, in.JsonData(t, []string{"hello", "world"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, in.JsonData(t, []string{"hello", "world"}), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/array/data", nil, in.JsonData(t, []in.Obj{{"item": "a"}, {"item": "b"}}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, in.JsonData(t, []in.Obj{{"item": "a"}, {"item": "b"}}), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/typeenum", nil, in.JsonData(t, in.Obj{"name": "A", "value": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, in.JsonData(t, in.Obj{"name": "A", "value": "hello"}), resp.BodyBytes)
- }),
- // CORS preflight request without CORS middleware enabled
- in.HttpCall(http.MethodOptions, "/typeenum", map[string][]string{
- "Origin": {"http://localhost:8892"},
- "Access-Control-Request-Method": {"GET"},
- "Access-Control-Request-Headers": {"x-forwarded-capabilities"},
- }, nil, func(t testing.TB, resp *in.HTTPResponse) {
- // should not return access control headers because we have not set up cors in this controller
- assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Origin"])
- assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Methods"])
- assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Headers"])
- }),
- in.HttpCall(http.MethodPost, "/external", nil, in.JsonData(t, in.Obj{"message": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, in.JsonData(t, in.Obj{"message": "hello"}), resp.BodyBytes)
- }),
- in.HttpCall(http.MethodPost, "/external2", nil, in.JsonData(t, in.Obj{"message": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 200, resp.Status)
- assert.Equal(t, []string{"application/json; charset=utf-8"}, resp.Headers["Content-Type"])
- assert.Equal(t, in.JsonData(t, in.Obj{"Message": "hello"}), resp.BodyBytes)
- }),
- // not lenient
- in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345, "extra": "blah"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 400, resp.Status)
- }),
- // lenient
- in.HttpCall(http.MethodPost, "/lenient", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345, "extra": "blah"}), func(t testing.TB, resp *in.HTTPResponse) {
- assert.Equal(t, 201, resp.Status)
- }),
+ in.SubTest{Name: "PostString", Action: in.HttpCall(http.MethodPost, "/string", nil, []byte("Hello, World!"), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "text/plain;charset=utf-8")
+ assert.Equal(t, []byte("Hello, World!"), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "GetError", Action: in.HttpCall(http.MethodGet, "/error", nil, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 500, resp.Status)
+ expectContentType(t, resp, "text/plain;charset=utf-8")
+ assert.True(t, strings.Contains(string(resp.BodyBytes), "Error"))
+ })},
+ in.SubTest{Name: "PostArrayString", Action: in.HttpCall(http.MethodPost, "/array/string", nil, in.JsonData(t, []string{"hello", "world"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/json;charset=utf-8")
+ assert.Equal(t, in.JsonData(t, []string{"hello", "world"}), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "PostArrayJSON", Action: in.HttpCall(http.MethodPost, "/array/data", nil, in.JsonData(t, []in.Obj{{"item": "a"}, {"item": "b"}}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/json;charset=utf-8")
+ assert.Equal(t, in.JsonData(t, []in.Obj{{"item": "a"}, {"item": "b"}}), resp.BodyBytes)
+ })},
+ // CORS preflight request without CORS middleware enabled
+ in.SubTest{Name: "TestOptionsRequest", Action: in.HttpCall(http.MethodOptions, "/typeenum", map[string][]string{
+ "Origin": {"http://localhost:8892"},
+ "Access-Control-Request-Method": {"GET"},
+ "Access-Control-Request-Headers": {"x-forwarded-capabilities"},
+ }, nil, func(t testing.TB, resp *in.HTTPResponse) {
+ // should not return access control headers because we have not set up cors in this controller
+ assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Origin"])
+ assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Methods"])
+ assert.Equal(t, nil, resp.Headers["Access-Control-Allow-Headers"])
+ })},
+ ),
+ in.IfLanguage("go", in.SubTests(
+ // Double, Int etc work in java with JSON encoding, but test/plain is not implemented yet
+ in.SubTest{Name: "PostInt", Action: in.HttpCall(http.MethodPost, "/int", nil, []byte("1234"), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "text/plain;charset=utf-8")
+ assert.Equal(t, []byte("1234"), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "PostFloat", Action: in.HttpCall(http.MethodPost, "/float", nil, []byte("1234.56789"), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "text/plain;charset=utf-8")
+ assert.Equal(t, []byte("1234.56789"), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "PostBool", Action: in.HttpCall(http.MethodPost, "/bool", nil, []byte("true"), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "text/plain;charset=utf-8")
+ assert.Equal(t, []byte("true"), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "PostTypeEnum", Action: in.HttpCall(http.MethodPost, "/typeenum", nil, in.JsonData(t, in.Obj{"name": "A", "value": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/json;charset=utf-8")
+ assert.Equal(t, in.JsonData(t, in.Obj{"name": "A", "value": "hello"}), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "TestExternalType", Action: in.HttpCall(http.MethodPost, "/external", nil, in.JsonData(t, in.Obj{"message": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/json;charset=utf-8")
+ assert.Equal(t, in.JsonData(t, in.Obj{"message": "hello"}), resp.BodyBytes)
+ })},
+ in.SubTest{Name: "TestExternalType2", Action: in.HttpCall(http.MethodPost, "/external2", nil, in.JsonData(t, in.Obj{"message": "hello"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 200, resp.Status)
+ expectContentType(t, resp, "application/json;charset=utf-8")
+ assert.Equal(t, in.JsonData(t, in.Obj{"Message": "hello"}), resp.BodyBytes)
+ })},
+ // not lenient
+ in.SubTest{Name: "TestExtraFieldStrict", Action: in.HttpCall(http.MethodPost, "/users", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345, "extra": "blah"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 400, resp.Status)
+ })},
+ // lenient
+ in.SubTest{Name: "TestExtraFieldLenient", Action: in.HttpCall(http.MethodPost, "/lenient", nil, in.JsonData(t, in.Obj{"user_id": 123, "postId": 345, "extra": "blah"}), func(t testing.TB, resp *in.HTTPResponse) {
+ assert.Equal(t, 201, resp.Status)
+ })},
+ )),
)
}
+func expectContentType(t testing.TB, resp *in.HTTPResponse, expected string) {
+ t.Helper()
+ headers := resp.Headers["Content-Type"]
+ for k, v := range headers {
+ headers[k] = strings.ReplaceAll(strings.ToLower(v), " ", "")
+ }
+ expected = strings.ReplaceAll(strings.ToLower(expected), " ", "")
+ assert.Equal(t, []string{expected}, headers)
+}
+
// Run with CORS enabled via FTL_CONTROLLER_ALLOW_ORIGIN and FTL_CONTROLLER_ALLOW_HEADERS
// This test is similar to TestHttpIngress above with the addition of CORS enabled in the controller.
func TestHttpIngressWithCors(t *testing.T) {
diff --git a/backend/controller/ingress/testdata/java/httpingress/ftl.toml b/backend/controller/ingress/testdata/java/httpingress/ftl.toml
new file mode 100644
index 0000000000..e277e2457d
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/ftl.toml
@@ -0,0 +1,2 @@
+module = "httpingress"
+language = "java"
diff --git a/backend/controller/ingress/testdata/java/httpingress/pom.xml b/backend/controller/ingress/testdata/java/httpingress/pom.xml
new file mode 100644
index 0000000000..3081bbc8b7
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/pom.xml
@@ -0,0 +1,14 @@
+
+
+ 4.0.0
+ xyz.block.ftl.examples
+ httpingress
+ 1.0-SNAPSHOT
+
+
+ xyz.block.ftl
+ ftl-build-parent-java
+ 1.0-SNAPSHOT
+
+
+
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/ArrayType.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/ArrayType.java
new file mode 100644
index 0000000000..bbcdebde11
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/ArrayType.java
@@ -0,0 +1,15 @@
+package xyz.block.ftl.java.test.http;
+
+public class ArrayType {
+
+ private String item;
+
+ public String getItem() {
+ return item;
+ }
+
+ public ArrayType setItem(String item) {
+ this.item = item;
+ return this;
+ }
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteRequest.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteRequest.java
new file mode 100644
index 0000000000..7ba6a56cfa
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteRequest.java
@@ -0,0 +1,13 @@
+package xyz.block.ftl.java.test.http;
+
+public class DeleteRequest {
+ private String userId;
+
+ public String getUserId() {
+ return userId;
+ }
+
+ public void setUserId(String userId) {
+ this.userId = userId;
+ }
+}
\ No newline at end of file
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteResponse.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteResponse.java
new file mode 100644
index 0000000000..0cfac28290
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/DeleteResponse.java
@@ -0,0 +1,4 @@
+package xyz.block.ftl.java.test.http;
+
+public class DeleteResponse {
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetRequest.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetRequest.java
new file mode 100644
index 0000000000..3e0bdef236
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetRequest.java
@@ -0,0 +1,23 @@
+package xyz.block.ftl.java.test.http;
+
+public class GetRequest {
+
+ private String userId;
+ private String postId;
+
+ public String getUserId() {
+ return userId;
+ }
+
+ public void setUserId(String userId) {
+ this.userId = userId;
+ }
+
+ public String getPostId() {
+ return postId;
+ }
+
+ public void setPostId(String postId) {
+ this.postId = postId;
+ }
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetResponse.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetResponse.java
new file mode 100644
index 0000000000..ee54ff2497
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/GetResponse.java
@@ -0,0 +1,27 @@
+package xyz.block.ftl.java.test.http;
+
+public class GetResponse {
+
+ private Nested nested;
+
+ private String msg;
+
+ public Nested getNested() {
+ return nested;
+ }
+
+ public GetResponse setNested(Nested nested) {
+ this.nested = nested;
+ return this;
+
+ }
+
+ public String getMsg() {
+ return msg;
+ }
+
+ public GetResponse setMsg(String msg) {
+ this.msg = msg;
+ return this;
+ }
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/Nested.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/Nested.java
new file mode 100644
index 0000000000..9885ee93a5
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/Nested.java
@@ -0,0 +1,18 @@
+package xyz.block.ftl.java.test.http;
+
+import com.fasterxml.jackson.annotation.JsonAlias;
+
+public class Nested {
+
+ @JsonAlias("good_stuff")
+ private String goodStuff;
+
+ public String getGoodStuff() {
+ return goodStuff;
+ }
+
+ public Nested setGoodStuff(String goodStuff) {
+ this.goodStuff = goodStuff;
+ return this;
+ }
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostRequest.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostRequest.java
new file mode 100644
index 0000000000..675140969c
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostRequest.java
@@ -0,0 +1,25 @@
+package xyz.block.ftl.java.test.http;
+
+import com.fasterxml.jackson.annotation.JsonAlias;
+
+public class PostRequest {
+ @JsonAlias("user_id")
+ private int userId;
+ private int postId;
+
+ public int getUserId() {
+ return userId;
+ }
+
+ public void setUserId(int userId) {
+ this.userId = userId;
+ }
+
+ public int getPostId() {
+ return postId;
+ }
+
+ public void setPostId(int postId) {
+ this.postId = postId;
+ }
+}
\ No newline at end of file
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostResponse.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostResponse.java
new file mode 100644
index 0000000000..4082def8a3
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PostResponse.java
@@ -0,0 +1,14 @@
+package xyz.block.ftl.java.test.http;
+
+public class PostResponse {
+ private boolean success;
+
+ public boolean isSuccess() {
+ return success;
+ }
+
+ public PostResponse setSuccess(boolean success) {
+ this.success = success;
+ return this;
+ }
+}
\ No newline at end of file
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutRequest.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutRequest.java
new file mode 100644
index 0000000000..f6219cf8c6
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutRequest.java
@@ -0,0 +1,13 @@
+package xyz.block.ftl.java.test.http;
+
+public class PutRequest {
+ private String postId;
+
+ public String getPostId() {
+ return postId;
+ }
+
+ public void setPostId(String postId) {
+ this.postId = postId;
+ }
+}
\ No newline at end of file
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutResponse.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutResponse.java
new file mode 100644
index 0000000000..27fa8e67e2
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/PutResponse.java
@@ -0,0 +1,5 @@
+package xyz.block.ftl.java.test.http;
+
+public class PutResponse {
+
+}
\ No newline at end of file
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/QueryParamRequest.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/QueryParamRequest.java
new file mode 100644
index 0000000000..bce8e6b608
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/QueryParamRequest.java
@@ -0,0 +1,18 @@
+package xyz.block.ftl.java.test.http;
+
+import org.jetbrains.annotations.Nullable;
+
+public class QueryParamRequest {
+
+ @Nullable
+ String foo;
+
+ public @Nullable String getFoo() {
+ return foo;
+ }
+
+ public QueryParamRequest setFoo(@Nullable String foo) {
+ this.foo = foo;
+ return this;
+ }
+}
diff --git a/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/TestHTTP.java b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/TestHTTP.java
new file mode 100644
index 0000000000..bb4da42d26
--- /dev/null
+++ b/backend/controller/ingress/testdata/java/httpingress/src/main/java/xyz/block/ftl/java/test/http/TestHTTP.java
@@ -0,0 +1,124 @@
+package xyz.block.ftl.java.test.http;
+
+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.Produces;
+
+import jakarta.ws.rs.core.MediaType;
+import org.jboss.resteasy.reactive.ResponseHeader;
+import org.jboss.resteasy.reactive.ResponseStatus;
+import org.jboss.resteasy.reactive.RestPath;
+import org.jboss.resteasy.reactive.RestQuery;
+
+@Path("/")
+public class TestHTTP {
+
+ @GET
+ @Path("/users/{userId}/posts/{postId}")
+ @ResponseHeader(name = "Get", value = "Header from FTL")
+ public GetResponse get(@RestPath String userId, @RestPath String postId) {
+ return new GetResponse()
+ .setMsg(String.format("UserID: %s, PostID: %s", userId, postId))
+ .setNested(new Nested().setGoodStuff("This is good stuff"));
+ }
+
+ @POST
+ @Path("/users")
+ @ResponseStatus(201)
+ @ResponseHeader(name = "Post", value = "Header from FTL")
+ public PostResponse post(PostRequest req) {
+ return new PostResponse().setSuccess(true);
+ }
+
+ @PUT
+ @Path("/users/{userId}")
+ @ResponseHeader(name = "Put", value = "Header from FTL")
+ public PutResponse put(PutRequest req) {
+ return new PutResponse();
+ }
+
+ @DELETE
+ @Path("/users/{userId}")
+ @ResponseHeader(name = "Delete", value = "Header from FTL")
+ @ResponseStatus(200)
+ public DeleteResponse delete(@RestPath String userId) {
+ System.out.println("delete");
+ return new DeleteResponse();
+ }
+
+ @GET
+ @Path("/queryparams")
+ public String query(@RestQuery String foo) {
+ return foo == null ? "No value" : foo;
+ }
+
+ @GET
+ @Path("/html")
+ @Produces("text/html; charset=utf-8")
+ public String html() {
+ return "HTML Page From FTL 🚀!
";
+ }
+
+ @POST
+ @Path("/bytes")
+ public byte[] bytes(byte[] b) {
+ return b;
+ }
+
+ @GET
+ @Path("/empty")
+ @ResponseStatus(200)
+ public void empty() {
+ }
+
+ @POST
+ @Path("/string")
+ public String string(String val) {
+ return val;
+ }
+
+ @POST
+ @Path("/int")
+ @Produces(MediaType.APPLICATION_JSON)
+ public int intMethod(int val) {
+ return val;
+ }
+
+ @POST
+ @Path("/float")
+ @Produces(MediaType.APPLICATION_JSON)
+ public float floatVerb(float val) {
+ return val;
+ }
+
+ @POST
+ @Path("/bool")
+ @Produces(MediaType.APPLICATION_JSON)
+ public boolean bool(boolean val) {
+ return val;
+ }
+
+ @GET
+ @Path("/error")
+ public String error() {
+ throw new RuntimeException("Error from FTL");
+ }
+
+ @POST
+ @Path("/array/string")
+ public String[] arrayString(String[] array) {
+ return array;
+ }
+
+ @POST
+ @Path("/array/data")
+ public List arrayData(List array) {
+ return array;
+ }
+
+}
diff --git a/jvm-runtime/ftl-runtime/common/deployment/src/main/java/xyz/block/ftl/deployment/FtlProcessor.java b/jvm-runtime/ftl-runtime/common/deployment/src/main/java/xyz/block/ftl/deployment/FtlProcessor.java
index 37eed9da1e..e09a4b5c4a 100644
--- a/jvm-runtime/ftl-runtime/common/deployment/src/main/java/xyz/block/ftl/deployment/FtlProcessor.java
+++ b/jvm-runtime/ftl-runtime/common/deployment/src/main/java/xyz/block/ftl/deployment/FtlProcessor.java
@@ -37,6 +37,7 @@
import org.jboss.resteasy.reactive.server.processor.scanning.MethodScanner;
import org.jetbrains.annotations.NotNull;
+import com.fasterxml.jackson.annotation.JsonAlias;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
@@ -81,7 +82,6 @@
import xyz.block.ftl.Verb;
import xyz.block.ftl.VerbName;
import xyz.block.ftl.runtime.FTLDatasourceCredentials;
-import xyz.block.ftl.runtime.FTLHttpHandler;
import xyz.block.ftl.runtime.FTLRecorder;
import xyz.block.ftl.runtime.JsonSerializationConfig;
import xyz.block.ftl.runtime.TopicHelper;
@@ -92,6 +92,7 @@
import xyz.block.ftl.runtime.builtin.HttpResponse;
import xyz.block.ftl.runtime.config.FTLConfigSource;
import xyz.block.ftl.runtime.config.FTLConfigSourceFactoryBuilder;
+import xyz.block.ftl.runtime.http.FTLHttpHandler;
import xyz.block.ftl.v1.CallRequest;
import xyz.block.ftl.v1.schema.Any;
import xyz.block.ftl.v1.schema.Array;
@@ -107,6 +108,7 @@
import xyz.block.ftl.v1.schema.IngressPathParameter;
import xyz.block.ftl.v1.schema.Int;
import xyz.block.ftl.v1.schema.Metadata;
+import xyz.block.ftl.v1.schema.MetadataAlias;
import xyz.block.ftl.v1.schema.MetadataCalls;
import xyz.block.ftl.v1.schema.MetadataCronJob;
import xyz.block.ftl.v1.schema.MetadataIngress;
@@ -296,7 +298,7 @@ public void registerVerbs(CombinedIndexBuildItem index,
for (var endpoint : restEndpoints.getEntries()) {
//TODO: naming
var verbName = methodToName(endpoint.getMethodInfo());
- recorder.registerHttpIngress(moduleName, verbName);
+ boolean base64 = false;
//TODO: handle type parameters properly
org.jboss.jandex.Type bodyParamType = VoidType.VOID;
@@ -309,13 +311,25 @@ public void registerVerbs(CombinedIndexBuildItem index,
}
}
+ if (bodyParamType instanceof ArrayType) {
+ org.jboss.jandex.Type component = ((ArrayType) bodyParamType).component();
+ if (component instanceof PrimitiveType) {
+ base64 = component.asPrimitiveType().equals(PrimitiveType.BYTE);
+ }
+ }
+
+ recorder.registerHttpIngress(moduleName, verbName, base64);
+
StringBuilder pathBuilder = new StringBuilder();
if (endpoint.getBasicResourceClassInfo().getPath() != null) {
pathBuilder.append(endpoint.getBasicResourceClassInfo().getPath());
}
if (endpoint.getResourceMethod().getPath() != null && !endpoint.getResourceMethod().getPath().isEmpty()) {
- if (pathBuilder.charAt(pathBuilder.length() - 1) != '/'
- && !endpoint.getResourceMethod().getPath().startsWith("/")) {
+ boolean builderEndsSlash = pathBuilder.charAt(pathBuilder.length() - 1) == '/';
+ boolean pathStartsSlash = endpoint.getResourceMethod().getPath().startsWith("/");
+ if (builderEndsSlash && pathStartsSlash) {
+ pathBuilder.setLength(pathBuilder.length() - 1);
+ } else if (!builderEndsSlash && !pathStartsSlash) {
pathBuilder.append('/');
}
pathBuilder.append(endpoint.getResourceMethod().getPath());
@@ -330,15 +344,17 @@ public void registerVerbs(CombinedIndexBuildItem index,
+ methodToName(endpoint.getMethodInfo())
+ " FTL does not support custom regular expressions");
} else if (i.type == URITemplate.Type.LITERAL) {
- if (i.literalText.equals("/")) {
- continue;
+ for (var part : i.literalText.split("/")) {
+ if (part.isEmpty()) {
+ continue;
+ }
+ pathComponents.add(IngressPathComponent.newBuilder()
+ .setIngressPathLiteral(IngressPathLiteral.newBuilder().setText(part))
+ .build());
}
- pathComponents.add(IngressPathComponent.newBuilder()
- .setIngressPathLiteral(IngressPathLiteral.newBuilder().setText(i.literalText.replace("/", "")))
- .build());
} else {
pathComponents.add(IngressPathComponent.newBuilder()
- .setIngressPathParameter(IngressPathParameter.newBuilder().setName(i.name.replace("/", "")))
+ .setIngressPathParameter(IngressPathParameter.newBuilder().setName(i.name))
.build());
}
}
@@ -813,8 +829,18 @@ private void buildDataElement(ExtractionContext context, Data.Builder data, DotN
//TODO: handle getters and setters properly, also Jackson annotations etc
for (var field : clazz.fields()) {
if (!Modifier.isStatic(field.flags())) {
- data.addFields(Field.newBuilder().setName(field.name())
- .setType(buildType(context, field.type(), data.getExport())).build());
+ Field.Builder builder = Field.newBuilder().setName(field.name())
+ .setType(buildType(context, field.type(), data.getExport()));
+ if (field.hasAnnotation(JsonAlias.class)) {
+ var aliases = field.annotation(JsonAlias.class);
+ if (aliases.value() != null) {
+ for (var alias : aliases.value().asStringArray()) {
+ builder.addMetadata(
+ Metadata.newBuilder().setAlias(MetadataAlias.newBuilder().setKind(0).setAlias(alias)));
+ }
+ }
+ }
+ data.addFields(builder.build());
}
}
buildDataElement(context, data, clazz.superName());
diff --git a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLRecorder.java b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLRecorder.java
index 6f0208c38e..5fd81a623d 100644
--- a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLRecorder.java
+++ b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLRecorder.java
@@ -12,6 +12,8 @@
import io.quarkus.arc.Arc;
import io.quarkus.runtime.annotations.Recorder;
+import xyz.block.ftl.runtime.http.FTLHttpHandler;
+import xyz.block.ftl.runtime.http.HTTPVerbInvoker;
import xyz.block.ftl.v1.CallRequest;
@Recorder
@@ -34,10 +36,11 @@ public void registerVerb(String module, String verbName, String methodName, List
}
}
- public void registerHttpIngress(String module, String verbName) {
+ public void registerHttpIngress(String module, String verbName, boolean base64Encoded) {
try {
- Arc.container().instance(VerbRegistry.class).get().register(module, verbName,
- Arc.container().instance(FTLHttpHandler.class).get());
+ FTLHttpHandler ftlHttpHandler = Arc.container().instance(FTLHttpHandler.class).get();
+ VerbRegistry verbRegistry = Arc.container().instance(VerbRegistry.class).get();
+ verbRegistry.register(module, verbName, new HTTPVerbInvoker(base64Encoded, ftlHttpHandler));
} catch (Exception e) {
throw new RuntimeException(e);
}
diff --git a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/JsonSerializationConfig.java b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/JsonSerializationConfig.java
index 5f19e2ca2d..a4343b4413 100644
--- a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/JsonSerializationConfig.java
+++ b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/JsonSerializationConfig.java
@@ -3,8 +3,7 @@
import java.io.IOException;
import java.util.Base64;
-import jakarta.enterprise.event.Observes;
-import jakarta.json.stream.JsonGenerator;
+import jakarta.inject.Singleton;
import com.fasterxml.jackson.core.JacksonException;
import com.fasterxml.jackson.core.JsonParser;
@@ -19,16 +18,19 @@
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.StdSerializer;
-import io.quarkus.runtime.StartupEvent;
+import io.quarkus.arc.Unremovable;
+import io.quarkus.jackson.ObjectMapperCustomizer;
/**
* This class configures the FTL serialization
*/
-public class JsonSerializationConfig {
+@Singleton
+@Unremovable
+public class JsonSerializationConfig implements ObjectMapperCustomizer {
- void startup(@Observes StartupEvent event, ObjectMapper mapper) {
+ @Override
+ public void customize(ObjectMapper mapper) {
mapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
-
SimpleModule module = new SimpleModule("ByteArraySerializer", new Version(1, 0, 0, ""));
module.addSerializer(byte[].class, new ByteArraySerializer());
module.addDeserializer(byte[].class, new ByteArrayDeserializer());
diff --git a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLHttpHandler.java b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/FTLHttpHandler.java
similarity index 75%
rename from jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLHttpHandler.java
rename to jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/FTLHttpHandler.java
index 48596bffe5..04144b79ae 100644
--- a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/FTLHttpHandler.java
+++ b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/FTLHttpHandler.java
@@ -1,11 +1,16 @@
-package xyz.block.ftl.runtime;
+package xyz.block.ftl.runtime.http;
import java.io.ByteArrayOutputStream;
import java.net.InetSocketAddress;
import java.nio.channels.Channels;
import java.nio.channels.WritableByteChannel;
import java.nio.charset.StandardCharsets;
-import java.util.*;
+import java.util.Base64;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Locale;
+import java.util.Map;
+import java.util.Set;
import java.util.concurrent.CompletableFuture;
import jakarta.inject.Singleton;
@@ -19,18 +24,27 @@
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.FileRegion;
-import io.netty.handler.codec.http.*;
+import io.netty.handler.codec.http.DefaultHttpRequest;
+import io.netty.handler.codec.http.DefaultLastHttpContent;
+import io.netty.handler.codec.http.HttpContent;
+import io.netty.handler.codec.http.HttpHeaderNames;
+import io.netty.handler.codec.http.HttpMethod;
+import io.netty.handler.codec.http.HttpResponse;
+import io.netty.handler.codec.http.HttpVersion;
+import io.netty.handler.codec.http.LastHttpContent;
import io.netty.util.ReferenceCountUtil;
import io.quarkus.netty.runtime.virtual.VirtualClientConnection;
import io.quarkus.netty.runtime.virtual.VirtualResponseHandler;
import io.quarkus.vertx.http.runtime.QuarkusHttpHeaders;
import io.quarkus.vertx.http.runtime.VertxHttpRecorder;
+import xyz.block.ftl.runtime.FTLRecorder;
+import xyz.block.ftl.runtime.builtin.HttpRequest;
import xyz.block.ftl.v1.CallRequest;
import xyz.block.ftl.v1.CallResponse;
@SuppressWarnings("unused")
@Singleton
-public class FTLHttpHandler implements VerbInvoker {
+public class FTLHttpHandler {
public static final String CONTENT_TYPE = "Content-Type";
final ObjectMapper mapper;
@@ -50,13 +64,16 @@ public FTLHttpHandler(ObjectMapper mapper) {
this.mapper = mapper;
}
- @Override
- public CallResponse handle(CallRequest in) {
+ public CallResponse handle(CallRequest in, boolean base64Encoded) {
try {
var body = mapper.createParser(in.getBody().newInput())
.readValueAs(xyz.block.ftl.runtime.builtin.HttpRequest.class);
body.getHeaders().put(FTLRecorder.X_FTL_VERB, List.of(in.getVerb().getName()));
- var ret = handleRequest(body);
+ var ret = handleRequest(body, base64Encoded);
+ if (ret.getBody() == null) {
+ ret.setBody("{}");
+ }
+ ret.getHeaders().remove("content-length");
var mappedResponse = mapper.writer().writeValueAsBytes(ret);
return CallResponse.newBuilder().setBody(ByteString.copyFrom(mappedResponse)).build();
} catch (Exception e) {
@@ -66,10 +83,10 @@ public CallResponse handle(CallRequest in) {
}
- public xyz.block.ftl.runtime.builtin.HttpResponse handleRequest(xyz.block.ftl.runtime.builtin.HttpRequest request) {
+ public xyz.block.ftl.runtime.builtin.HttpResponse handleRequest(HttpRequest request, boolean base64Encoded) {
InetSocketAddress clientAddress = null;
try {
- return nettyDispatch(clientAddress, request);
+ return nettyDispatch(clientAddress, request, base64Encoded);
} catch (Exception e) {
log.error("Request Failure", e);
xyz.block.ftl.runtime.builtin.HttpResponse res = new xyz.block.ftl.runtime.builtin.HttpResponse();
@@ -83,12 +100,15 @@ public xyz.block.ftl.runtime.builtin.HttpResponse handleRequest(xyz.block.ftl.ru
private class NettyResponseHandler implements VirtualResponseHandler {
xyz.block.ftl.runtime.builtin.HttpResponse responseBuilder = new xyz.block.ftl.runtime.builtin.HttpResponse();
+ final boolean base64Encoded;
ByteArrayOutputStream baos;
WritableByteChannel byteChannel;
final xyz.block.ftl.runtime.builtin.HttpRequest request;
CompletableFuture future = new CompletableFuture<>();
+ boolean json = false;
- public NettyResponseHandler(xyz.block.ftl.runtime.builtin.HttpRequest request) {
+ public NettyResponseHandler(boolean base64Encoded, xyz.block.ftl.runtime.builtin.HttpRequest request) {
+ this.base64Encoded = base64Encoded;
this.request = request;
}
@@ -114,6 +134,12 @@ public void handleMessage(Object msg) {
}
headers.put(name, allForName);
}
+ if (res.headers().contains(CONTENT_TYPE)) {
+ String contentType = res.headers().get(CONTENT_TYPE);
+ if (contentType != null && !contentType.isEmpty()) {
+ json = contentType.toLowerCase(Locale.ROOT).contains("application/json");
+ }
+ }
}
if (msg instanceof HttpContent) {
HttpContent content = (HttpContent) msg;
@@ -137,15 +163,21 @@ public void handleMessage(Object msg) {
}
if (msg instanceof LastHttpContent) {
if (baos != null) {
+ if (json) {
+ responseBuilder.setBody(baos.toString(StandardCharsets.UTF_8));
+ } else if (base64Encoded) {
+ responseBuilder.setBody(
+ mapper.writer().writeValueAsString(Base64.getEncoder().encodeToString(baos.toByteArray())));
+ } else {
+ responseBuilder.setBody(mapper.writer().writeValueAsString(baos.toString(StandardCharsets.UTF_8)));
+ }
List ct = responseBuilder.getHeaders().get(CONTENT_TYPE);
if (ct == null || ct.isEmpty()) {
//TODO: how to handle this
responseBuilder.setBody(baos.toString(StandardCharsets.UTF_8));
} else if (ct.get(0).contains(MediaType.TEXT_PLAIN)) {
// need to encode as JSON string
- responseBuilder.setBody(mapper.writer().writeValueAsString(baos.toString(StandardCharsets.UTF_8)));
} else {
- responseBuilder.setBody(baos.toString(StandardCharsets.UTF_8));
}
}
future.complete(responseBuilder);
@@ -167,7 +199,7 @@ public void close() {
}
private xyz.block.ftl.runtime.builtin.HttpResponse nettyDispatch(InetSocketAddress clientAddress,
- xyz.block.ftl.runtime.builtin.HttpRequest request)
+ HttpRequest request, boolean base64Encoded)
throws Exception {
QuarkusHttpHeaders quarkusHeaders = new QuarkusHttpHeaders();
quarkusHeaders.setContextObject(xyz.block.ftl.runtime.builtin.HttpRequest.class, request);
@@ -212,10 +244,22 @@ private xyz.block.ftl.runtime.builtin.HttpResponse nettyDispatch(InetSocketAddre
if (request.getBody() != null) {
// See https://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.3
nettyRequest.headers().add(HttpHeaderNames.TRANSFER_ENCODING, "chunked");
- ByteBuf body = Unpooled.copiedBuffer(request.getBody().toString(), StandardCharsets.UTF_8); //TODO: do we need to look at the request encoding?
- requestContent = new DefaultLastHttpContent(body);
+ if (base64Encoded) {
+ requestContent = new DefaultLastHttpContent(
+ Unpooled.copiedBuffer(Base64.getDecoder().decode(request.getBody().asText())));
+ } else if (request.getBody().isTextual()) {
+ requestContent = new DefaultLastHttpContent(
+ Unpooled.copiedBuffer(request.getBody().asText(), StandardCharsets.UTF_8));
+ } else if (request.getBody().isBigDecimal() || request.getBody().isDouble() || request.getBody().isFloat()
+ || request.getBody().isInt() || request.getBody().isBigInteger()) {
+ requestContent = new DefaultLastHttpContent(
+ Unpooled.copiedBuffer(request.getBody().toString(), StandardCharsets.UTF_8));
+ } else {
+ ByteBuf body = Unpooled.copiedBuffer(request.getBody().toString(), StandardCharsets.UTF_8); //TODO: do we need to look at the request encoding?
+ requestContent = new DefaultLastHttpContent(body);
+ }
}
- NettyResponseHandler handler = new NettyResponseHandler(request);
+ NettyResponseHandler handler = new NettyResponseHandler(base64Encoded, request);
VirtualClientConnection connection = VirtualClientConnection.connect(handler, VertxHttpRecorder.VIRTUAL_HTTP,
clientAddress);
diff --git a/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/HTTPVerbInvoker.java b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/HTTPVerbInvoker.java
new file mode 100644
index 0000000000..b5d3c32bb5
--- /dev/null
+++ b/jvm-runtime/ftl-runtime/common/runtime/src/main/java/xyz/block/ftl/runtime/http/HTTPVerbInvoker.java
@@ -0,0 +1,24 @@
+package xyz.block.ftl.runtime.http;
+
+import xyz.block.ftl.runtime.VerbInvoker;
+import xyz.block.ftl.v1.CallRequest;
+import xyz.block.ftl.v1.CallResponse;
+
+public class HTTPVerbInvoker implements VerbInvoker {
+
+ /**
+ * If this is true then the request is base 64 encoded bytes
+ */
+ final boolean base64Encoded;
+ final FTLHttpHandler ftlHttpHandler;
+
+ public HTTPVerbInvoker(boolean base64Encoded, FTLHttpHandler ftlHttpHandler) {
+ this.base64Encoded = base64Encoded;
+ this.ftlHttpHandler = ftlHttpHandler;
+ }
+
+ @Override
+ public CallResponse handle(CallRequest in) {
+ return ftlHttpHandler.handle(in, base64Encoded);
+ }
+}
diff --git a/jvm-runtime/ftl-runtime/java/integration-tests/pom.xml b/jvm-runtime/ftl-runtime/java/integration-tests/pom.xml
index 3577f02602..2161409dc3 100644
--- a/jvm-runtime/ftl-runtime/java/integration-tests/pom.xml
+++ b/jvm-runtime/ftl-runtime/java/integration-tests/pom.xml
@@ -29,6 +29,12 @@
rest-assured
test
+
+
+ xyz.block.ftl
+ ftl-java-runtime-deployment
+ test
+
io.quarkus
quarkus-junit5-mockito
diff --git a/jvm-runtime/ftl-runtime/java/integration-tests/src/main/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResource.java b/jvm-runtime/ftl-runtime/java/integration-tests/src/main/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResource.java
index 55f9fe3007..85907b006d 100644
--- a/jvm-runtime/ftl-runtime/java/integration-tests/src/main/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResource.java
+++ b/jvm-runtime/ftl-runtime/java/integration-tests/src/main/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResource.java
@@ -16,24 +16,36 @@
*/
package xyz.block.ftl.java.runtime.it;
+import java.nio.charset.StandardCharsets;
+
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.ws.rs.Consumes;
import jakarta.ws.rs.POST;
+import jakarta.ws.rs.Path;
import jakarta.ws.rs.core.MediaType;
import ftl.echo.EchoClient;
import ftl.echo.EchoRequest;
import xyz.block.ftl.Verb;
+@Path("/test")
@ApplicationScoped
public class FtlJavaRuntimeResource {
@POST
@Consumes(MediaType.APPLICATION_JSON)
+ @Path("/post")
public String post(Person person) {
return "Hello " + person.first() + " " + person.last();
}
+ @POST
+ @Consumes(MediaType.APPLICATION_JSON)
+ @Path("/bytes")
+ public String bytesHttp(byte[] data) {
+ return "Hello " + new String(data, StandardCharsets.UTF_8);
+ }
+
@Verb
public String hello(String name, EchoClient echoClient) {
return "Hello " + echoClient.call(new EchoRequest().setName(name)).getMessage();
diff --git a/jvm-runtime/ftl-runtime/java/integration-tests/src/test/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResourceTest.java b/jvm-runtime/ftl-runtime/java/integration-tests/src/test/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResourceTest.java
index 77b17a2df0..a440670c37 100644
--- a/jvm-runtime/ftl-runtime/java/integration-tests/src/test/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResourceTest.java
+++ b/jvm-runtime/ftl-runtime/java/integration-tests/src/test/java/xyz/block/ftl/java/runtime/it/FtlJavaRuntimeResourceTest.java
@@ -1,5 +1,8 @@
package xyz.block.ftl.java.runtime.it;
+import java.nio.charset.StandardCharsets;
+import java.util.Base64;
+import java.util.HashMap;
import java.util.function.Function;
import jakarta.inject.Inject;
@@ -9,10 +12,12 @@
import org.junit.jupiter.api.Test;
import org.mockito.Mockito;
+import ftl.builtin.HttpRequest;
+import ftl.builtin.HttpResponse;
import ftl.echo.EchoClient;
import ftl.echo.EchoRequest;
import ftl.echo.EchoResponse;
-import io.quarkus.test.common.QuarkusTestResource;
+import io.quarkus.test.common.WithTestResource;
import io.quarkus.test.junit.QuarkusTest;
import xyz.block.ftl.VerbClient;
import xyz.block.ftl.VerbClientDefinition;
@@ -22,7 +27,7 @@
import xyz.block.ftl.java.test.internal.TestVerbServer;
@QuarkusTest
-@QuarkusTestResource(FTLTestResource.class)
+@WithTestResource(FTLTestResource.class)
public class FtlJavaRuntimeResourceTest {
@FTLManaged
@@ -37,6 +42,14 @@ public class FtlJavaRuntimeResourceTest {
@Inject
BytesClient bytesClient;
+ @FTLManaged
+ @Inject
+ PostClient postClient;
+
+ @FTLManaged
+ @Inject
+ BytesHTTPClient bytesHttpClient;
+
@Test
public void testHelloEndpoint() {
TestVerbServer.registerFakeVerb("echo", "echo", new Function() {
@@ -61,6 +74,33 @@ public void testBytesSerialization() {
Assertions.assertArrayEquals(new byte[] { 1, 2 }, bytesClient.call(new byte[] { 1, 2 }));
}
+ @Test
+ public void testHttpPost() {
+ HttpRequest request = new HttpRequest()
+ .setMethod("POST")
+ .setPath("/test/post")
+ .setQuery(new HashMap<>())
+ .setPathParameters(new HashMap<>())
+ .setHeaders(new HashMap<>())
+ .setBody(new Person("Stuart", "Douglas"));
+ HttpResponse response = postClient.call(request);
+ Assertions.assertEquals("Hello Stuart Douglas", response.getBody());
+ }
+
+ @Test
+ public void testHttpBytes() {
+ HttpRequest request = new HttpRequest()
+ .setMethod("POST")
+ .setPath("/test/bytes")
+ .setQuery(new HashMap<>())
+ .setPathParameters(new HashMap<>())
+ .setHeaders(new HashMap<>())
+ .setBody("Stuart Douglas".getBytes(java.nio.charset.StandardCharsets.UTF_8));
+ HttpResponse response = bytesHttpClient.call(request);
+ Assertions.assertArrayEquals("Hello Stuart Douglas".getBytes(StandardCharsets.UTF_8),
+ Base64.getDecoder().decode(response.getBody()));
+ }
+
@VerbClientDefinition(name = "publish")
interface PublishVerbClient extends VerbClientSink {
}
@@ -72,4 +112,12 @@ interface HelloClient extends VerbClient {
@VerbClientDefinition(name = "bytes")
interface BytesClient extends VerbClient {
}
+
+ @VerbClientDefinition(name = "post")
+ interface PostClient extends VerbClient, HttpResponse> {
+ }
+
+ @VerbClientDefinition(name = "bytesHttp")
+ interface BytesHTTPClient extends VerbClient, HttpResponse> {
+ }
}
diff --git a/jvm-runtime/ftl-runtime/pom.xml b/jvm-runtime/ftl-runtime/pom.xml
index 93380fc60f..efc4c74995 100644
--- a/jvm-runtime/ftl-runtime/pom.xml
+++ b/jvm-runtime/ftl-runtime/pom.xml
@@ -50,6 +50,11 @@
ftl-java-runtime
${project.version}
+
+ xyz.block.ftl
+ ftl-java-runtime-deployment
+ ${project.version}
+
xyz.block.ftl
ftl-kotlin-runtime