diff --git a/cmd/docgen/docs/base_test.go b/cmd/docgen/docs/base_test.go index c8053bc39..7b92e274d 100644 --- a/cmd/docgen/docs/base_test.go +++ b/cmd/docgen/docs/base_test.go @@ -46,7 +46,7 @@ func TestGenerateDocs(t *testing.T) { assert.Equal(t, 88, len(functions)) types := context["types"].([]any) - assert.Equal(t, 19, len(types)) + assert.Equal(t, 20, len(types)) root := context["root"].([]any) assert.Equal(t, 14, len(root)) diff --git a/flows/actions/base.go b/flows/actions/base.go index 0ef65c0b3..48c98c4f7 100644 --- a/flows/actions/base.go +++ b/flows/actions/base.go @@ -13,7 +13,6 @@ import ( "github.com/nyaruka/gocommon/urns" "github.com/nyaruka/gocommon/uuids" "github.com/nyaruka/goflow/assets" - "github.com/nyaruka/goflow/excellent/types" "github.com/nyaruka/goflow/flows" "github.com/nyaruka/goflow/flows/events" "github.com/nyaruka/goflow/flows/modifiers" @@ -156,17 +155,6 @@ func (a *baseAction) saveWebhookResult(run flows.Run, step flows.Step, name stri a.saveResult(run, step, name, value, category, "", input, extra, logEvent) } -func (a *baseAction) updateWebhook(run flows.Run, call *flows.WebhookCall) { - parsed := types.JSONToXValue(call.ResponseJSON) - - switch typed := parsed.(type) { - case nil, *types.XError: - run.SetWebhook(types.XObjectEmpty) - default: - run.SetWebhook(typed) - } -} - // helper to apply a contact modifier func (a *baseAction) applyModifier(run flows.Run, mod flows.Modifier, logModifier flows.ModifierCallback, logEvent flows.EventCallback) bool { logModifier(mod) diff --git a/flows/actions/base_test.go b/flows/actions/base_test.go index c49f28d76..8dd80632c 100644 --- a/flows/actions/base_test.go +++ b/flows/actions/base_test.go @@ -257,7 +257,7 @@ func testActionType(t *testing.T, assetsJSON json.RawMessage, typeName string) { actual.Events, _ = jsonx.Marshal(runEvents[ignoreEventCount:]) if tc.Webhook != nil { - actual.Webhook, _ = jsonx.Marshal(run.Webhook()) + actual.Webhook = jsonx.MustMarshal(run.Webhook()) } if tc.ContactAfter != nil { actual.ContactAfter, _ = jsonx.Marshal(session.Contact()) diff --git a/flows/actions/call_resthook.go b/flows/actions/call_resthook.go index a3f70f30a..52527c5ab 100644 --- a/flows/actions/call_resthook.go +++ b/flows/actions/call_resthook.go @@ -130,7 +130,7 @@ func (a *CallResthookAction) Execute(run flows.Run, step flows.Step, logModifier asResult := a.pickResultCall(calls) if asResult != nil { - a.updateWebhook(run, asResult) + run.SetWebhook(asResult) } if a.ResultName != "" { diff --git a/flows/actions/call_webhook.go b/flows/actions/call_webhook.go index 45a65247f..89de8be77 100644 --- a/flows/actions/call_webhook.go +++ b/flows/actions/call_webhook.go @@ -135,7 +135,7 @@ func (a *CallWebhookAction) call(run flows.Run, step flows.Step, url, method, bo logEvent(events.NewError(err)) } if call != nil { - a.updateWebhook(run, call) + run.SetWebhook(call) status := callStatus(call, err, false) diff --git a/flows/actions/testdata/_assets.json b/flows/actions/testdata/_assets.json index 6a4a3337b..06984f663 100644 --- a/flows/actions/testdata/_assets.json +++ b/flows/actions/testdata/_assets.json @@ -3,7 +3,7 @@ { "uuid": "bead76f5-dac4-4c9d-996c-c62b326e8c0a", "name": "Action Tester", - "spec_version": "13.2", + "spec_version": "13.3.0", "language": "eng", "type": "messaging", "revision": 123, diff --git a/flows/actions/testdata/call_resthook.json b/flows/actions/testdata/call_resthook.json index a98f584d2..432a4ebdb 100644 --- a/flows/actions/testdata/call_resthook.json +++ b/flows/actions/testdata/call_resthook.json @@ -21,13 +21,19 @@ "http://temba.io/": [ { "status": 200, + "headers": { + "Content-Type": "application/json" + }, "body": "{ \"ok\": \"true\" }" } ], "http://unavailable.com/": [ { "status": 503, - "body": "{ \"errors\": [\"service unavailable\"] }" + "headers": { + "Content-Type": "text/plain" + }, + "body": "service unavailable" } ] }, @@ -99,7 +105,7 @@ "url": "http://temba.io/", "status_code": 200, "request": "POST / HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Length: 898\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n{\"channel\":null,\"contact\":{\"language\":\"eng\",\"name\":\"Ryan Lewis\",\"urn\":\"tel:+12065551212\",\"uuid\":\"5d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f\"},\"flow\":{\"name\":\"Action Tester\",\"revision\":123,\"uuid\":\"bead76f5-dac4-4c9d-996c-c62b326e8c0a\"},\"input\":{\"attachments\":[{\"content_type\":\"image/jpeg\",\"url\":\"http://http://s3.amazon.com/bucket/test.jpg\"},{\"content_type\":\"audio/mp3\",\"url\":\"http://s3.amazon.com/bucket/test.mp3\"}],\"channel\":null,\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"text\":\"Hi everybody\",\"type\":\"msg\",\"urn\":{\"display\":\"(206) 555-1212\",\"path\":\"+12065551212\",\"scheme\":\"tel\"},\"uuid\":\"aa90ce99-3b4d-44ba-b0ca-79e63d9ed842\"},\"path\":[{\"arrived_on\":\"2018-10-18T14:20:30.000123Z\",\"exit_uuid\":\"\",\"node_uuid\":\"72a1f5df-49f9-45df-94c9-d86f7ea064e5\",\"uuid\":\"59d74b86-3e2f-4a93-aece-b05d2fdcde0c\"}],\"results\":{},\"run\":{\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"uuid\":\"e7187099-7d38-4f60-955c-325957214c42\"}}", - "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\n\r\n{ \"ok\": \"true\" }", + "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\nContent-Type: application/json\r\n\r\n{ \"ok\": \"true\" }", "elapsed_ms": 0, "retries": 0, "status": "success", @@ -113,12 +119,12 @@ "url": "http://unavailable.com/", "status_code": 503, "request": "POST / HTTP/1.1\r\nHost: unavailable.com\r\nUser-Agent: goflow-testing\r\nContent-Length: 898\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n{\"channel\":null,\"contact\":{\"language\":\"eng\",\"name\":\"Ryan Lewis\",\"urn\":\"tel:+12065551212\",\"uuid\":\"5d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f\"},\"flow\":{\"name\":\"Action Tester\",\"revision\":123,\"uuid\":\"bead76f5-dac4-4c9d-996c-c62b326e8c0a\"},\"input\":{\"attachments\":[{\"content_type\":\"image/jpeg\",\"url\":\"http://http://s3.amazon.com/bucket/test.jpg\"},{\"content_type\":\"audio/mp3\",\"url\":\"http://s3.amazon.com/bucket/test.mp3\"}],\"channel\":null,\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"text\":\"Hi everybody\",\"type\":\"msg\",\"urn\":{\"display\":\"(206) 555-1212\",\"path\":\"+12065551212\",\"scheme\":\"tel\"},\"uuid\":\"aa90ce99-3b4d-44ba-b0ca-79e63d9ed842\"},\"path\":[{\"arrived_on\":\"2018-10-18T14:20:30.000123Z\",\"exit_uuid\":\"\",\"node_uuid\":\"72a1f5df-49f9-45df-94c9-d86f7ea064e5\",\"uuid\":\"59d74b86-3e2f-4a93-aece-b05d2fdcde0c\"}],\"results\":{},\"run\":{\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"uuid\":\"e7187099-7d38-4f60-955c-325957214c42\"}}", - "response": "HTTP/1.0 503 Service Unavailable\r\nContent-Length: 37\r\n\r\n{ \"errors\": [\"service unavailable\"] }", + "response": "HTTP/1.0 503 Service Unavailable\r\nContent-Length: 19\r\nContent-Type: text/plain\r\n\r\nservice unavailable", "elapsed_ms": 0, "retries": 0, "status": "response_error", "resthook": "new-registration", - "extraction": "valid" + "extraction": "ignored" } ], "inspection": { @@ -135,13 +141,19 @@ "http://temba.io/": [ { "status": 200, + "headers": { + "Content-Type": "application/json" + }, "body": "{ \"ok\": \"true\" }" } ], "http://unavailable.com/": [ { "status": 503, - "body": "{ \"errors\": [\"service unavailable\"] }" + "headers": { + "Content-Type": "text/plain" + }, + "body": "service unavailable" } ] }, @@ -214,7 +226,7 @@ "url": "http://temba.io/", "status_code": 200, "request": "POST / HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Length: 898\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n{\"channel\":null,\"contact\":{\"language\":\"eng\",\"name\":\"Ryan Lewis\",\"urn\":\"tel:+12065551212\",\"uuid\":\"5d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f\"},\"flow\":{\"name\":\"Action Tester\",\"revision\":123,\"uuid\":\"bead76f5-dac4-4c9d-996c-c62b326e8c0a\"},\"input\":{\"attachments\":[{\"content_type\":\"image/jpeg\",\"url\":\"http://http://s3.amazon.com/bucket/test.jpg\"},{\"content_type\":\"audio/mp3\",\"url\":\"http://s3.amazon.com/bucket/test.mp3\"}],\"channel\":null,\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"text\":\"Hi everybody\",\"type\":\"msg\",\"urn\":{\"display\":\"(206) 555-1212\",\"path\":\"+12065551212\",\"scheme\":\"tel\"},\"uuid\":\"aa90ce99-3b4d-44ba-b0ca-79e63d9ed842\"},\"path\":[{\"arrived_on\":\"2018-10-18T14:20:30.000123Z\",\"exit_uuid\":\"\",\"node_uuid\":\"72a1f5df-49f9-45df-94c9-d86f7ea064e5\",\"uuid\":\"59d74b86-3e2f-4a93-aece-b05d2fdcde0c\"}],\"results\":{},\"run\":{\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"uuid\":\"e7187099-7d38-4f60-955c-325957214c42\"}}", - "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\n\r\n{ \"ok\": \"true\" }", + "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\nContent-Type: application/json\r\n\r\n{ \"ok\": \"true\" }", "elapsed_ms": 0, "retries": 0, "status": "success", @@ -228,12 +240,12 @@ "url": "http://unavailable.com/", "status_code": 503, "request": "POST / HTTP/1.1\r\nHost: unavailable.com\r\nUser-Agent: goflow-testing\r\nContent-Length: 898\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n{\"channel\":null,\"contact\":{\"language\":\"eng\",\"name\":\"Ryan Lewis\",\"urn\":\"tel:+12065551212\",\"uuid\":\"5d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f\"},\"flow\":{\"name\":\"Action Tester\",\"revision\":123,\"uuid\":\"bead76f5-dac4-4c9d-996c-c62b326e8c0a\"},\"input\":{\"attachments\":[{\"content_type\":\"image/jpeg\",\"url\":\"http://http://s3.amazon.com/bucket/test.jpg\"},{\"content_type\":\"audio/mp3\",\"url\":\"http://s3.amazon.com/bucket/test.mp3\"}],\"channel\":null,\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"text\":\"Hi everybody\",\"type\":\"msg\",\"urn\":{\"display\":\"(206) 555-1212\",\"path\":\"+12065551212\",\"scheme\":\"tel\"},\"uuid\":\"aa90ce99-3b4d-44ba-b0ca-79e63d9ed842\"},\"path\":[{\"arrived_on\":\"2018-10-18T14:20:30.000123Z\",\"exit_uuid\":\"\",\"node_uuid\":\"72a1f5df-49f9-45df-94c9-d86f7ea064e5\",\"uuid\":\"59d74b86-3e2f-4a93-aece-b05d2fdcde0c\"}],\"results\":{},\"run\":{\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"uuid\":\"e7187099-7d38-4f60-955c-325957214c42\"}}", - "response": "HTTP/1.0 503 Service Unavailable\r\nContent-Length: 37\r\n\r\n{ \"errors\": [\"service unavailable\"] }", + "response": "HTTP/1.0 503 Service Unavailable\r\nContent-Length: 19\r\nContent-Type: text/plain\r\n\r\nservice unavailable", "elapsed_ms": 0, "retries": 0, "status": "response_error", "resthook": "new-registration", - "extraction": "valid" + "extraction": "ignored" }, { "type": "run_result_changed", @@ -242,18 +254,16 @@ "name": "My Result", "value": "503", "category": "Failure", - "input": "POST http://unavailable.com/", - "extra": { - "errors": [ - "service unavailable" - ] - } + "input": "POST http://unavailable.com/" } ], "webhook": { - "errors": [ - "service unavailable" - ] + "__default__": "POST http://unavailable.com/", + "headers": { + "Content-Type": "text/plain" + }, + "json": null, + "status": 503 }, "inspection": { "dependencies": [], @@ -287,6 +297,9 @@ "http://temba.io/": [ { "status": 200, + "headers": { + "Content-Type": "application/json" + }, "body": "{ \"ok\": \"true\" }" } ] @@ -360,7 +373,7 @@ "url": "http://temba.io/", "status_code": 200, "request": "POST / HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Length: 898\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n{\"channel\":null,\"contact\":{\"language\":\"eng\",\"name\":\"Ryan Lewis\",\"urn\":\"tel:+12065551212\",\"uuid\":\"5d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f\"},\"flow\":{\"name\":\"Action Tester\",\"revision\":123,\"uuid\":\"bead76f5-dac4-4c9d-996c-c62b326e8c0a\"},\"input\":{\"attachments\":[{\"content_type\":\"image/jpeg\",\"url\":\"http://http://s3.amazon.com/bucket/test.jpg\"},{\"content_type\":\"audio/mp3\",\"url\":\"http://s3.amazon.com/bucket/test.mp3\"}],\"channel\":null,\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"text\":\"Hi everybody\",\"type\":\"msg\",\"urn\":{\"display\":\"(206) 555-1212\",\"path\":\"+12065551212\",\"scheme\":\"tel\"},\"uuid\":\"aa90ce99-3b4d-44ba-b0ca-79e63d9ed842\"},\"path\":[{\"arrived_on\":\"2018-10-18T14:20:30.000123Z\",\"exit_uuid\":\"\",\"node_uuid\":\"72a1f5df-49f9-45df-94c9-d86f7ea064e5\",\"uuid\":\"59d74b86-3e2f-4a93-aece-b05d2fdcde0c\"}],\"results\":{},\"run\":{\"created_on\":\"2018-10-18T14:20:30.000123Z\",\"uuid\":\"e7187099-7d38-4f60-955c-325957214c42\"}}", - "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\n\r\n{ \"ok\": \"true\" }", + "response": "HTTP/1.0 200 OK\r\nContent-Length: 16\r\nContent-Type: application/json\r\n\r\n{ \"ok\": \"true\" }", "elapsed_ms": 0, "retries": 0, "status": "success", @@ -395,7 +408,14 @@ } ], "webhook": { - "ok": "true" + "__default__": "POST http://temba.io/", + "headers": { + "Content-Type": "application/json" + }, + "json": { + "ok": "true" + }, + "status": 200 }, "inspection": { "dependencies": [], @@ -622,9 +642,14 @@ } ], "webhook": { - "errors": [ - "service unavailable" - ] + "__default__": "POST http://unavailable.com/", + "headers": {}, + "json": { + "errors": [ + "service unavailable" + ] + }, + "status": 503 }, "inspection": { "dependencies": [], diff --git a/flows/actions/testdata/call_webhook.json b/flows/actions/testdata/call_webhook.json index cfd5a3140..0538f20fa 100644 --- a/flows/actions/testdata/call_webhook.json +++ b/flows/actions/testdata/call_webhook.json @@ -29,6 +29,9 @@ "http://temba.io/?q=": [ { "status": 200, + "headers": { + "Content-Type": "application/json" + }, "body": "{ \"ok\": true }" } ] @@ -69,7 +72,7 @@ "url": "http://temba.io/?q=", "status_code": 200, "request": "POST /?q= HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Length: 0\r\nAuthentication: Token -\r\nAccept-Encoding: gzip\r\n\r\n", - "response": "HTTP/1.0 200 OK\r\nContent-Length: 14\r\n\r\n{ \"ok\": true }", + "response": "HTTP/1.0 200 OK\r\nContent-Length: 14\r\nContent-Type: application/json\r\n\r\n{ \"ok\": true }", "elapsed_ms": 0, "retries": 0, "status": "success", @@ -77,7 +80,14 @@ } ], "webhook": { - "ok": true + "__default__": "POST http://temba.io/?q=", + "headers": { + "Content-Type": "application/json" + }, + "json": { + "ok": true + }, + "status": 200 }, "templates": [ "http://temba.io/?q=@(1 / 0)", @@ -114,7 +124,7 @@ "text": "webhook URL evaluated to empty string" } ], - "webhook": {}, + "webhook": null, "templates": [ "@(\"\")" ], @@ -142,7 +152,7 @@ "text": "webhook URL evaluated to an invalid URL: ':xxxxx'" } ], - "webhook": {}, + "webhook": null, "templates": [ "@(\":xxxxx\")" ], @@ -170,7 +180,7 @@ "text": "webhook URL evaluated to an invalid URL: 'http://example.com?%7B%22contact%22%3A%7B%22channel%22%3A%7B%22address%22%3A%22%2B17036975131%22%2C%22name%22%3A%22My%20Android%20Phone%22%2C%22uuid%22%3A%2257f1078f-88aa-46f4-a59a-948a5739c03d%22%7D%2C%22created_on%22%3A%222018-06-20T11%3A40%3A30.123456Z%22%2C%22fields%22%3A%7B%22age%22%3Anull%2C%22gender%22%3A%22Male%22%7D%2C%22first_name%22%3A%22Ryan%22%2C%22groups%22%3A%5B%7B%22name%22%3A%22Testers%22%2C%22uuid%22%3A%22b7cf0d83-f1c9-411c-96fd-c511a4cfa86d%22%7D%2C%7B%22name%22%3A%22Males%22%2C%22uuid%22%3A%220ec97956-c451-48a0-a180-1ce766623e31%22%7D%5D%2C%22id%22%3A%220%22%2C%22language%22%3A%22eng%22%2C%22last_seen_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22name%22%3A%22Ryan%20Lewis%22%2C%22status%22%3A%22active%22%2C%22tickets%22%3A%5B%5D%2C%22timezone%22%3A%22America%2FGuayaquil%22%2C%22urn%22%3A%22tel%3A%2B12065551212%22%2C%22urns%22%3A%5B%22tel%3A%2B12065551212%22%2C%22twitterid%3A54784326227%23nyaruka%22%5D%2C%22uuid%22%3A%225d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f%22%7D%2C%22created_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22exited_on%22%3Anull%2C%22flow%22%3A%7B%22name%22%3A%22Action%20Tester%22%2C%22revision%22%3A123%2C%22uuid%22%3A%22bead76f5-dac4-4c9d-996c-c62b326e8c0a%22%7D%2C%22path%22%3A%5B%7B%22arrived_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22exit_uuid%22%3A%22%22%2C%22node_uuid%22%3A%2272a1f5df-49f9-45df-94c9-d86f7ea064e5%22%2C%22uuid%22%3A%2259d74b86-3e2f-4a93-aece-b05d2fdcde0c%22%7D%5D%2C%22results%22%3A%7B%7D%2C%22status%22%3A%22active%22%2C%22uuid%22%3A%22e7187099-7d38-4f60-955c-325957214c42%22%7D%7B%22contact%22%3A%7B%22channel%22%3A%7B%22address%22%3A%22%2B17036975131%22%2C%22name%22%3A%22My%20Android%20Phone%22%2C%22uuid%22%3A%2257f1078f-88aa-46f4-a59a-948a5739c03d%22%7D%2C%22created_on%22%3A%222018-06-20T11%3A40%3A30.123456Z%22%2C%22fields%22%3A%7B%22age%22%3Anull%2C%22gender%22%3A%22Male%22%7D%2C%22first_name%22%3A%22Ryan%22%2C%22groups%22%3A%5B%7B%22name%22%3A%22Testers%22%2C%22uuid%22%3A%22b7cf0d83-f1c9-411c-96fd-c511a4cfa86d%22%7D%2C%7B%22name%22%3A%22Males%22%2C%22uuid%22%3A%220ec97956-c451-48a0-a180-1ce766623e31%22%7D%5D%2C%22id%22%3A%220%22%2C%22language%22%3A%22eng%22%2C%22last_seen_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22name%22%3A%22Ryan%20Lewis%22%2C%22status%22%3A%22active%22%2C%22tickets%22%3A%5B%5D%2C%22timezone%22%3A%22America%2FGuayaquil%22%2C%22urn%22%3A%22tel%3A%2B12065551212%22%2C%22urns%22%3A%5B%22tel%3A%2B12065551212%22%2C%22twitterid%3A54784326227%23nyaruka%22%5D%2C%22uuid%22%3A%225d76d86b-3bb9-4d5a-b822-c9d86f5d8e4f%22%7D%2C%22created_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22exited_on%22%3Anull%2C%22flow%22%3A%7B%22name%22%3A%22Action%20Tester%22%2C%22revision%22%3A123%2C%22uuid%22%3A%22bead76f5-dac4-4c9d-996c-c62b326e8c0a%22%7D%2C%22path%22%3A%5B%7B%22arrived_on%22%3A%222018-10-18T14%3A20%3A30.000123Z%22%2C%22exit_uuid%22%3A%22%22%2C%22node_uuid%22%3A%2272a1f5df-49f9-45df-94c9-d86f7ea064e5%22%2C%22uuid%22%3A%2259d74b86-3e2f-4a93-aece-b05d2fdcde0c%22%7D%5D%2C%22results%22%3A%7B%7D%2C%22status%22%3A%22active%22%2C%22uuid%22%3A%22e7187099-7d38-4f60-955c-325957214c42%22%7D'" } ], - "webhook": {}, + "webhook": null, "templates": [ "http://example.com?@(url_encode(json(run)) & url_encode(json(run)))" ], @@ -188,6 +198,9 @@ "http://temba.io/": [ { "status": 200, + "headers": { + "Content-Type": "application/json" + }, "body": "{ \"ok\": true }" } ] @@ -211,7 +224,7 @@ "url": "http://temba.io/", "status_code": 200, "request": "POST / HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Length: 9\r\nX-Something: Male\r\nAccept-Encoding: gzip\r\n\r\nHi there!", - "response": "HTTP/1.0 200 OK\r\nContent-Length: 14\r\n\r\n{ \"ok\": true }", + "response": "HTTP/1.0 200 OK\r\nContent-Length: 14\r\nContent-Type: application/json\r\n\r\n{ \"ok\": true }", "elapsed_ms": 0, "retries": 0, "status": "success", @@ -231,7 +244,14 @@ } ], "webhook": { - "ok": true + "__default__": "POST http://temba.io/", + "headers": { + "Content-Type": "application/json" + }, + "json": { + "ok": true + }, + "status": 200 }, "templates": [ "http://temba.io/", @@ -297,7 +317,12 @@ } ], "webhook": { - "ok": true + "__default__": "POST http://temba.io/", + "headers": {}, + "json": { + "ok": true + }, + "status": 200 }, "templates": [ " http://temba.io/ ", @@ -355,7 +380,14 @@ "input": "GET http://temba.io/" } ], - "webhook": {}, + "webhook": { + "__default__": "GET http://temba.io/", + "headers": { + "Content-Type": "application/json" + }, + "json": null, + "status": 200 + }, "templates": [ "http://temba.io/" ], @@ -421,7 +453,12 @@ } ], "webhook": { - "big": "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Proin sed nunc vehicula, commodo ipsum et, consectetur massa. Suspendisse potenti. Ut feugiat volutpat purus vel viverra. Fusce commodo, massa eget malesuada aliquam, dolor lectus porta tortor, ultrices lobortis lacus tellus non velit. Interdum et malesuada fames ac ante ipsum primis in faucibus. Phasellus in viverra metus. Ut lobortis metus elit, elementum posuere ex consequat non. Donec elementum rutrum orci non dictum. Nam ut ultricies nisi, a viverra nisl. Sed et nibh vitae metus bibendum lobortis sed in ex. Nunc porta elit eget ipsum bibendum gravida. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. Suspendisse potenti. Etiam quis ligula quis lacus ultricies fringilla. Integer nec pharetra nunc. Curabitur pharetra, dolor fringilla ultricies ornare, purus nisl ultrices augue, nec pulvinar ipsum orci consequat quam. Proin auctor justo non eleifend facilisis. Praesent eget justo elit. Ut nec augue purus. Cras nulla risus, bibendum ac est ut, pharetra bibendum elit. Integer interdum, lorem nec pellentesque ornare, nulla mauris pretium arcu, non lacinia risus odio vel nibh. Proin bibendum nulla vel nulla lacinia faucibus. Quisque accumsan sapien malesuada, pulvinar elit non, sollicitudin enim. Aliquam iaculis, massa non tempus hendrerit, ante nunc semper orci, et pretium libero tellus at urna. Nullam maximus sem condimentum, vestibulum eros sit amet, elementum odio. Mauris nisl augue, tristique id eleifend at, elementum vitae elit. Aenean ut iaculis felis. Curabitur id mollis sem. Phasellus quis bibendum est, id hendrerit nulla. Sed consequat metus ex, vitae pharetra lorem commodo id. Etiam eu nisl a ligula laoreet semper. Maecenas non ornare urna. Vestibulum posuere sapien quis dolor scelerisque euismod. Fusce eget neque ac nisl auctor commodo id vitae massa. Suspendisse tincidunt leo at erat dignissim imperdiet. Fusce pulvinar consectetur vehicula. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Praesent ut aliquet dui. Fusce at sollicitudin urna. Vivamus sed neque elit. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Vivamus sed urna accumsan nulla euismod pulvinar eu ut tortor. In egestas id lectus at ultrices. Nam a cursus lectus, a laoreet lectus. Vivamus pharetra sapien vel diam hendrerit, vitae consequat quam iaculis. Orci varius natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Etiam ac felis at velit venenatis molestie. Nunc risus lacus, fringilla eu libero sit amet, dictum interdum tortor. In nec viverra est. Cras imperdiet urna eget ullamcorper consectetur. In faucibus finibus quam. Nulla dictum dolor tristique, rutrum nisi ut, facilisis sem. Aliquam erat volutpat. Donec scelerisque nec justo sed lobortis. Donec posuere, mi vitae molestie finibus, felis lectus facilisis ligula, et dignissim magna diam at turpis. Integer ac ante pulvinar ipsum malesuada convallis eu a orci. Nunc nec accumsan felis. Sed quam purus, bibendum eget mattis non, sodales ut felis. Ut a erat et orci elementum vulputate in eget purus. Suspendisse et posuere lectus. Fusce tempor enim arcu, eu ultricies erat condimentum pulvinar. Nam luctus consequat lectus, eget elementum nunc varius sed. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed sit amet posuere velit, eu vehicula erat. Nulla faucibus at dolor in tincidunt. Nam quis neque ut dui congue laoreet. Etiam ex metus, laoreet lobortis magna non, vehicula dapibus tellus. Integer sit amet orci aliquam, venenatis risus sit amet, cursus metus. Aenean sit amet lectus id neque eleifend pellentesque sed vitae sapien. Vivamus lacus risus, volutpat et mauris quis, porttitor tempus nisl. In tincidunt, elit semper varius posuere, arcu nulla suscipit urna, sit amet posuere ipsum leo a est. Cras ipsum sapien, varius sed mauris a, aliquam consectetur nisi. Integer et ante sit amet tellus dictum sagittis et in lorem. Nulla vel diam elementum, maximus libero dictum, semper orci. Phasellus in facilisis tortor, in vulputate purus. Vivamus rhoncus sem tempus, pharetra turpis vitae, laoreet sem. Pellentesque egestas tellus velit. In laoreet tempor erat. Ut dui erat, pulvinar eu libero imperdiet, fringilla tincidunt est. Ut vitae lectus non velit mollis euismod. Fusce risus neque, sodales at libero in, sagittis posuere tortor. Sed eu congue arcu. Sed augue arcu, tristique in rhoncus ac, laoreet in ex. Vestibulum tristique ullamcorper scelerisque. Suspendisse potenti. Donec eleifend odio eget neque porta accumsan. Pellentesque nec enim risus. Proin vulputate ex tincidunt imperdiet feugiat. Fusce egestas felis dui, mollis fermentum risus consequat scelerisque. Nunc sit amet pretium lectus. Nullam gravida maximus porta. Donec lobortis tincidunt pulvinar. Suspendisse laoreet justo hendrerit, fringilla orci sed, molestie urna. Aenean vel mi a lorem facilisis efficitur. Vestibulum finibus sem et ante volutpat, ut tempus nulla fermentum. Integer justo diam, gravida non odio quis, bibendum blandit risus. Integer ut ipsum dui. Mauris imperdiet eget nisi vitae gravida. Maecenas viverra sem a orci cursus commodo. Suspendisse scelerisque placerat sapien ac fermentum. Nam facilisis interdum sapien at bibendum. Ut malesuada lacus sem. Aliquam neque felis, elementum a tortor in, mattis suscipit elit. Suspendisse molestie, nibh nec viverra lobortis, nunc velit vehicula neque, porttitor lacinia nunc purus vel sem. Phasellus rutrum eget orci in gravida. Nulla placerat in leo a vulputate. Proin vitae ante a est elementum rhoncus. Vivamus convallis arcu elit, sit amet accumsan enim pellentesque non. In blandit justo tellus. Fusce eget arcu laoreet urna tempus laoreet. Praesent ac sagittis ante. Vivamus eu leo at nisi eleifend feugiat a at sem. Nullam fermentum arcu eu lorem maximus, at mattis nisi ultricies. Phasellus faucibus massa nisl, non tempor nulla gravida in. Morbi rutrum ligula at sem scelerisque sollicitudin. Vestibulum egestas ultrices hendrerit. Aliquam consectetur purus justo, nec rutrum risus tristique quis. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Quisque ac porta dui. Maecenas efficitur nec magna accumsan maximus. Pellentesque viverra pharetra tempor. Praesent massa purus, porttitor vel scelerisque et, tincidunt a nunc. Donec vitae bibendum tellus. Mauris ante massa, maximus a tellus ullamcorper, mollis iaculis est. Sed interdum justo at diam luctus finibus. Morbi ornare consequat enim, ut lacinia mi ullamcorper at. Vestibulum volutpat tellus in neque ultrices aliquet. Pellentesque sollicitudin viverra pulvinar. Donec faucibus a felis at pulvinar. Sed pretium sem vitae erat auctor, ac sodales ligula laoreet. Praesent lacinia tortor vel vestibulum mollis. Donec bibendum id lorem porttitor faucibus. Maecenas a dui condimentum, sodales eros quis, finibus dolor. Vivamus non neque eros. Mauris laoreet euismod fringilla. Phasellus iaculis aliquet ipsum nec tempus. Vestibulum maximus nunc sed orci porttitor, at tincidunt erat accumsan. Ut tempus nisl in lacinia aliquet. Nulla nec justo non ipsum faucibus volutpat. Donec sit amet sem a risus pharetra venenatis. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. Pellentesque nulla justo, varius eu volutpat non, rhoncus consequat tortor. Nulla ultricies pretium luctus. Aliquam vitae dui ac nunc dictum sagittis vel vitae lectus. In ultricies ultrices tortor eu tincidunt. Mauris erat velit, semper eu rutrum id, tristique et turpis. Nunc elementum gravida lectus, eu dapibus purus. Pellentesque orci lacus, pharetra vel est quis, tincidunt eleifend est. Phasellus commodo est ex, eu dictum odio scelerisque a. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; In posuere tellus purus, nec rutrum turpis rhoncus at. Morbi tempus nulla non dui consequat, nec accumsan neque scelerisque. Curabitur ante metus, varius et feugiat eu, rhoncus et leo. Duis vulputate elit eget dolor malesuada suscipit. Curabitur tristique finibus mollis. Nunc sagittis mattis volutpat. Morbi auctor nec tellus et dignissim. In iaculis magna eu justo finibus, vitae facilisis tellus rhoncus. Sed in euismod lacus. Proin erat eros, auctor quis libero a, pellentesque fringilla ante. Donec vestibulum odio consectetur dui malesuada porta. Mauris convallis auctor hendrerit. Nullam ut augue ut mi egestas volutpat. Praesent finibus velit sit amet volutpat congue. Integer faucibus ultrices erat, non cursus nunc commodo ac. Cras finibus enim lacus, et vulputate ipsum euismod eu. Donec in pellentesque nulla. Etiam semper quam in felis elementum, at bibendum elit condimentum. Morbi finibus lacus quis neque tincidunt porta. Pellentesque rutrum fringilla velit, convallis posuere dui aliquet vel. Vestibulum tristique ante et lacinia malesuada. Vivamus ac nulla eu purus mattis condimentum. Ut id nisi eu lectus efficitur vehicula. Duis ut turpis sit amet enim elementum dapibus ac ut mauris. Proin vestibulum feugiat consequat. Nunc vestibulum interdum magna, nec ultrices odio laoreet eget. Morbi nisi orci, pharetra nec eleifend vitae, tincidunt ut quam. Nunc lacinia vestibulum ultrices. Curabitur a sodales diam. Ut sed elit id urna molestie bibendum. Sed interdum, elit et pharetra semper, turpis nisi malesuada ex, vitae sollicitudin massa dui sit amet dui. Donec dapibus ornare diam ac commodo. In ornare at dolor vel consequat. Aenean eu justo ultricies, vestibulum nisi non, fermentum dolor. Etiam mauris lacus, euismod in dolor ut, accumsan varius magna. Orci varius natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Pellentesque ac aliquam mi. Vivamus vulputate faucibus ipsum, eget bibendum justo ultrices ac. Maecenas id rhoncus lectus, nec mattis nibh. Proin dui lacus, lobortis nec enim nec, imperdiet commodo nisl. Orci varius sed." + "__default__": "GET http://temba.io/", + "headers": {}, + "json": { + "big": "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Proin sed nunc vehicula, commodo ipsum et, consectetur massa. Suspendisse potenti. Ut feugiat volutpat purus vel viverra. Fusce commodo, massa eget malesuada aliquam, dolor lectus porta tortor, ultrices lobortis lacus tellus non velit. Interdum et malesuada fames ac ante ipsum primis in faucibus. Phasellus in viverra metus. Ut lobortis metus elit, elementum posuere ex consequat non. Donec elementum rutrum orci non dictum. Nam ut ultricies nisi, a viverra nisl. Sed et nibh vitae metus bibendum lobortis sed in ex. Nunc porta elit eget ipsum bibendum gravida. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. Suspendisse potenti. Etiam quis ligula quis lacus ultricies fringilla. Integer nec pharetra nunc. Curabitur pharetra, dolor fringilla ultricies ornare, purus nisl ultrices augue, nec pulvinar ipsum orci consequat quam. Proin auctor justo non eleifend facilisis. Praesent eget justo elit. Ut nec augue purus. Cras nulla risus, bibendum ac est ut, pharetra bibendum elit. Integer interdum, lorem nec pellentesque ornare, nulla mauris pretium arcu, non lacinia risus odio vel nibh. Proin bibendum nulla vel nulla lacinia faucibus. Quisque accumsan sapien malesuada, pulvinar elit non, sollicitudin enim. Aliquam iaculis, massa non tempus hendrerit, ante nunc semper orci, et pretium libero tellus at urna. Nullam maximus sem condimentum, vestibulum eros sit amet, elementum odio. Mauris nisl augue, tristique id eleifend at, elementum vitae elit. Aenean ut iaculis felis. Curabitur id mollis sem. Phasellus quis bibendum est, id hendrerit nulla. Sed consequat metus ex, vitae pharetra lorem commodo id. Etiam eu nisl a ligula laoreet semper. Maecenas non ornare urna. Vestibulum posuere sapien quis dolor scelerisque euismod. Fusce eget neque ac nisl auctor commodo id vitae massa. Suspendisse tincidunt leo at erat dignissim imperdiet. Fusce pulvinar consectetur vehicula. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Praesent ut aliquet dui. Fusce at sollicitudin urna. Vivamus sed neque elit. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Vivamus sed urna accumsan nulla euismod pulvinar eu ut tortor. In egestas id lectus at ultrices. Nam a cursus lectus, a laoreet lectus. Vivamus pharetra sapien vel diam hendrerit, vitae consequat quam iaculis. Orci varius natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Etiam ac felis at velit venenatis molestie. Nunc risus lacus, fringilla eu libero sit amet, dictum interdum tortor. In nec viverra est. Cras imperdiet urna eget ullamcorper consectetur. In faucibus finibus quam. Nulla dictum dolor tristique, rutrum nisi ut, facilisis sem. Aliquam erat volutpat. Donec scelerisque nec justo sed lobortis. Donec posuere, mi vitae molestie finibus, felis lectus facilisis ligula, et dignissim magna diam at turpis. Integer ac ante pulvinar ipsum malesuada convallis eu a orci. Nunc nec accumsan felis. Sed quam purus, bibendum eget mattis non, sodales ut felis. Ut a erat et orci elementum vulputate in eget purus. Suspendisse et posuere lectus. Fusce tempor enim arcu, eu ultricies erat condimentum pulvinar. Nam luctus consequat lectus, eget elementum nunc varius sed. Interdum et malesuada fames ac ante ipsum primis in faucibus. Sed sit amet posuere velit, eu vehicula erat. Nulla faucibus at dolor in tincidunt. Nam quis neque ut dui congue laoreet. Etiam ex metus, laoreet lobortis magna non, vehicula dapibus tellus. Integer sit amet orci aliquam, venenatis risus sit amet, cursus metus. Aenean sit amet lectus id neque eleifend pellentesque sed vitae sapien. Vivamus lacus risus, volutpat et mauris quis, porttitor tempus nisl. In tincidunt, elit semper varius posuere, arcu nulla suscipit urna, sit amet posuere ipsum leo a est. Cras ipsum sapien, varius sed mauris a, aliquam consectetur nisi. Integer et ante sit amet tellus dictum sagittis et in lorem. Nulla vel diam elementum, maximus libero dictum, semper orci. Phasellus in facilisis tortor, in vulputate purus. Vivamus rhoncus sem tempus, pharetra turpis vitae, laoreet sem. Pellentesque egestas tellus velit. In laoreet tempor erat. Ut dui erat, pulvinar eu libero imperdiet, fringilla tincidunt est. Ut vitae lectus non velit mollis euismod. Fusce risus neque, sodales at libero in, sagittis posuere tortor. Sed eu congue arcu. Sed augue arcu, tristique in rhoncus ac, laoreet in ex. Vestibulum tristique ullamcorper scelerisque. Suspendisse potenti. Donec eleifend odio eget neque porta accumsan. Pellentesque nec enim risus. Proin vulputate ex tincidunt imperdiet feugiat. Fusce egestas felis dui, mollis fermentum risus consequat scelerisque. Nunc sit amet pretium lectus. Nullam gravida maximus porta. Donec lobortis tincidunt pulvinar. Suspendisse laoreet justo hendrerit, fringilla orci sed, molestie urna. Aenean vel mi a lorem facilisis efficitur. Vestibulum finibus sem et ante volutpat, ut tempus nulla fermentum. Integer justo diam, gravida non odio quis, bibendum blandit risus. Integer ut ipsum dui. Mauris imperdiet eget nisi vitae gravida. Maecenas viverra sem a orci cursus commodo. Suspendisse scelerisque placerat sapien ac fermentum. Nam facilisis interdum sapien at bibendum. Ut malesuada lacus sem. Aliquam neque felis, elementum a tortor in, mattis suscipit elit. Suspendisse molestie, nibh nec viverra lobortis, nunc velit vehicula neque, porttitor lacinia nunc purus vel sem. Phasellus rutrum eget orci in gravida. Nulla placerat in leo a vulputate. Proin vitae ante a est elementum rhoncus. Vivamus convallis arcu elit, sit amet accumsan enim pellentesque non. In blandit justo tellus. Fusce eget arcu laoreet urna tempus laoreet. Praesent ac sagittis ante. Vivamus eu leo at nisi eleifend feugiat a at sem. Nullam fermentum arcu eu lorem maximus, at mattis nisi ultricies. Phasellus faucibus massa nisl, non tempor nulla gravida in. Morbi rutrum ligula at sem scelerisque sollicitudin. Vestibulum egestas ultrices hendrerit. Aliquam consectetur purus justo, nec rutrum risus tristique quis. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Quisque ac porta dui. Maecenas efficitur nec magna accumsan maximus. Pellentesque viverra pharetra tempor. Praesent massa purus, porttitor vel scelerisque et, tincidunt a nunc. Donec vitae bibendum tellus. Mauris ante massa, maximus a tellus ullamcorper, mollis iaculis est. Sed interdum justo at diam luctus finibus. Morbi ornare consequat enim, ut lacinia mi ullamcorper at. Vestibulum volutpat tellus in neque ultrices aliquet. Pellentesque sollicitudin viverra pulvinar. Donec faucibus a felis at pulvinar. Sed pretium sem vitae erat auctor, ac sodales ligula laoreet. Praesent lacinia tortor vel vestibulum mollis. Donec bibendum id lorem porttitor faucibus. Maecenas a dui condimentum, sodales eros quis, finibus dolor. Vivamus non neque eros. Mauris laoreet euismod fringilla. Phasellus iaculis aliquet ipsum nec tempus. Vestibulum maximus nunc sed orci porttitor, at tincidunt erat accumsan. Ut tempus nisl in lacinia aliquet. Nulla nec justo non ipsum faucibus volutpat. Donec sit amet sem a risus pharetra venenatis. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. Pellentesque nulla justo, varius eu volutpat non, rhoncus consequat tortor. Nulla ultricies pretium luctus. Aliquam vitae dui ac nunc dictum sagittis vel vitae lectus. In ultricies ultrices tortor eu tincidunt. Mauris erat velit, semper eu rutrum id, tristique et turpis. Nunc elementum gravida lectus, eu dapibus purus. Pellentesque orci lacus, pharetra vel est quis, tincidunt eleifend est. Phasellus commodo est ex, eu dictum odio scelerisque a. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; In posuere tellus purus, nec rutrum turpis rhoncus at. Morbi tempus nulla non dui consequat, nec accumsan neque scelerisque. Curabitur ante metus, varius et feugiat eu, rhoncus et leo. Duis vulputate elit eget dolor malesuada suscipit. Curabitur tristique finibus mollis. Nunc sagittis mattis volutpat. Morbi auctor nec tellus et dignissim. In iaculis magna eu justo finibus, vitae facilisis tellus rhoncus. Sed in euismod lacus. Proin erat eros, auctor quis libero a, pellentesque fringilla ante. Donec vestibulum odio consectetur dui malesuada porta. Mauris convallis auctor hendrerit. Nullam ut augue ut mi egestas volutpat. Praesent finibus velit sit amet volutpat congue. Integer faucibus ultrices erat, non cursus nunc commodo ac. Cras finibus enim lacus, et vulputate ipsum euismod eu. Donec in pellentesque nulla. Etiam semper quam in felis elementum, at bibendum elit condimentum. Morbi finibus lacus quis neque tincidunt porta. Pellentesque rutrum fringilla velit, convallis posuere dui aliquet vel. Vestibulum tristique ante et lacinia malesuada. Vivamus ac nulla eu purus mattis condimentum. Ut id nisi eu lectus efficitur vehicula. Duis ut turpis sit amet enim elementum dapibus ac ut mauris. Proin vestibulum feugiat consequat. Nunc vestibulum interdum magna, nec ultrices odio laoreet eget. Morbi nisi orci, pharetra nec eleifend vitae, tincidunt ut quam. Nunc lacinia vestibulum ultrices. Curabitur a sodales diam. Ut sed elit id urna molestie bibendum. Sed interdum, elit et pharetra semper, turpis nisi malesuada ex, vitae sollicitudin massa dui sit amet dui. Donec dapibus ornare diam ac commodo. In ornare at dolor vel consequat. Aenean eu justo ultricies, vestibulum nisi non, fermentum dolor. Etiam mauris lacus, euismod in dolor ut, accumsan varius magna. Orci varius natoque penatibus et magnis dis parturient montes, nascetur ridiculus mus. Interdum et malesuada fames ac ante ipsum primis in faucibus. Pellentesque ac aliquam mi. Vivamus vulputate faucibus ipsum, eget bibendum justo ultrices ac. Maecenas id rhoncus lectus, nec mattis nibh. Proin dui lacus, lobortis nec enim nec, imperdiet commodo nisl. Orci varius sed." + }, + "status": 200 }, "templates": [ "http://temba.io/" @@ -494,9 +531,14 @@ } ], "webhook": { - "errors": [ - "bad_request" - ] + "__default__": "POST http://temba.io/", + "headers": {}, + "json": { + "errors": [ + "bad_request" + ] + }, + "status": 400 }, "templates": [ "http://temba.io/", @@ -562,7 +604,12 @@ "input": "POST http://temba.io/" } ], - "webhook": {}, + "webhook": { + "__default__": "POST http://temba.io/", + "headers": {}, + "json": null, + "status": 0 + }, "templates": [ "http://temba.io/", "Hi there!" @@ -635,7 +682,12 @@ "input": "GET http://temba.io/" } ], - "webhook": {}, + "webhook": { + "__default__": "GET http://temba.io/", + "headers": {}, + "json": null, + "status": 200 + }, "templates": [ "http://temba.io/" ], diff --git a/flows/definition/flow.go b/flows/definition/flow.go index 44eafa948..fb18df8c5 100644 --- a/flows/definition/flow.go +++ b/flows/definition/flow.go @@ -19,7 +19,7 @@ import ( ) // CurrentSpecVersion is the flow spec version supported by this library -var CurrentSpecVersion = semver.MustParse("13.2.0") +var CurrentSpecVersion = semver.MustParse("13.3.0") // IsVersionSupported checks the given version is supported func IsVersionSupported(v *semver.Version) bool { diff --git a/flows/definition/migrations/13_x.go b/flows/definition/migrations/13_x.go index 0749bb9e8..d2e69b1a0 100644 --- a/flows/definition/migrations/13_x.go +++ b/flows/definition/migrations/13_x.go @@ -3,13 +3,29 @@ package migrations import ( "github.com/Masterminds/semver" "github.com/nyaruka/gocommon/uuids" + "github.com/nyaruka/goflow/excellent/tools" ) func init() { + registerMigration(semver.MustParse("13.3.0"), Migrate13_3) registerMigration(semver.MustParse("13.2.0"), Migrate13_2) registerMigration(semver.MustParse("13.1.0"), Migrate13_1) } +// Migrate13_3 refactors template expressions that reference @webhook to use @webhook.json +// +// @version 13_3 "13.3" +func Migrate13_3(f Flow, cfg *Config) (Flow, error) { + RewriteTemplates(f, GetTemplateCatalog(semver.MustParse("13.2.0")), func(s string) string { + // some optimizations here... + // 1. we can parse templates as if @(...) and @webhook are only valid top-levels + // 2. we can treat adding .json as a lookup to webhook as a simple renaming of webhook to webhook.json + refactored, _ := tools.RefactorTemplate(s, []string{"webhook"}, tools.ContextRefRename("webhook", "webhook.json")) + return refactored + }) + return f, nil +} + // Migrate13_2 replaces `base` as a flow language with `und` which indicates text with undetermined language // in the ISO-639-3 standard. // diff --git a/flows/definition/migrations/specdata/templates.json b/flows/definition/migrations/specdata/templates.json index 8178c62b4..b825af354 100644 --- a/flows/definition/migrations/specdata/templates.json +++ b/flows/definition/migrations/specdata/templates.json @@ -1,4 +1,94 @@ { + "13.3.0": { + "actions": { + "add_contact_groups": [ + ".groups[*].name_match" + ], + "add_contact_urn": [ + ".path" + ], + "add_input_labels": [ + ".labels[*].name_match" + ], + "call_classifier": [ + ".input" + ], + "call_resthook": [], + "call_webhook": [ + ".body", + ".headers.*", + ".url" + ], + "enter_flow": [], + "open_ticket": [ + ".assignee.email_match", + ".body" + ], + "play_audio": [ + ".audio_url" + ], + "remove_contact_groups": [ + ".groups[*].name_match" + ], + "request_optin": [], + "say_msg": [ + ".text" + ], + "send_broadcast": [ + ".attachments[*]", + ".contact_query", + ".groups[*].name_match", + ".legacy_vars[*]", + ".quick_replies[*]", + ".text" + ], + "send_email": [ + ".addresses[*]", + ".body", + ".subject" + ], + "send_msg": [ + ".attachments[*]", + ".quick_replies[*]", + ".templating.components[*].params[*]", + ".templating.variables[*]", + ".text" + ], + "set_contact_channel": [], + "set_contact_field": [ + ".value" + ], + "set_contact_language": [ + ".language" + ], + "set_contact_name": [ + ".name" + ], + "set_contact_status": [], + "set_contact_timezone": [ + ".timezone" + ], + "set_run_result": [ + ".value" + ], + "start_session": [ + ".contact_query", + ".groups[*].name_match", + ".legacy_vars[*]" + ], + "transfer_airtime": [] + }, + "routers": { + "random": [ + ".operand", + ".cases[*].arguments[*]" + ], + "switch": [ + ".operand", + ".cases[*].arguments[*]" + ] + } + }, "13.2.0": { "actions": { "add_contact_groups": [ diff --git a/flows/definition/migrations/testdata/migrations/13.3.0.json b/flows/definition/migrations/testdata/migrations/13.3.0.json new file mode 100644 index 000000000..ede597ec5 --- /dev/null +++ b/flows/definition/migrations/testdata/migrations/13.3.0.json @@ -0,0 +1,232 @@ +[ + { + "description": "flow with localization", + "original": { + "uuid": "76f0a02f-3b75-4b86-9064-e9195e1b3a02", + "name": "Testing @webhook", + "spec_version": "13.2.0", + "language": "eng", + "type": "messaging", + "localization": { + "spa": { + "8eebd020-1af5-431c-b943-aa670fc74da9": { + "text": [ + "@webhook" + ] + } + } + }, + "nodes": [ + { + "uuid": "365293c7-633c-45bd-96b7-0b059766588d", + "actions": [ + { + "uuid": "8eebd020-1af5-431c-b943-aa670fc74da9", + "type": "send_msg", + "text": "@webhook @(webhook[\"foo\"]) @contact.fields.webhook @results.webhook.extra flows@@nyaruka.com" + } + ], + "router": { + "type": "switch", + "wait": { + "type": "msg" + }, + "default_category_uuid": "5ce6c69a-fdfe-4594-ab71-26be534d31c3", + "result_name": "Question", + "operand": "@webhook.foo", + "cases": [ + { + "uuid": "e27c3bce-1095-4d08-9164-dc4530a0688a", + "type": "has_any_word", + "arguments": [ + "@webhook.bar" + ], + "category_uuid": "2ab9b033-77a8-4e56-a558-b568c00c9492" + }, + { + "uuid": "4a6c3b0b-0658-4a93-ae37-bee68f6a6a87", + "type": "has_any_word", + "arguments": [ + "no", + "nope" + ], + "category_uuid": "c7bca181-0cb3-4ec6-8555-f7e5644238ad" + } + ], + "categories": [ + { + "uuid": "2ab9b033-77a8-4e56-a558-b568c00c9492", + "name": "Yes", + "exit_uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + }, + { + "uuid": "c7bca181-0cb3-4ec6-8555-f7e5644238ad", + "name": "No", + "exit_uuid": "9ad71fc4-c2f8-4aab-a193-7bafad172ca0" + }, + { + "uuid": "5ce6c69a-fdfe-4594-ab71-26be534d31c3", + "name": "Other", + "exit_uuid": "e80bc037-3b57-45b5-9f19-a8346a475578" + } + ] + }, + "exits": [ + { + "uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + }, + { + "uuid": "9ad71fc4-c2f8-4aab-a193-7bafad172ca0" + }, + { + "uuid": "e80bc037-3b57-45b5-9f19-a8346a475578" + } + ] + } + ] + }, + "migrated": { + "uuid": "76f0a02f-3b75-4b86-9064-e9195e1b3a02", + "name": "Testing @webhook", + "spec_version": "13.3.0", + "language": "eng", + "type": "messaging", + "localization": { + "spa": { + "8eebd020-1af5-431c-b943-aa670fc74da9": { + "text": [ + "@webhook.json" + ] + } + } + }, + "nodes": [ + { + "uuid": "365293c7-633c-45bd-96b7-0b059766588d", + "actions": [ + { + "uuid": "8eebd020-1af5-431c-b943-aa670fc74da9", + "type": "send_msg", + "text": "@webhook.json @(webhook.json[\"foo\"]) @contact.fields.webhook @results.webhook.extra flows@@nyaruka.com" + } + ], + "router": { + "type": "switch", + "wait": { + "type": "msg" + }, + "default_category_uuid": "5ce6c69a-fdfe-4594-ab71-26be534d31c3", + "result_name": "Question", + "operand": "@webhook.json.foo", + "cases": [ + { + "uuid": "e27c3bce-1095-4d08-9164-dc4530a0688a", + "type": "has_any_word", + "arguments": [ + "@webhook.json.bar" + ], + "category_uuid": "2ab9b033-77a8-4e56-a558-b568c00c9492" + }, + { + "uuid": "4a6c3b0b-0658-4a93-ae37-bee68f6a6a87", + "type": "has_any_word", + "arguments": [ + "no", + "nope" + ], + "category_uuid": "c7bca181-0cb3-4ec6-8555-f7e5644238ad" + } + ], + "categories": [ + { + "uuid": "2ab9b033-77a8-4e56-a558-b568c00c9492", + "name": "Yes", + "exit_uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + }, + { + "uuid": "c7bca181-0cb3-4ec6-8555-f7e5644238ad", + "name": "No", + "exit_uuid": "9ad71fc4-c2f8-4aab-a193-7bafad172ca0" + }, + { + "uuid": "5ce6c69a-fdfe-4594-ab71-26be534d31c3", + "name": "Other", + "exit_uuid": "e80bc037-3b57-45b5-9f19-a8346a475578" + } + ] + }, + "exits": [ + { + "uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + }, + { + "uuid": "9ad71fc4-c2f8-4aab-a193-7bafad172ca0" + }, + { + "uuid": "e80bc037-3b57-45b5-9f19-a8346a475578" + } + ] + } + ] + } + }, + { + "description": "flow without localization", + "original": { + "uuid": "76f0a02f-3b75-4b86-9064-e9195e1b3a02", + "name": "Testing @webhook", + "spec_version": "13.2.0", + "language": "eng", + "type": "messaging", + "nodes": [ + { + "uuid": "365293c7-633c-45bd-96b7-0b059766588d", + "actions": [ + { + "uuid": "8eebd020-1af5-431c-b943-aa670fc74da9", + "type": "set_contact_field", + "field": { + "key": "gender", + "name": "Gender" + }, + "value": "@webhook.result" + } + ], + "exits": [ + { + "uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + } + ] + } + ] + }, + "migrated": { + "uuid": "76f0a02f-3b75-4b86-9064-e9195e1b3a02", + "name": "Testing @webhook", + "spec_version": "13.3.0", + "language": "eng", + "type": "messaging", + "nodes": [ + { + "uuid": "365293c7-633c-45bd-96b7-0b059766588d", + "actions": [ + { + "uuid": "8eebd020-1af5-431c-b943-aa670fc74da9", + "type": "set_contact_field", + "field": { + "key": "gender", + "name": "Gender" + }, + "value": "@webhook.json.result" + } + ], + "exits": [ + { + "uuid": "3bd19c40-1114-4b83-b12e-f0c38054ba3f" + } + ] + } + ] + } + } +] \ No newline at end of file diff --git a/flows/definition/testdata/TestChangeLanguage_change_language_to_ara.snap b/flows/definition/testdata/TestChangeLanguage_change_language_to_ara.snap index c0f191b3d..b86e10cef 100644 --- a/flows/definition/testdata/TestChangeLanguage_change_language_to_ara.snap +++ b/flows/definition/testdata/TestChangeLanguage_change_language_to_ara.snap @@ -1,7 +1,7 @@ { "uuid": "19cad1f2-9110-4271-98d4-1b968bf19410", "name": "Change Language", - "spec_version": "13.2.0", + "spec_version": "13.3.0", "language": "ara", "type": "messaging", "revision": 16, diff --git a/flows/definition/testdata/TestChangeLanguage_change_language_to_kin.snap b/flows/definition/testdata/TestChangeLanguage_change_language_to_kin.snap index 9f909d066..465bf1555 100644 --- a/flows/definition/testdata/TestChangeLanguage_change_language_to_kin.snap +++ b/flows/definition/testdata/TestChangeLanguage_change_language_to_kin.snap @@ -1,7 +1,7 @@ { "uuid": "19cad1f2-9110-4271-98d4-1b968bf19410", "name": "Change Language", - "spec_version": "13.2.0", + "spec_version": "13.3.0", "language": "kin", "type": "messaging", "revision": 16, diff --git a/flows/definition/testdata/TestChangeLanguage_change_language_to_spa.snap b/flows/definition/testdata/TestChangeLanguage_change_language_to_spa.snap index d22e31f01..b89a182e9 100644 --- a/flows/definition/testdata/TestChangeLanguage_change_language_to_spa.snap +++ b/flows/definition/testdata/TestChangeLanguage_change_language_to_spa.snap @@ -1,7 +1,7 @@ { "uuid": "19cad1f2-9110-4271-98d4-1b968bf19410", "name": "Change Language", - "spec_version": "13.2.0", + "spec_version": "13.3.0", "language": "spa", "type": "messaging", "revision": 16, diff --git a/flows/definition/testdata/change_language.json b/flows/definition/testdata/change_language.json index 8844fd3cf..31c81e06f 100644 --- a/flows/definition/testdata/change_language.json +++ b/flows/definition/testdata/change_language.json @@ -3,7 +3,7 @@ { "uuid": "19cad1f2-9110-4271-98d4-1b968bf19410", "name": "Change Language", - "spec_version": "13.2.0", + "spec_version": "13.3.0", "language": "eng", "type": "messaging", "revision": 16, diff --git a/flows/engine/testdata/templates.json b/flows/engine/testdata/templates.json index 546ab2013..5267cb885 100644 --- a/flows/engine/testdata/templates.json +++ b/flows/engine/testdata/templates.json @@ -313,19 +313,27 @@ "output": "completed" }, { - "template": "@webhook", + "template": "@webhook.headers", + "output": "{Content-Length: 43, Content-Type: text/plain; charset=utf-8, Date: Wed, 11 Apr 2018 18:24:30 GMT}" + }, + { + "template": "@(webhook.headers[\"Content-Type\"])", + "output": "text/plain; charset=utf-8" + }, + { + "template": "@webhook.json", "output": "{results: [{state: WA}, {state: IN}]}" }, { - "template": "@webhook.results", + "template": "@webhook.json.results", "output": "[{state: WA}, {state: IN}]" }, { - "template": "@(webhook.results[1])", + "template": "@(webhook.json.results[1])", "output": "{state: IN}" }, { - "template": "@(webhook.results[1].state)", + "template": "@(webhook.json.results[1].state)", "output": "IN" }, { diff --git a/flows/interfaces.go b/flows/interfaces.go index 8430ff5be..827afc19a 100644 --- a/flows/interfaces.go +++ b/flows/interfaces.go @@ -412,8 +412,8 @@ type Run interface { Session() Session SaveResult(*Result) SetStatus(RunStatus) - Webhook() types.XValue - SetWebhook(types.XValue) + Webhook() *WebhookCall + SetWebhook(*WebhookCall) CreateStep(Node) Step Path() []Step diff --git a/flows/runs/legacy.go b/flows/runs/legacy.go index 8836771c5..b050c7afe 100644 --- a/flows/runs/legacy.go +++ b/flows/runs/legacy.go @@ -1,11 +1,13 @@ package runs import ( + "net/http" "regexp" "sort" "strconv" "strings" + "github.com/nyaruka/gocommon/httpx" "github.com/nyaruka/goflow/envs" "github.com/nyaruka/goflow/excellent/types" "github.com/nyaruka/goflow/flows" @@ -100,7 +102,7 @@ func arrayToObject(array *types.XArray) *types.XObject { } // finds the last webhook response that was saved as extra on a result -func lastWebhookSavedAsExtra(r *run) types.XValue { +func lastWebhookSavedAsExtra(r *run) *flows.WebhookCall { for i := len(r.events) - 1; i >= 0; i-- { switch typed := r.events[i].(type) { case *events.WebhookCalledEvent: @@ -110,11 +112,11 @@ func lastWebhookSavedAsExtra(r *run) types.XValue { if resultEvent != nil { asResultEvent := resultEvent.(*events.RunResultChangedEvent) if asResultEvent.Extra != nil { - value := types.JSONToXValue([]byte(asResultEvent.Extra)) - if value != nil { - value.SetDeprecated("webhook recreated from extra") + return &flows.WebhookCall{ + Trace: &httpx.Trace{Response: &http.Response{}}, + ResponseJSON: asResultEvent.Extra, + Recreated: true, } - return value } } default: diff --git a/flows/runs/run.go b/flows/runs/run.go index 36e6045b9..132af562a 100644 --- a/flows/runs/run.go +++ b/flows/runs/run.go @@ -36,7 +36,7 @@ type run struct { modifiedOn time.Time exitedOn *time.Time - webhook types.XValue + webhook *flows.WebhookCall legacyExtra *legacyExtra } @@ -56,7 +56,7 @@ func NewRun(session flows.Session, flow flows.Flow, parent flows.Run) flows.Run modifiedOn: now, } - r.webhook = types.XObjectEmpty + r.webhook = nil r.legacyExtra = newLegacyExtra(r) return r @@ -94,11 +94,9 @@ func (r *run) SetStatus(status flows.RunStatus) { r.modifiedOn = dates.Now() } -func (r *run) Webhook() types.XValue { - return r.webhook -} -func (r *run) SetWebhook(value types.XValue) { - r.webhook = value +func (r *run) Webhook() *flows.WebhookCall { return r.webhook } +func (r *run) SetWebhook(call *flows.WebhookCall) { + r.webhook = call } // ParentInSession returns the parent of the run within the same session if one exists @@ -247,7 +245,7 @@ func (r *run) RootContext(env envs.Environment) map[string]types.XValue { "resume": flows.Context(env, r.Session().CurrentResume()), "input": flows.Context(env, r.Session().Input()), "globals": flows.Context(env, r.Session().Assets().Globals()), - "webhook": r.webhook, + "webhook": flows.Context(env, r.webhook), "node": node, "legacy_extra": r.legacyExtra.ToXValue(env), } diff --git a/flows/services.go b/flows/services.go index 3b20c3c22..02321fa40 100644 --- a/flows/services.go +++ b/flows/services.go @@ -1,6 +1,8 @@ package flows import ( + "encoding/json" + "fmt" "net/http" "time" @@ -9,6 +11,7 @@ import ( "github.com/nyaruka/gocommon/urns" "github.com/nyaruka/gocommon/uuids" "github.com/nyaruka/goflow/envs" + "github.com/nyaruka/goflow/excellent/types" "github.com/shopspring/decimal" ) @@ -48,6 +51,62 @@ type WebhookCall struct { *httpx.Trace ResponseJSON []byte ResponseCleaned bool // whether response had to be cleaned to make it valid JSON + Recreated bool // whether the call was recreated from a result +} + +// Context returns the properties available in expressions +// +// __default__:text -> the method and URL +// status:number -> the response status code +// headers:any -> the response headers +// json:any -> the response body if valid JSON +// +// @context webhook +func (w *WebhookCall) Context(env envs.Environment) map[string]types.XValue { + status := types.NewXNumberFromInt(0) + headers := types.XObjectEmpty + var json types.XValue + + // TODO remove when users stop relying on this + if w.Recreated { + json = types.JSONToXValue(w.ResponseJSON) + if types.IsXError(json) { + json = nil + } + if json != nil { + json.SetDeprecated("webhook recreated from extra") + } + + return map[string]types.XValue{"json": json} + } + + if w.Response != nil { + status = types.NewXNumberFromInt(w.Response.StatusCode) + + headers = types.NewXLazyObject(func() map[string]types.XValue { + values := make(map[string]types.XValue, len(w.Response.Header)) + for k := range w.Response.Header { + values[k] = types.NewXText(w.Response.Header.Get(k)) + } + return values + }) + + json = types.JSONToXValue(w.ResponseJSON) + if types.IsXError(json) { + json = nil + } + } + + return map[string]types.XValue{ + "__default__": types.NewXText(fmt.Sprintf("%s %s", w.Request.Method, w.Request.URL.String())), + "status": status, + "headers": headers, + "json": json, + } +} + +func (w *WebhookCall) MarshalJSON() ([]byte, error) { + return json.Marshal(w.Context(nil)) } // WebhookService provides webhook functionality to the engine diff --git a/flows/services_test.go b/flows/services_test.go index ce0680e68..b420eae9e 100644 --- a/flows/services_test.go +++ b/flows/services_test.go @@ -6,7 +6,10 @@ import ( "testing" "github.com/nyaruka/gocommon/httpx" + "github.com/nyaruka/goflow/envs" + "github.com/nyaruka/goflow/excellent/types" "github.com/nyaruka/goflow/flows" + "github.com/nyaruka/goflow/test" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" @@ -59,3 +62,59 @@ func TestHTTPLogs(t *testing.T) { log4 := flows.NewHTTPLog(trace4, flows.HTTPStatusFromCode, nil) assert.Equal(t, "http://temba.io/?x="+strings.Repeat("x", 2026)+"...", log4.URL) // trimmed } + +func TestWebhookCall(t *testing.T) { + defer httpx.SetRequestor(httpx.DefaultRequestor) + + httpx.SetRequestor(httpx.NewMockRequestor(map[string][]*httpx.MockResponse{ + "http://temba.io/": { + httpx.NewMockResponse(200, map[string]string{"Content-Type": "application/json"}, []byte(`{"foo":123}`)), + httpx.NewMockResponse(400, nil, []byte("is error")), + httpx.MockConnectionError, + httpx.MockConnectionError, + httpx.MockConnectionError, + }, + })) + + eng := test.NewEngine() + env := envs.NewBuilder().Build() + svc, err := eng.Services().Webhook(nil) + require.NoError(t, err) + + request := func(method string) *flows.WebhookCall { + req1, err := httpx.NewRequest(method, "http://temba.io/", nil, nil) + require.NoError(t, err) + + call, err := svc.Call(req1) + require.NoError(t, err) + require.NotNil(t, call) + return call + } + + call1 := request("GET") + + test.AssertXEqual(t, types.NewXObject(map[string]types.XValue{ + "__default__": types.NewXText("GET http://temba.io/"), + "status": types.NewXNumberFromInt(200), + "headers": types.NewXObject(map[string]types.XValue{"Content-Type": types.NewXText("application/json")}), + "json": types.NewXObject(map[string]types.XValue{"foo": types.NewXNumberFromInt(123)}), + }), flows.Context(env, call1)) + + call2 := request("POST") + + test.AssertXEqual(t, types.NewXObject(map[string]types.XValue{ + "__default__": types.NewXText("POST http://temba.io/"), + "status": types.NewXNumberFromInt(400), + "headers": types.XObjectEmpty, + "json": nil, + }), flows.Context(env, call2)) + + call3 := request("GET") + + test.AssertXEqual(t, types.NewXObject(map[string]types.XValue{ + "__default__": types.NewXText("GET http://temba.io/"), + "status": types.NewXNumberFromInt(0), + "headers": types.XObjectEmpty, + "json": nil, + }), flows.Context(env, call3)) +} diff --git a/locale/cs/flows.po b/locale/cs/flows.po index 2da1b7747..1af16cbea 100644 --- a/locale/cs/flows.po +++ b/locale/cs/flows.po @@ -6,7 +6,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Last-Translator: trendspotter , 2021\n" "Language-Team: Czech (https://www.transifex.com/rapidpro/teams/226/cs/)\n" "Language: cs\n" @@ -817,6 +817,11 @@ msgstr "datum posledního setkání s kontaktem" msgid "the localized category of the result" msgstr "lokalizovaná kategorie výsledku" +#, fuzzy +#| msgid "the name or URN" +msgid "the method and URL" +msgstr "Název URN" + msgid "the name" msgstr "název" @@ -878,6 +883,17 @@ msgstr "preferované URN kontaktu" msgid "the preferred channel of the contact" msgstr "preferovaný kanál kontaktu" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +#, fuzzy +#| msgid "the current status of the run" +msgid "the response status code" +msgstr "aktuální stav běhu" + msgid "the result for {key}" msgstr "výsledek pro {key}" diff --git a/locale/en_US/flows.po b/locale/en_US/flows.po index 92b96dee3..6b971262e 100644 --- a/locale/en_US/flows.po +++ b/locale/en_US/flows.po @@ -3,7 +3,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Language: en_US\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" @@ -688,6 +688,9 @@ msgstr "" msgid "the localized category of the result" msgstr "" +msgid "the method and URL" +msgstr "" + msgid "the name" msgstr "" @@ -745,6 +748,15 @@ msgstr "" msgid "the preferred channel of the contact" msgstr "" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +msgid "the response status code" +msgstr "" + msgid "the result for {key}" msgstr "" diff --git a/locale/es/flows.po b/locale/es/flows.po index 7ddbceb88..1ac28e3d0 100644 --- a/locale/es/flows.po +++ b/locale/es/flows.po @@ -8,7 +8,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Last-Translator: Rowan Seymour , 2022\n" "Language-Team: Spanish (https://www.transifex.com/rapidpro/teams/226/es/)\n" "Language: es\n" @@ -853,6 +853,11 @@ msgstr "la fecha de última vista del contacto" msgid "the localized category of the result" msgstr "la categoría localizada del resultado" +#, fuzzy +#| msgid "the name or URN" +msgid "the method and URL" +msgstr "el nombre o URN" + msgid "the name" msgstr "el nombre" @@ -914,6 +919,17 @@ msgstr "la URN preferida del contacto" msgid "the preferred channel of the contact" msgstr "el canal preferido del contacto" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +#, fuzzy +#| msgid "the current status of the run" +msgid "the response status code" +msgstr "el estado actual de la ejecución" + msgid "the result for {key}" msgstr "el resultado de {key}" diff --git a/locale/fr/flows.po b/locale/fr/flows.po index 3a4e96d91..cd5c09108 100644 --- a/locale/fr/flows.po +++ b/locale/fr/flows.po @@ -3,7 +3,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Language-Team: French (https://www.transifex.com/rapidpro/teams/226/fr/)\n" "Language: fr\n" "MIME-Version: 1.0\n" @@ -690,6 +690,9 @@ msgstr "" msgid "the localized category of the result" msgstr "" +msgid "the method and URL" +msgstr "" + msgid "the name" msgstr "" @@ -747,6 +750,15 @@ msgstr "" msgid "the preferred channel of the contact" msgstr "" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +msgid "the response status code" +msgstr "" + msgid "the result for {key}" msgstr "" diff --git a/locale/mn/flows.po b/locale/mn/flows.po index 2d4438ac9..73d03ca67 100644 --- a/locale/mn/flows.po +++ b/locale/mn/flows.po @@ -3,7 +3,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Language-Team: Mongolian (https://www.transifex.com/rapidpro/teams/226/mn/)\n" "Language: mn\n" "MIME-Version: 1.0\n" @@ -690,6 +690,9 @@ msgstr "" msgid "the localized category of the result" msgstr "" +msgid "the method and URL" +msgstr "" + msgid "the name" msgstr "" @@ -747,6 +750,15 @@ msgstr "" msgid "the preferred channel of the contact" msgstr "" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +msgid "the response status code" +msgstr "" + msgid "the result for {key}" msgstr "" diff --git a/locale/pt_BR/flows.po b/locale/pt_BR/flows.po index 5f0f79151..7aaa7ce57 100644 --- a/locale/pt_BR/flows.po +++ b/locale/pt_BR/flows.po @@ -7,7 +7,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Last-Translator: Ilhasoft , 2021\n" "Language-Team: Portuguese (Brazil) (https://www.transifex.com/rapidpro/teams/226/pt_BR/)\n" "Language: pt_BR\n" @@ -730,6 +730,11 @@ msgstr "a última data vista do contato" msgid "the localized category of the result" msgstr "a categoria localizada do resultado" +#, fuzzy +#| msgid "the name or URN" +msgid "the method and URL" +msgstr "o nome da URN" + msgid "the name" msgstr "o nome" @@ -791,6 +796,17 @@ msgstr "a URN preferida do contato" msgid "the preferred channel of the contact" msgstr "o canal preferido do contato" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +#, fuzzy +#| msgid "the current status of the run" +msgid "the response status code" +msgstr "o status da atual execução" + msgid "the result for {key}" msgstr "o resultado para {key}" diff --git a/locale/ru/flows.po b/locale/ru/flows.po index fff2f2201..e52fd6d20 100644 --- a/locale/ru/flows.po +++ b/locale/ru/flows.po @@ -3,7 +3,7 @@ #, fuzzy msgid "" msgstr "" -"POT-Creation-Date: 2024-02-28 12:31-0500\n" +"POT-Creation-Date: 2024-02-28 15:43-0500\n" "Language-Team: Russian (https://www.transifex.com/rapidpro/teams/226/ru/)\n" "Language: ru\n" "MIME-Version: 1.0\n" @@ -690,6 +690,9 @@ msgstr "" msgid "the localized category of the result" msgstr "" +msgid "the method and URL" +msgstr "" + msgid "the name" msgstr "" @@ -747,6 +750,15 @@ msgstr "" msgid "the preferred channel of the contact" msgstr "" +msgid "the response body if valid JSON" +msgstr "" + +msgid "the response headers" +msgstr "" + +msgid "the response status code" +msgstr "" + msgid "the result for {key}" msgstr "" diff --git a/services/webhooks/testdata/webhook_flow.json b/services/webhooks/testdata/webhook_flow.json index 801ecb4af..952e0e752 100644 --- a/services/webhooks/testdata/webhook_flow.json +++ b/services/webhooks/testdata/webhook_flow.json @@ -3,7 +3,7 @@ { "name": "Webhook Test", "uuid": "bb38eefb-3cd9-4f80-9867-9c84ae276f7a", - "spec_version": "13.1.0", + "spec_version": "13.3.0", "language": "eng", "type": "messaging", "nodes": [ @@ -66,7 +66,7 @@ { "uuid": "7553120e-735c-47a8-a531-3e1b892682b1", "type": "set_contact_name", - "name": "@webhook.name" + "name": "@webhook.json.name" }, { "uuid": "2646d38e-dd18-47f2-b765-1ad79c7c8e5e", @@ -75,7 +75,7 @@ "name": "Joined", "key": "joined" }, - "value": "@webhook.joined" + "value": "@webhook.json.joined" } ], "exits": [ diff --git a/test/testdata/runner/webhook_results.test.json b/test/testdata/runner/webhook_results.test.json index 8c6e429db..cc3ac30ab 100644 --- a/test/testdata/runner/webhook_results.test.json +++ b/test/testdata/runner/webhook_results.test.json @@ -229,6 +229,12 @@ }, { "created_on": "2018-07-06T12:30:29.123456789Z", + "step_uuid": "5ecda5fc-951c-437b-a17e-f85e49829fb9", + "text": "error evaluating @webhook.json: null doesn't support lookups", + "type": "error" + }, + { + "created_on": "2018-07-06T12:30:31.123456789Z", "msg": { "locale": "eng-US", "text": "1. \n2. ", @@ -239,7 +245,7 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:35.123456789Z", + "created_on": "2018-07-06T12:30:37.123456789Z", "elapsed_ms": 1000, "extraction": "valid", "request": "GET /2 HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n", @@ -253,7 +259,7 @@ }, { "category": "Success", - "created_on": "2018-07-06T12:30:39.123456789Z", + "created_on": "2018-07-06T12:30:41.123456789Z", "extra": { "greeting": "hello" }, @@ -264,7 +270,7 @@ "value": "200" }, { - "created_on": "2018-07-06T12:30:43.123456789Z", + "created_on": "2018-07-06T12:30:45.123456789Z", "msg": { "locale": "eng-US", "text": "Would you like to continue again?\n\n1. \n2. {greeting: hello}\n3. {greeting: hello}", @@ -275,8 +281,8 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:48.123456789Z", - "expires_on": "2018-07-13T12:30:47.123456789Z", + "created_on": "2018-07-06T12:30:50.123456789Z", + "expires_on": "2018-07-13T12:30:49.123456789Z", "step_uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034", "type": "msg_wait" } @@ -295,7 +301,7 @@ "exit_uuid": "bd58fffa-f763-4622-bed6-70f1fcd83159", "flow_uuid": "68dae09d-db22-4879-90a7-a89395e3167b", "node_uuid": "23eb8d34-59b6-46b6-991a-440381c54947", - "time": "2018-07-06T12:30:31.123456789Z" + "time": "2018-07-06T12:30:33.123456789Z" }, { "destination_uuid": "71e72160-bb45-4abf-ba22-ab646178722a", @@ -303,14 +309,14 @@ "flow_uuid": "68dae09d-db22-4879-90a7-a89395e3167b", "node_uuid": "4eab7a66-0b55-45f6-803f-129a6f49e723", "operand": "Success", - "time": "2018-07-06T12:30:41.123456789Z" + "time": "2018-07-06T12:30:43.123456789Z" }, { "destination_uuid": "a28a6ec4-8e43-4362-9c0f-32be98f0b00c", "exit_uuid": "20d4d0a1-b1a8-4bc8-a50d-c5f6cf09cc88", "flow_uuid": "68dae09d-db22-4879-90a7-a89395e3167b", "node_uuid": "71e72160-bb45-4abf-ba22-ab646178722a", - "time": "2018-07-06T12:30:45.123456789Z" + "time": "2018-07-06T12:30:47.123456789Z" } ], "session": { @@ -409,6 +415,12 @@ }, { "created_on": "2018-07-06T12:30:29.123456789Z", + "step_uuid": "5ecda5fc-951c-437b-a17e-f85e49829fb9", + "text": "error evaluating @webhook.json: null doesn't support lookups", + "type": "error" + }, + { + "created_on": "2018-07-06T12:30:31.123456789Z", "msg": { "locale": "eng-US", "text": "1. \n2. ", @@ -419,7 +431,7 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:35.123456789Z", + "created_on": "2018-07-06T12:30:37.123456789Z", "elapsed_ms": 1000, "extraction": "valid", "request": "GET /2 HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n", @@ -433,7 +445,7 @@ }, { "category": "Success", - "created_on": "2018-07-06T12:30:39.123456789Z", + "created_on": "2018-07-06T12:30:41.123456789Z", "extra": { "greeting": "hello" }, @@ -444,7 +456,7 @@ "value": "200" }, { - "created_on": "2018-07-06T12:30:43.123456789Z", + "created_on": "2018-07-06T12:30:45.123456789Z", "msg": { "locale": "eng-US", "text": "Would you like to continue again?\n\n1. \n2. {greeting: hello}\n3. {greeting: hello}", @@ -455,8 +467,8 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:48.123456789Z", - "expires_on": "2018-07-13T12:30:47.123456789Z", + "created_on": "2018-07-06T12:30:50.123456789Z", + "expires_on": "2018-07-13T12:30:49.123456789Z", "step_uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034", "type": "msg_wait" } @@ -467,7 +479,7 @@ "revision": 23, "uuid": "68dae09d-db22-4879-90a7-a89395e3167b" }, - "modified_on": "2018-07-06T12:30:50.123456789Z", + "modified_on": "2018-07-06T12:30:52.123456789Z", "path": [ { "arrived_on": "2018-07-06T12:30:01.123456789Z", @@ -494,19 +506,19 @@ "uuid": "5ecda5fc-951c-437b-a17e-f85e49829fb9" }, { - "arrived_on": "2018-07-06T12:30:32.123456789Z", + "arrived_on": "2018-07-06T12:30:34.123456789Z", "exit_uuid": "24493dc0-687e-4d16-98e5-6e422624729b", "node_uuid": "4eab7a66-0b55-45f6-803f-129a6f49e723", "uuid": "a4d15ed4-5b24-407f-b86e-4b881f09a186" }, { - "arrived_on": "2018-07-06T12:30:42.123456789Z", + "arrived_on": "2018-07-06T12:30:44.123456789Z", "exit_uuid": "20d4d0a1-b1a8-4bc8-a50d-c5f6cf09cc88", "node_uuid": "71e72160-bb45-4abf-ba22-ab646178722a", "uuid": "b88ce93d-4360-4455-a691-235cbe720980" }, { - "arrived_on": "2018-07-06T12:30:46.123456789Z", + "arrived_on": "2018-07-06T12:30:48.123456789Z", "node_uuid": "a28a6ec4-8e43-4362-9c0f-32be98f0b00c", "uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034" } @@ -522,7 +534,7 @@ }, "call_2": { "category": "Success", - "created_on": "2018-07-06T12:30:37.123456789Z", + "created_on": "2018-07-06T12:30:39.123456789Z", "extra": { "greeting": "hello" }, @@ -572,7 +584,7 @@ { "events": [ { - "created_on": "2018-07-06T12:30:52.123456789Z", + "created_on": "2018-07-06T12:30:54.123456789Z", "msg": { "text": "Sure", "urn": "tel:+12065551212", @@ -583,7 +595,7 @@ }, { "category": "All Responses", - "created_on": "2018-07-06T12:30:56.123456789Z", + "created_on": "2018-07-06T12:30:58.123456789Z", "input": "Sure", "name": "Response 2", "step_uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034", @@ -591,13 +603,13 @@ "value": "Sure" }, { - "created_on": "2018-07-06T12:31:00.123456789Z", + "created_on": "2018-07-06T12:31:02.123456789Z", "step_uuid": "44fe8d72-00ed-4736-acca-bbca70987315", "text": "deprecated context value accessed: webhook recreated from extra", "type": "warning" }, { - "created_on": "2018-07-06T12:31:02.123456789Z", + "created_on": "2018-07-06T12:31:04.123456789Z", "msg": { "locale": "eng-US", "text": "Finally..\n\n1. \n2. {greeting: hello}\n3. {greeting: hello}", @@ -615,7 +627,7 @@ "flow_uuid": "68dae09d-db22-4879-90a7-a89395e3167b", "node_uuid": "a28a6ec4-8e43-4362-9c0f-32be98f0b00c", "operand": "Sure", - "time": "2018-07-06T12:30:58.123456789Z" + "time": "2018-07-06T12:31:00.123456789Z" } ], "session": { @@ -714,6 +726,12 @@ }, { "created_on": "2018-07-06T12:30:29.123456789Z", + "step_uuid": "5ecda5fc-951c-437b-a17e-f85e49829fb9", + "text": "error evaluating @webhook.json: null doesn't support lookups", + "type": "error" + }, + { + "created_on": "2018-07-06T12:30:31.123456789Z", "msg": { "locale": "eng-US", "text": "1. \n2. ", @@ -724,7 +742,7 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:35.123456789Z", + "created_on": "2018-07-06T12:30:37.123456789Z", "elapsed_ms": 1000, "extraction": "valid", "request": "GET /2 HTTP/1.1\r\nHost: temba.io\r\nUser-Agent: goflow-testing\r\nContent-Type: application/json\r\nAccept-Encoding: gzip\r\n\r\n", @@ -738,7 +756,7 @@ }, { "category": "Success", - "created_on": "2018-07-06T12:30:39.123456789Z", + "created_on": "2018-07-06T12:30:41.123456789Z", "extra": { "greeting": "hello" }, @@ -749,7 +767,7 @@ "value": "200" }, { - "created_on": "2018-07-06T12:30:43.123456789Z", + "created_on": "2018-07-06T12:30:45.123456789Z", "msg": { "locale": "eng-US", "text": "Would you like to continue again?\n\n1. \n2. {greeting: hello}\n3. {greeting: hello}", @@ -760,13 +778,13 @@ "type": "msg_created" }, { - "created_on": "2018-07-06T12:30:48.123456789Z", - "expires_on": "2018-07-13T12:30:47.123456789Z", + "created_on": "2018-07-06T12:30:50.123456789Z", + "expires_on": "2018-07-13T12:30:49.123456789Z", "step_uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034", "type": "msg_wait" }, { - "created_on": "2018-07-06T12:30:52.123456789Z", + "created_on": "2018-07-06T12:30:54.123456789Z", "msg": { "text": "Sure", "urn": "tel:+12065551212", @@ -777,7 +795,7 @@ }, { "category": "All Responses", - "created_on": "2018-07-06T12:30:56.123456789Z", + "created_on": "2018-07-06T12:30:58.123456789Z", "input": "Sure", "name": "Response 2", "step_uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034", @@ -785,13 +803,13 @@ "value": "Sure" }, { - "created_on": "2018-07-06T12:31:00.123456789Z", + "created_on": "2018-07-06T12:31:02.123456789Z", "step_uuid": "44fe8d72-00ed-4736-acca-bbca70987315", "text": "deprecated context value accessed: webhook recreated from extra", "type": "warning" }, { - "created_on": "2018-07-06T12:31:02.123456789Z", + "created_on": "2018-07-06T12:31:04.123456789Z", "msg": { "locale": "eng-US", "text": "Finally..\n\n1. \n2. {greeting: hello}\n3. {greeting: hello}", @@ -802,13 +820,13 @@ "type": "msg_created" } ], - "exited_on": "2018-07-06T12:31:04.123456789Z", + "exited_on": "2018-07-06T12:31:06.123456789Z", "flow": { "name": "Webhook Results", "revision": 23, "uuid": "68dae09d-db22-4879-90a7-a89395e3167b" }, - "modified_on": "2018-07-06T12:31:04.123456789Z", + "modified_on": "2018-07-06T12:31:06.123456789Z", "path": [ { "arrived_on": "2018-07-06T12:30:01.123456789Z", @@ -835,25 +853,25 @@ "uuid": "5ecda5fc-951c-437b-a17e-f85e49829fb9" }, { - "arrived_on": "2018-07-06T12:30:32.123456789Z", + "arrived_on": "2018-07-06T12:30:34.123456789Z", "exit_uuid": "24493dc0-687e-4d16-98e5-6e422624729b", "node_uuid": "4eab7a66-0b55-45f6-803f-129a6f49e723", "uuid": "a4d15ed4-5b24-407f-b86e-4b881f09a186" }, { - "arrived_on": "2018-07-06T12:30:42.123456789Z", + "arrived_on": "2018-07-06T12:30:44.123456789Z", "exit_uuid": "20d4d0a1-b1a8-4bc8-a50d-c5f6cf09cc88", "node_uuid": "71e72160-bb45-4abf-ba22-ab646178722a", "uuid": "b88ce93d-4360-4455-a691-235cbe720980" }, { - "arrived_on": "2018-07-06T12:30:46.123456789Z", + "arrived_on": "2018-07-06T12:30:48.123456789Z", "exit_uuid": "066c4b62-72f2-460b-a671-b4fa919c745a", "node_uuid": "a28a6ec4-8e43-4362-9c0f-32be98f0b00c", "uuid": "4f15f627-b1e2-4851-8dbf-00ecf5d03034" }, { - "arrived_on": "2018-07-06T12:30:59.123456789Z", + "arrived_on": "2018-07-06T12:31:01.123456789Z", "exit_uuid": "28236174-02ce-49b0-bdce-403afd9850fb", "node_uuid": "066c0ea6-68f1-4849-a4f5-5ef3465e9e97", "uuid": "44fe8d72-00ed-4736-acca-bbca70987315" @@ -870,7 +888,7 @@ }, "call_2": { "category": "Success", - "created_on": "2018-07-06T12:30:37.123456789Z", + "created_on": "2018-07-06T12:30:39.123456789Z", "extra": { "greeting": "hello" }, @@ -889,7 +907,7 @@ }, "response_2": { "category": "All Responses", - "created_on": "2018-07-06T12:30:54.123456789Z", + "created_on": "2018-07-06T12:30:56.123456789Z", "input": "Sure", "name": "Response 2", "node_uuid": "a28a6ec4-8e43-4362-9c0f-32be98f0b00c",