Skip to content

Commit

Permalink
Add media support config for D3C
Browse files Browse the repository at this point in the history
  • Loading branch information
norkans7 committed Sep 5, 2023
1 parent fe66ad4 commit f42aadb
Show file tree
Hide file tree
Showing 3 changed files with 85 additions and 44 deletions.
57 changes: 35 additions & 22 deletions handlers/dialog360/dialog360.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,15 @@ var (
maxMsgLength = 1000
)

// see https://developers.facebook.com/docs/whatsapp/cloud-api/reference/media#supported-media-types
var mediaSupport = map[handlers.MediaType]handlers.MediaTypeSupport{
handlers.MediaType("image/webp"): {Types: []string{"image/webp"}, MaxBytes: 100 * 1024, MaxWidth: 512, MaxHeight: 512},
handlers.MediaTypeImage: {Types: []string{"image/jpeg", "image/png"}, MaxBytes: 5 * 1024 * 1024},
handlers.MediaTypeAudio: {Types: []string{"audio/aac", "audio/mp4", "audio/mpeg", "audio/amr", "audio/ogg"}, MaxBytes: 16 * 1024 * 1024},
handlers.MediaTypeVideo: {Types: []string{"video/mp4", "video/3gp"}, MaxBytes: 16 * 1024 * 1024},
handlers.MediaTypeApplication: {MaxBytes: 100 * 1024 * 1024},
}

func init() {
courier.RegisterHandler(newWAHandler(courier.ChannelType("D3C"), "360Dialog"))
}
Expand Down Expand Up @@ -549,10 +558,15 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann

var payloadAudio wacMTPayload

for i := 0; i < len(msgParts)+len(msg.Attachments()); i++ {
attachments, err := handlers.ResolveAttachments(ctx, h.Backend(), msg.Attachments(), mediaSupport, false)
if err != nil {
return nil, errors.Wrap(err, "error resolving attachments")
}

for i := 0; i < len(msgParts)+len(attachments); i++ {
payload := wacMTPayload{MessagingProduct: "whatsapp", RecipientType: "individual", To: msg.URN().Path()}

if len(msg.Attachments()) == 0 {
if len(attachments) == 0 {
// do we have a template?
templating, err := h.getTemplating(msg)
if err != nil {
Expand All @@ -573,14 +587,14 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
template.Components = append(payload.Template.Components, component)

} else {
if i < (len(msgParts) + len(msg.Attachments()) - 1) {
if i < (len(msgParts) + len(attachments) - 1) {
// this is still a msg part
text := &wacText{PreviewURL: false}
payload.Type = "text"
if strings.Contains(msgParts[i-len(msg.Attachments())], "https://") || strings.Contains(msgParts[i-len(msg.Attachments())], "http://") {
if strings.Contains(msgParts[i-len(attachments)], "https://") || strings.Contains(msgParts[i-len(attachments)], "http://") {
text.PreviewURL = true
}
text.Body = msgParts[i-len(msg.Attachments())]
text.Body = msgParts[i-len(attachments)]
payload.Text = text
} else {
if len(qrs) > 0 {
Expand All @@ -589,7 +603,7 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
if len(qrs) <= 3 {
interactive := wacInteractive{Type: "button", Body: struct {
Text string "json:\"text\""
}{Text: msgParts[i-len(msg.Attachments())]}}
}{Text: msgParts[i-len(attachments)]}}

btns := make([]wacMTButton, len(qrs))
for i, qr := range qrs {
Expand All @@ -608,7 +622,7 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
} else if len(qrs) <= 10 {
interactive := wacInteractive{Type: "list", Body: struct {
Text string "json:\"text\""
}{Text: msgParts[i-len(msg.Attachments())]}}
}{Text: msgParts[i-len(attachments)]}}

section := wacMTSection{
Rows: make([]wacMTSectionRow, len(qrs)),
Expand Down Expand Up @@ -636,33 +650,32 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
// this is still a msg part
text := &wacText{PreviewURL: false}
payload.Type = "text"
if strings.Contains(msgParts[i-len(msg.Attachments())], "https://") || strings.Contains(msgParts[i-len(msg.Attachments())], "http://") {
if strings.Contains(msgParts[i-len(attachments)], "https://") || strings.Contains(msgParts[i-len(attachments)], "http://") {
text.PreviewURL = true
}
text.Body = msgParts[i-len(msg.Attachments())]
text.Body = msgParts[i-len(attachments)]
payload.Text = text
}
}
}

} else if i < len(msg.Attachments()) && (len(qrs) == 0 || len(qrs) > 3) {
attType, attURL := handlers.SplitAttachment(msg.Attachments()[i])
splitedAttType := strings.Split(attType, "/")
attType = splitedAttType[0]
attFormat := splitedAttType[1]
} else if i < len(attachments) && (len(qrs) == 0 || len(qrs) > 3) {
attURL := attachments[i].Media.URL()
attType := attachments[i].Type
attContentType := attachments[i].Media.ContentType()
if attType == "application" {
attType = "document"
}
payload.Type = attType
payload.Type = string(attType)
media := wacMTMedia{Link: attURL}

if len(msgParts) == 1 && attType != "audio" && len(msg.Attachments()) == 1 && len(msg.QuickReplies()) == 0 {
if len(msgParts) == 1 && attType != "audio" && len(attachments) == 1 && len(msg.QuickReplies()) == 0 {
media.Caption = msgParts[i]
hasCaption = true
}

if attType == "image" {
if attFormat == "webp" {
if attContentType == "image/webp" {
payload.Type = "sticker"
payload.Sticker = &media
} else {
Expand Down Expand Up @@ -692,10 +705,10 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
Text string "json:\"text\""
}{Text: msgParts[i]}}

if len(msg.Attachments()) > 0 {
if len(attachments) > 0 {
hasCaption = true
attType, attURL := handlers.SplitAttachment(msg.Attachments()[i])
attType = strings.Split(attType, "/")[0]
attURL := attachments[i].Media.URL()
attType := attachments[i].Type
if attType == "application" {
attType = "document"
}
Expand Down Expand Up @@ -799,10 +812,10 @@ func (h *handler) Send(ctx context.Context, msg courier.Msg, clog *courier.Chann
// this is still a msg part
text := &wacText{PreviewURL: false}
payload.Type = "text"
if strings.Contains(msgParts[i-len(msg.Attachments())], "https://") || strings.Contains(msgParts[i-len(msg.Attachments())], "http://") {
if strings.Contains(msgParts[i-len(attachments)], "https://") || strings.Contains(msgParts[i-len(attachments)], "http://") {
text.PreviewURL = true
}
text.Body = msgParts[i-len(msg.Attachments())]
text.Body = msgParts[i-len(attachments)]
payload.Text = text
}
}
Expand Down
70 changes: 49 additions & 21 deletions handlers/dialog360/dialog360_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -289,6 +289,10 @@ func buildMockD3MediaService(testChannels []courier.Channel, testCases []Channel
fileURL = "https://lookaside.fbsbx.com/whatsapp_business/attachments/?mid=id_audio"
}

if strings.HasSuffix(r.URL.Path, "id_sticker") {
fileURL = "https://lookaside.fbsbx.com/whatsapp_business/attachments/?mid=id_sticker"
}

w.WriteHeader(http.StatusOK)
w.Write([]byte(fmt.Sprintf(`{ "url": "%s" }`, fileURL)))
}))
Expand Down Expand Up @@ -364,12 +368,12 @@ var SendTestCasesD3C = []ChannelSendTestCase{
Label: "Audio Send",
MsgText: "audio caption",
MsgURN: "whatsapp:250788123123",
MsgAttachments: []string{"audio/mpeg:https://foo.bar/audio.mp3"},
MsgAttachments: []string{"audio/mpeg:http://mock.com/3456/test.mp3"},
MockResponses: map[MockedRequest]*httpx.MockResponse{
{
Method: "POST",
Path: "/messages",
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"audio","audio":{"link":"https://foo.bar/audio.mp3"}}`,
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"audio","audio":{"link":"http://mock.com/3456/test.mp3"}}`,
}: httpx.NewMockResponse(201, nil, []byte(`{ "messages": [{"id": "157b5e14568e8"}] }`)),
{
Method: "POST",
Expand All @@ -385,10 +389,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
Label: "Document Send",
MsgText: "document caption",
MsgURN: "whatsapp:250788123123",
MsgAttachments: []string{"application/pdf:https://foo.bar/document.pdf"},
MsgAttachments: []string{"application/pdf:http://mock.com/7890/test.pdf"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"document","document":{"link":"https://foo.bar/document.pdf","caption":"document caption","filename":"document.pdf"}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"document","document":{"link":"http://mock.com/7890/test.pdf","caption":"document caption","filename":"test.pdf"}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -398,10 +402,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
Label: "Image Send",
MsgText: "image caption",
MsgURN: "whatsapp:250788123123",
MsgAttachments: []string{"image/jpeg:https://foo.bar/image.jpg"},
MsgAttachments: []string{"image/jpeg:http://mock.com/1234/test.jpg"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"image","image":{"link":"https://foo.bar/image.jpg","caption":"image caption"}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"image","image":{"link":"http://mock.com/1234/test.jpg","caption":"image caption"}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -411,10 +415,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
Label: "Sticker Send",
MsgText: "sticker caption",
MsgURN: "whatsapp:250788123123",
MsgAttachments: []string{"image/webp:https://foo.bar/sticker.webp"},
MsgAttachments: []string{"image/webp:http://mock.com/8901/test.webp"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"sticker","sticker":{"link":"https://foo.bar/sticker.webp","caption":"sticker caption"}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"sticker","sticker":{"link":"http://mock.com/8901/test.webp","caption":"sticker caption"}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -424,10 +428,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
Label: "Video Send",
MsgText: "video caption",
MsgURN: "whatsapp:250788123123",
MsgAttachments: []string{"video/mp4:https://foo.bar/video.mp4"},
MsgAttachments: []string{"video/mp4:http://mock.com/5678/test.mp4"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"video","video":{"link":"https://foo.bar/video.mp4","caption":"video caption"}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"video","video":{"link":"http://mock.com/5678/test.mp4","caption":"video caption"}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand Down Expand Up @@ -514,10 +518,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
MsgText: "Interactive Button Msg",
MsgURN: "whatsapp:250788123123",
MsgQuickReplies: []string{"BUTTON1"},
MsgAttachments: []string{"image/jpeg:https://foo.bar/image.jpg"},
MsgAttachments: []string{"image/jpeg:http://mock.com/1234/test.jpg"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"image","image":{"link":"https://foo.bar/image.jpg"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"image","image":{"link":"http://mock.com/1234/test.jpg"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -528,10 +532,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
MsgText: "Interactive Button Msg",
MsgURN: "whatsapp:250788123123",
MsgQuickReplies: []string{"BUTTON1"},
MsgAttachments: []string{"video/mp4:https://foo.bar/video.mp4"},
MsgAttachments: []string{"video/mp4:http://mock.com/5678/test.mp4"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"video","video":{"link":"https://foo.bar/video.mp4"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"video","video":{"link":"http://mock.com/5678/test.mp4"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -542,10 +546,10 @@ var SendTestCasesD3C = []ChannelSendTestCase{
MsgText: "Interactive Button Msg",
MsgURN: "whatsapp:250788123123",
MsgQuickReplies: []string{"BUTTON1"},
MsgAttachments: []string{"document/pdf:https://foo.bar/document.pdf"},
MsgAttachments: []string{"document/pdf:http://mock.com/7890/test.pdf"},
MockResponseBody: `{ "messages": [{"id": "157b5e14568e8"}] }`,
MockResponseStatus: 201,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"document","document":{"link":"https://foo.bar/document.pdf","filename":"document.pdf"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestBody: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"interactive","interactive":{"type":"button","header":{"type":"document","document":{"link":"http://mock.com/7890/test.pdf","filename":"test.pdf"}},"body":{"text":"Interactive Button Msg"},"action":{"buttons":[{"type":"reply","reply":{"id":"0","title":"BUTTON1"}}]}}}`,
ExpectedRequestPath: "/messages",
ExpectedMsgStatus: "W",
ExpectedExternalID: "157b5e14568e8",
Expand All @@ -556,12 +560,12 @@ var SendTestCasesD3C = []ChannelSendTestCase{
MsgText: "Interactive Button Msg",
MsgURN: "whatsapp:250788123123",
MsgQuickReplies: []string{"ROW1", "ROW2", "ROW3"},
MsgAttachments: []string{"audio/mp3:https://foo.bar/audio.mp3"},
MsgAttachments: []string{"audio/mp3:http://mock.com/3456/test.mp3"},
MockResponses: map[MockedRequest]*httpx.MockResponse{
{
Method: "POST",
Path: "/messages",
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"audio","audio":{"link":"https://foo.bar/audio.mp3"}}`,
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"audio","audio":{"link":"http://mock.com/3456/test.mp3"}}`,
}: httpx.NewMockResponse(201, nil, []byte(`{ "messages": [{"id": "157b5e14568e8"}] }`)),
{
Method: "POST",
Expand All @@ -578,12 +582,12 @@ var SendTestCasesD3C = []ChannelSendTestCase{
MsgText: "Interactive List Msg",
MsgURN: "whatsapp:250788123123",
MsgQuickReplies: []string{"ROW1", "ROW2", "ROW3", "ROW4"},
MsgAttachments: []string{"image/jpeg:https://foo.bar/image.jpg"},
MsgAttachments: []string{"image/jpeg:http://mock.com/1234/test.jpg"},
MockResponses: map[MockedRequest]*httpx.MockResponse{
{
Method: "POST",
Path: "/messages",
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"image","image":{"link":"https://foo.bar/image.jpg"}}`,
Body: `{"messaging_product":"whatsapp","recipient_type":"individual","to":"250788123123","type":"image","image":{"link":"http://mock.com/1234/test.jpg"}}`,
}: httpx.NewMockResponse(201, nil, []byte(`{ "messages": [{"id": "157b5e14568e8"}] }`)),
{
Method: "POST",
Expand Down Expand Up @@ -629,6 +633,30 @@ var SendTestCasesD3C = []ChannelSendTestCase{
},
}

// setupMedia takes care of having the media files needed to our test server host
func setupMedia(mb *test.MockBackend) {
imageJPG := test.NewMockMedia("test.jpg", "image/jpeg", "http://mock.com/1234/test.jpg", 1024*1024, 640, 480, 0, nil)

audioM4A := test.NewMockMedia("test.m4a", "audio/mp4", "http://mock.com/2345/test.m4a", 1024*1024, 0, 0, 200, nil)
audioMP3 := test.NewMockMedia("test.mp3", "audio/mpeg", "http://mock.com/3456/test.mp3", 1024*1024, 0, 0, 200, []courier.Media{audioM4A})

thumbJPG := test.NewMockMedia("test.jpg", "image/jpeg", "http://mock.com/4567/test.jpg", 1024*1024, 640, 480, 0, nil)
videoMP4 := test.NewMockMedia("test.mp4", "video/mp4", "http://mock.com/5678/test.mp4", 1024*1024, 0, 0, 1000, []courier.Media{thumbJPG})

videoMOV := test.NewMockMedia("test.mov", "video/quicktime", "http://mock.com/6789/test.mov", 100*1024*1024, 0, 0, 2000, nil)

filePDF := test.NewMockMedia("test.pdf", "application/pdf", "http://mock.com/7890/test.pdf", 100*1024*1024, 0, 0, 0, nil)

stickerWEBP := test.NewMockMedia("test.webp", "image/webp", "http://mock.com/8901/test.webp", 50*1024, 480, 480, 0, nil)

mb.MockMedia(imageJPG)
mb.MockMedia(audioMP3)
mb.MockMedia(videoMP4)
mb.MockMedia(videoMOV)
mb.MockMedia(filePDF)
mb.MockMedia(stickerWEBP)
}

func TestSending(t *testing.T) {
// shorter max msg length for testing
maxMsgLength = 100
Expand All @@ -639,7 +667,7 @@ func TestSending(t *testing.T) {
})
checkRedacted := []string{"the-auth-token"}

RunChannelSendTestCases(t, ChannelWAC, newWAHandler(courier.ChannelType("D3C"), "360Dialog"), SendTestCasesD3C, checkRedacted, nil)
RunChannelSendTestCases(t, ChannelWAC, newWAHandler(courier.ChannelType("D3C"), "360Dialog"), SendTestCasesD3C, checkRedacted, setupMedia)
}
func TestGetSupportedLanguage(t *testing.T) {
assert.Equal(t, languageInfo{"en", "Menu"}, getSupportedLanguage(courier.NilLocale))
Expand Down
2 changes: 1 addition & 1 deletion handlers/dialog360/testdata/wac/audioWAC.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@
"audio": {
"file": "/usr/local/wamedia/shared/b1cf38-8734-4ad3-b4a1-ef0c10d0d683",
"id": "id_audio",
"mime_type": "image/jpeg",
"mime_type": "audio/mpeg",
"sha256": "29ed500fa64eb55fc19dc4124acb300e5dcc54a0f822a301ae99944db",
"caption": "Check out my new phone!"
},
Expand Down

0 comments on commit f42aadb

Please sign in to comment.