package messenger import ( "bytes" "encoding/json" "fmt" "image" "image/jpeg" "io" "mime/multipart" "net/http" ) const ( // SendMessageURL is API endpoint for sending messages. SendMessageURL = "https://graph.facebook.com/v2.6/me/messages" ) // Response is used for responding to events with messages. type Response struct { token string to Recipient } // Text sends a textual message. func (r *Response) Text(message string) error { return r.TextWithReplies(message, nil) } // TextWithReplies sends a textual message with some replies func (r *Response) TextWithReplies(message string, replies []QuickReply) error { m := SendMessage{ Recipient: r.to, Message: MessageData{ Text: message, QuickReplies: replies, }, } data, err := json.Marshal(m) if err != nil { return nil } req, err := http.NewRequest("POST", SendMessageURL, bytes.NewBuffer(data)) if err != nil { return err } req.Header.Set("Content-Type", "application/json") req.URL.RawQuery = "access_token=" + r.token client := &http.Client{} resp, err := client.Do(req) defer resp.Body.Close() return err } // Image sends an image. func (r *Response) Image(im image.Image) error { var b bytes.Buffer w := multipart.NewWriter(&b) data, err := w.CreateFormFile("fielddata", "meme.jpg") if err != nil { return err } imageBytes := new(bytes.Buffer) err = jpeg.Encode(imageBytes, im, nil) if err != nil { return err } _, err = io.Copy(data, imageBytes) if err != nil { return err } w.WriteField("recipient", fmt.Sprintf(`{"id":"%v"}`, r.to.ID)) w.WriteField("message", `{"attachment":{"type":"image", "payload":{}}}`) req, err := http.NewRequest("POST", SendMessageURL, &b) if err != nil { return err } req.URL.RawQuery = "access_token=" + r.token req.Header.Set("Content-Type", w.FormDataContentType()) client := &http.Client{} resp, err := client.Do(req) if err != nil { return err } var res bytes.Buffer res.ReadFrom(resp.Body) fmt.Println(res.String(), "DONE!") return nil } // ButtonTemplate sends a message with the main contents being button elements func (r *Response) ButtonTemplate(text string, buttons *[]StructuredMessageButton) error { m := SendStructuredMessage{ Recipient: r.to, Message: StructuredMessageData{ Attachment: StructuredMessageAttachment{ Type: "template", Payload: StructuredMessagePayload{ TemplateType: "button", Text: text, Buttons: buttons, Elements: nil, }, }, }, } data, err := json.Marshal(m) if err != nil { return nil } req, err := http.NewRequest("POST", SendMessageURL, bytes.NewBuffer(data)) if err != nil { return err } req.Header.Set("Content-Type", "application/json") req.URL.RawQuery = "access_token=" + r.token client := &http.Client{} resp, err := client.Do(req) defer resp.Body.Close() return err } // GenericTemplate is a message which allows for structural elements to be sent func (r *Response) GenericTemplate(text string, elements *[]StructuredMessageElement) error { m := SendStructuredMessage{ Recipient: r.to, Message: StructuredMessageData{ Attachment: StructuredMessageAttachment{ Type: "template", Payload: StructuredMessagePayload{ TemplateType: "generic", Buttons: nil, Elements: elements, }, }, }, } data, err := json.Marshal(m) if err != nil { return nil } req, err := http.NewRequest("POST", SendMessageURL, bytes.NewBuffer(data)) if err != nil { return err } req.Header.Set("Content-Type", "application/json") req.URL.RawQuery = "access_token=" + r.token client := &http.Client{} resp, err := client.Do(req) defer resp.Body.Close() return err } // SendMessage is the information sent in an API request to Facebook. type SendMessage struct { Recipient Recipient `json:"recipient"` Message MessageData `json:"message"` } // MessageData is a text message with optional replies to be sent. type MessageData struct { Text string `json:"text,omitempty"` QuickReplies []QuickReply `json:"quick_replies,omitempty"` } // SendStructuredMessage is a structured message template. type SendStructuredMessage struct { Recipient Recipient `json:"recipient"` Message StructuredMessageData `json:"message"` } // StructuredMessageData is an attachment sent with a structured message. type StructuredMessageData struct { Attachment StructuredMessageAttachment `json:"attachment"` } // StructuredMessageAttachment is the attachment of a structured message. type StructuredMessageAttachment struct { // Type must be template Type string `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 { // TemplateType must be button, generic or receipt TemplateType string `json:"template_type"` Text string `json:"text,omitempty"` Elements *[]StructuredMessageElement `json:"elements,omitempty"` Buttons *[]StructuredMessageButton `json:"buttons,omitempty"` } // StructuredMessageElement is a response containing structural elements type StructuredMessageElement struct { Title string `json:"title"` ImageURL string `json:"image_url"` Subtitle string `json:"subtitle"` Buttons []StructuredMessageButton `json:"buttons"` } // StructuredMessageButton is a response containing buttons type StructuredMessageButton struct { Type string `json:"type"` URL string `json:"url"` Title string `json:"title"` }