1
0
mirror of synced 2024-11-25 14:26:09 +03:00
messenger/response.go

553 lines
17 KiB
Go
Raw Normal View History

2016-04-13 10:01:42 +03:00
package messenger
import (
"bytes"
"encoding/json"
2016-04-16 11:01:32 +03:00
"fmt"
"image"
"image/jpeg"
"io"
"io/ioutil"
2016-04-16 11:01:32 +03:00
"mime/multipart"
2016-04-13 10:01:42 +03:00
"net/http"
"net/textproto"
"strings"
2019-07-31 00:38:58 +03:00
"golang.org/x/xerrors"
2016-04-13 10:01:42 +03:00
)
// AttachmentType is attachment type.
type AttachmentType string
type MessagingType string
type TopElementStyle string
type ImageAspectRatio string
2016-04-13 10:01:42 +03:00
const (
2021-06-04 14:42:12 +03:00
// DefaultSendAPIVersion is a default Send API version
DefaultSendAPIVersion = "v2.11"
2016-04-14 03:23:45 +03:00
// SendMessageURL is API endpoint for sending messages.
2021-06-04 14:42:12 +03:00
SendMessageURL = "https://graph.facebook.com/%s/me/messages"
2019-03-25 21:42:34 +03:00
// ThreadControlURL is the API endpoint for passing thread control.
2021-06-04 14:42:12 +03:00
ThreadControlURL = "https://graph.facebook.com/%s/me/pass_thread_control"
2019-03-25 21:42:34 +03:00
// InboxPageID is managed by facebook for secondary pass to inbox features: https://developers.facebook.com/docs/messenger-platform/handover-protocol/pass-thread-control
InboxPageID = 263902037430900
// ImageAttachment is image attachment type.
ImageAttachment AttachmentType = "image"
// AudioAttachment is audio attachment type.
AudioAttachment AttachmentType = "audio"
// VideoAttachment is video attachment type.
VideoAttachment AttachmentType = "video"
// FileAttachment is file attachment type.
FileAttachment AttachmentType = "file"
// ResponseType is response messaging type.
ResponseType MessagingType = "RESPONSE"
// UpdateType is update messaging type.
UpdateType MessagingType = "UPDATE"
// MessageTagType is message_tag messaging type.
MessageTagType MessagingType = "MESSAGE_TAG"
// NonPromotionalSubscriptionType is NON_PROMOTIONAL_SUBSCRIPTION messaging type.
NonPromotionalSubscriptionType MessagingType = "NON_PROMOTIONAL_SUBSCRIPTION"
// TopElementStyle is compact.
CompactTopElementStyle TopElementStyle = "compact"
// TopElementStyle is large.
LargeTopElementStyle TopElementStyle = "large"
// ImageAspectRatio is horizontal (1.91:1). Default.
HorizontalImageAspectRatio ImageAspectRatio = "horizontal"
// ImageAspectRatio is square.
SquareImageAspectRatio ImageAspectRatio = "square"
2016-04-13 10:01:42 +03:00
)
// QueryResponse is the response sent back by Facebook when setting up things
// like greetings or call-to-actions.
type QueryResponse struct {
Error *QueryError `json:"error,omitempty"`
RecipientID string `json:"recipient_id"`
MessageID string `json:"message_id"`
}
// QueryError is representing an error sent back by Facebook.
type QueryError struct {
Message string `json:"message"`
Type string `json:"type"`
Code int `json:"code"`
ErrorSubcode int `json:"error_subcode"`
FBTraceID string `json:"fbtrace_id"`
}
// QueryError implements error.
2019-07-31 00:38:58 +03:00
func (e QueryError) Error() string {
return e.Message
}
func checkFacebookError(r io.Reader) error {
var err error
qr := QueryResponse{}
2022-01-27 17:17:20 +03:00
decoder := json.NewDecoder(r)
err = decoder.Decode(&qr)
2019-07-25 14:37:27 +03:00
if err != nil {
2022-01-28 12:24:58 +03:00
return NewUnmarshalError(err).WithReader(decoder.Buffered())
2019-07-25 14:37:27 +03:00
}
if qr.Error != nil {
2019-07-31 00:38:58 +03:00
return xerrors.Errorf("facebook error: %w", qr.Error)
}
2017-07-31 22:24:49 +03:00
return nil
}
func getFacebookQueryResponse(r io.Reader) (QueryResponse, error) {
qr := QueryResponse{}
2022-01-27 17:17:20 +03:00
decoder := json.NewDecoder(r)
if err := decoder.Decode(&qr); err != nil {
2022-01-28 12:24:58 +03:00
return qr, NewUnmarshalError(err).WithReader(decoder.Buffered())
}
if qr.Error != nil {
return qr, xerrors.Errorf("facebook error: %w", qr.Error)
}
return qr, nil
}
2016-04-14 03:23:45 +03:00
// Response is used for responding to events with messages.
2016-04-13 10:01:42 +03:00
type Response struct {
2021-06-04 14:42:12 +03:00
token string
to Recipient
sendAPIVersion string
2016-04-13 10:01:42 +03:00
}
2019-01-02 00:26:58 +03:00
// SetToken is for using DispatchMessage from outside.
func (r *Response) SetToken(token string) {
r.token = token
}
2016-04-14 03:23:45 +03:00
// Text sends a textual message.
func (r *Response) Text(message string, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
return r.TextWithReplies(message, nil, messagingType, metadata, tags...)
}
// TextWithReplies sends a textual message with some replies
// messagingType should be one of the following: "RESPONSE","UPDATE","MESSAGE_TAG","NON_PROMOTIONAL_SUBSCRIPTION"
// only supply tags when messagingType == "MESSAGE_TAG"
// (see https://developers.facebook.com/docs/messenger-platform/send-messages#messaging_types for more).
func (r *Response) TextWithReplies(message string, replies []QuickReply, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
var tag string
if len(tags) > 0 {
tag = tags[0]
}
2016-04-13 10:01:42 +03:00
m := SendMessage{
MessagingType: messagingType,
Recipient: r.to,
2016-04-13 10:01:42 +03:00
Message: MessageData{
Text: message,
Attachment: nil,
QuickReplies: replies,
Metadata: metadata,
},
Tag: tag,
}
return r.DispatchMessage(&m)
}
// AttachmentWithReplies sends a attachment message with some replies.
func (r *Response) AttachmentWithReplies(attachment *StructuredMessageAttachment, replies []QuickReply, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
var tag string
if len(tags) > 0 {
tag = tags[0]
}
m := SendMessage{
MessagingType: messagingType,
Recipient: r.to,
Message: MessageData{
Attachment: attachment,
QuickReplies: replies,
Metadata: metadata,
2016-04-13 10:01:42 +03:00
},
Tag: tag,
2016-04-13 10:01:42 +03:00
}
return r.DispatchMessage(&m)
2016-04-13 10:01:42 +03:00
}
2016-04-16 11:01:32 +03:00
// Image sends an image.
func (r *Response) Image(im image.Image) (QueryResponse, error) {
var qr QueryResponse
2016-04-16 11:01:32 +03:00
imageBytes := new(bytes.Buffer)
2016-11-11 06:28:32 +03:00
err := jpeg.Encode(imageBytes, im, nil)
2016-04-16 11:01:32 +03:00
if err != nil {
return qr, err
2016-04-16 11:01:32 +03:00
}
2016-11-11 06:28:32 +03:00
return r.AttachmentData(ImageAttachment, "meme.jpg", imageBytes)
2016-04-16 11:01:32 +03:00
}
// Attachment sends an image, sound, video or a regular file to a chat.
func (r *Response) Attachment(dataType AttachmentType, url string, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
var tag string
if len(tags) > 0 {
tag = tags[0]
}
m := SendStructuredMessage{
MessagingType: messagingType,
Recipient: r.to,
Message: StructuredMessageData{
Metadata: metadata,
Attachment: StructuredMessageAttachment{
Type: dataType,
Payload: StructuredMessagePayload{
Url: url,
},
},
},
Tag: tag,
}
return r.DispatchMessage(&m)
}
// copied from multipart package.
var quoteEscaper = strings.NewReplacer("\\", "\\\\", `"`, "\\\"")
// copied from multipart package.
func escapeQuotes(s string) string {
return quoteEscaper.Replace(s)
}
// copied from multipart package with slight changes due to fixed content-type there.
func createFormFile(filename string, w *multipart.Writer, contentType string) (io.Writer, error) {
h := make(textproto.MIMEHeader)
h.Set("Content-Disposition",
fmt.Sprintf(`form-data; name="filedata"; filename="%s"`,
escapeQuotes(filename)))
h.Set("Content-Type", contentType)
return w.CreatePart(h)
}
2016-11-11 06:28:32 +03:00
// AttachmentData sends an image, sound, video or a regular file to a chat via an io.Reader.
func (r *Response) AttachmentData(dataType AttachmentType, filename string, filedata io.Reader) (QueryResponse, error) {
var qr QueryResponse
2016-11-11 06:28:32 +03:00
filedataBytes, err := ioutil.ReadAll(filedata)
if err != nil {
return qr, err
}
contentType := http.DetectContentType(filedataBytes[:512])
fmt.Println("Content-type detected:", contentType)
var body bytes.Buffer
multipartWriter := multipart.NewWriter(&body)
data, err := createFormFile(filename, multipartWriter, contentType)
2016-11-11 06:28:32 +03:00
if err != nil {
return qr, err
2016-11-11 06:28:32 +03:00
}
_, err = bytes.NewBuffer(filedataBytes).WriteTo(data)
2016-11-11 06:28:32 +03:00
if err != nil {
return qr, err
2016-11-11 06:28:32 +03:00
}
multipartWriter.WriteField("recipient", fmt.Sprintf(`{"id":"%v"}`, r.to.ID))
multipartWriter.WriteField("message", fmt.Sprintf(`{"attachment":{"type":"%v", "payload":{}}}`, dataType))
2016-11-11 06:28:32 +03:00
2021-06-04 14:42:12 +03:00
req, err := http.NewRequest("POST", fmt.Sprintf(SendMessageURL, r.sendAPIVersion), &body)
2016-11-11 06:28:32 +03:00
if err != nil {
return qr, err
2016-11-11 06:28:32 +03:00
}
req.URL.RawQuery = "access_token=" + r.token
req.Header.Set("Content-Type", multipartWriter.FormDataContentType())
2016-11-11 06:28:32 +03:00
client := &http.Client{}
resp, err := client.Do(req)
if err != nil {
return qr, err
2016-11-11 06:28:32 +03:00
}
2019-11-15 17:01:19 +03:00
defer resp.Body.Close()
2016-11-11 06:28:32 +03:00
return getFacebookQueryResponse(resp.Body)
2016-11-11 06:28:32 +03:00
}
// ButtonTemplate sends a message with the main contents being button elements.
func (r *Response) ButtonTemplate(text string, buttons *[]StructuredMessageButton, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
var tag string
if len(tags) > 0 {
tag = tags[0]
}
2016-05-04 11:09:34 +03:00
m := SendStructuredMessage{
MessagingType: messagingType,
Recipient: r.to,
2016-05-04 11:09:34 +03:00
Message: StructuredMessageData{
Metadata: metadata,
2016-05-04 11:09:34 +03:00
Attachment: StructuredMessageAttachment{
Type: "template",
Payload: StructuredMessagePayload{
TemplateType: "button",
Text: text,
Buttons: buttons,
Elements: nil,
},
},
},
Tag: tag,
}
2016-05-04 11:09:34 +03:00
return r.DispatchMessage(&m)
}
// GenericTemplate is a message which allows for structural elements to be sent.
func (r *Response) GenericTemplate(elements *[]StructuredMessageElement, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
var tag string
if len(tags) > 0 {
tag = tags[0]
}
2016-05-04 11:09:34 +03:00
m := SendStructuredMessage{
MessagingType: messagingType,
Recipient: r.to,
2016-05-04 11:09:34 +03:00
Message: StructuredMessageData{
Metadata: metadata,
2016-05-04 11:09:34 +03:00
Attachment: StructuredMessageAttachment{
Type: "template",
Payload: StructuredMessagePayload{
TemplateType: "generic",
Buttons: nil,
Elements: elements,
},
},
},
Tag: tag,
}
return r.DispatchMessage(&m)
}
// ListTemplate sends a list of elements.
func (r *Response) ListTemplate(elements *[]StructuredMessageElement, messagingType MessagingType, tags ...string) (QueryResponse, error) {
2019-04-14 13:28:09 +03:00
var tag string
if len(tags) > 0 {
tag = tags[0]
}
m := SendStructuredMessage{
MessagingType: messagingType,
Recipient: r.to,
Message: StructuredMessageData{
Attachment: StructuredMessageAttachment{
Type: "template",
Payload: StructuredMessagePayload{
TopElementStyle: "compact",
TemplateType: "list",
Buttons: nil,
Elements: elements,
},
},
},
Tag: tag,
}
return r.DispatchMessage(&m)
}
2022-07-29 15:40:30 +03:00
// SenderAction sends an info about sender action.
func (r *Response) SenderAction(action SenderAction) (QueryResponse, error) {
m := SendSenderAction{
Recipient: r.to,
SenderAction: action,
}
return r.DispatchMessage(&m)
}
// DispatchMessage posts the message to messenger, return the error if there's any.
func (r *Response) DispatchMessage(m interface{}) (QueryResponse, error) {
var res QueryResponse
data, err := json.Marshal(m)
if err != nil {
return res, err
}
2021-06-04 14:42:12 +03:00
req, err := http.NewRequest("POST", fmt.Sprintf(SendMessageURL, r.sendAPIVersion), bytes.NewBuffer(data))
if err != nil {
return res, err
}
req.Header.Set("Content-Type", "application/json")
req.URL.RawQuery = "access_token=" + r.token
resp, err := http.DefaultClient.Do(req)
if err != nil {
return res, err
}
defer resp.Body.Close()
return getFacebookQueryResponse(resp.Body)
}
2019-03-25 21:42:34 +03:00
// PassThreadToInbox Uses Messenger Handover Protocol for live inbox
// https://developers.facebook.com/docs/messenger-platform/handover-protocol/#inbox
func (r *Response) PassThreadToInbox() error {
p := passThreadControl{
Recipient: r.to,
TargetAppID: InboxPageID,
Metadata: "Passing to inbox secondary app",
}
data, err := json.Marshal(p)
if err != nil {
return err
}
2021-06-04 14:42:12 +03:00
req, err := http.NewRequest("POST", fmt.Sprintf(ThreadControlURL, r.sendAPIVersion), bytes.NewBuffer(data))
2019-03-25 21:42:34 +03:00
if err != nil {
return err
}
req.Header.Set("Content-Type", "application/json")
req.URL.RawQuery = "access_token=" + r.token
resp, err := http.DefaultClient.Do(req)
if err != nil {
return err
}
defer resp.Body.Close()
return checkFacebookError(resp.Body)
}
2016-04-14 03:23:45 +03:00
// SendMessage is the information sent in an API request to Facebook.
2016-04-13 10:01:42 +03:00
type SendMessage struct {
MessagingType MessagingType `json:"messaging_type"`
Recipient Recipient `json:"recipient"`
Message MessageData `json:"message"`
Tag string `json:"tag,omitempty"`
2016-04-13 10:01:42 +03:00
}
// MessageData is a message consisting of text or an attachment, with an additional selection of optional quick replies.
2016-04-13 10:01:42 +03:00
type MessageData struct {
Text string `json:"text,omitempty"`
Attachment *StructuredMessageAttachment `json:"attachment,omitempty"`
QuickReplies []QuickReply `json:"quick_replies,omitempty"`
Metadata string `json:"metadata,omitempty"`
2016-04-13 10:01:42 +03:00
}
2016-05-04 11:09:34 +03:00
// SendStructuredMessage is a structured message template.
type SendStructuredMessage struct {
MessagingType MessagingType `json:"messaging_type"`
Recipient Recipient `json:"recipient"`
Message StructuredMessageData `json:"message"`
Tag string `json:"tag,omitempty"`
}
2016-05-04 11:09:34 +03:00
// StructuredMessageData is an attachment sent with a structured message.
type StructuredMessageData struct {
Attachment StructuredMessageAttachment `json:"attachment"`
Metadata string `json:"metadata,omitempty"`
}
2016-05-04 11:09:34 +03:00
// StructuredMessageAttachment is the attachment of a structured message.
type StructuredMessageAttachment struct {
2016-05-04 11:09:34 +03:00
// Type must be template
2019-02-22 19:16:14 +03:00
Title string `json:"title,omitempty"`
URL string `json:"url,omitempty"`
Type AttachmentType `json:"type"`
// Payload is the information for the file which was sent in the attachment.
Payload StructuredMessagePayload `json:"payload"`
}
// StructuredMessagePayload is the actual payload of an attachment.
type StructuredMessagePayload struct {
2016-05-04 11:09:34 +03:00
// TemplateType must be button, generic or receipt
TemplateType string `json:"template_type,omitempty"`
TopElementStyle TopElementStyle `json:"top_element_style,omitempty"`
Text string `json:"text,omitempty"`
ImageAspectRatio ImageAspectRatio `json:"image_aspect_ratio,omitempty"`
2018-11-13 16:00:30 +03:00
Sharable bool `json:"sharable,omitempty"`
Elements *[]StructuredMessageElement `json:"elements,omitempty"`
Buttons *[]StructuredMessageButton `json:"buttons,omitempty"`
Url string `json:"url,omitempty"`
2019-02-26 01:52:06 +03:00
AttachmentID string `json:"attachment_id,omitempty"`
ReceiptMessagePayload
}
type ReceiptMessagePayload struct {
RecipientName string `json:"recipient_name,omitempty"`
OrderNumber string `json:"order_number,omitempty"`
Currency string `json:"currency,omitempty"`
PaymentMethod string `json:"payment_method,omitempty"`
Timestamp int64 `json:"timestamp,omitempty"`
Address *Address `json:"address,omitempty"`
Summary *Summary `json:"summary,omitempty"`
Adjustments []Adjustment `json:"adjustments,omitempty"`
}
type Address struct {
Street1 string `json:"street_1,omitempty"`
Street2 string `json:"street_2,omitempty"`
City string `json:"city,omitempty"`
PostalCode string `json:"postal_code,omitempty"`
State string `json:"state,omitempty"`
Country string `json:"country,omitempty"`
}
type Summary struct {
Subtotal float32 `json:"subtotal,omitempty"`
ShippingCost float32 `json:"shipping_cost,omitempty"`
TotalTax float32 `json:"total_tax,omitempty"`
TotalCost float32 `json:"total_cost,omitempty"`
}
type Adjustment struct {
Name string `json:"name,omitempty"`
Amount float32 `json:"amount,omitempty"`
}
// StructuredMessageElement is a response containing structural elements.
type StructuredMessageElement struct {
Title string `json:"title"`
ImageURL string `json:"image_url"`
ItemURL string `json:"item_url,omitempty"`
Subtitle string `json:"subtitle"`
DefaultAction *DefaultAction `json:"default_action,omitempty"`
2021-02-17 15:25:52 +03:00
Buttons *[]StructuredMessageButton `json:"buttons,omitempty"`
ReceiptMessageElement
}
type ReceiptMessageElement struct {
Quantity float32 `json:"quantity,omitempty"`
Price float32 `json:"price,omitempty"`
Currency string `json:"currency,omitempty"`
}
// DefaultAction is a response containing default action properties.
type DefaultAction struct {
Type string `json:"type"`
URL string `json:"url,omitempty"`
WebviewHeightRatio string `json:"webview_height_ratio,omitempty"`
MessengerExtensions bool `json:"messenger_extensions,omitempty"`
FallbackURL string `json:"fallback_url,omitempty"`
WebviewShareButton string `json:"webview_share_button,omitempty"`
}
// StructuredMessageButton is a response containing buttons.
type StructuredMessageButton struct {
2019-02-27 01:07:06 +03:00
Type string `json:"type"`
URL string `json:"url,omitempty"`
Title string `json:"title,omitempty"`
Payload string `json:"payload,omitempty"`
WebviewHeightRatio string `json:"webview_height_ratio,omitempty"`
MessengerExtensions bool `json:"messenger_extensions,omitempty"`
FallbackURL string `json:"fallback_url,omitempty"`
WebviewShareButton string `json:"webview_share_button,omitempty"`
ShareContents *StructuredMessageData `json:"share_contents,omitempty"`
}
// SendSenderAction is the information about sender action.
type SendSenderAction struct {
2022-07-29 15:40:30 +03:00
Recipient Recipient `json:"recipient"`
SenderAction SenderAction `json:"sender_action"`
}