1
0
Fork 0
mirror of synced 2025-04-05 14:13:34 +03:00

Compare commits

..

25 commits

Author SHA1 Message Date
20736ef44d
Merge pull request #25 from Neur0toxine/send-reactions
add reaction sending support for Instagram
2025-03-21 16:15:48 +03:00
99c9a7ba00 remove Go version that was carelessly added by IDE 2025-03-21 16:13:25 +03:00
23ed06d035 add reaction sending support for Instagram 2025-03-21 16:08:48 +03:00
f017927e56
Merge pull request #21 from RenCurs/update-linter
update linter and it's rules
2025-03-21 13:50:43 +03:00
5c54b47713
Merge pull request #23 from Neur0toxine/add-reactions-messageinfo
add reactions to MessageInfo
2025-03-21 13:50:29 +03:00
354f45a393 add reactions to MessageInfo 2025-03-21 13:46:57 +03:00
8178ac8b66
Merge pull request #22 from dendd1/master
update ig referral message
2025-02-25 09:44:20 +03:00
Суханов Данила
4c8bef9e8f update ig referral message 2025-02-24 17:24:41 +03:00
Ruslan Efanov
99463a68f2 update linter and it's rules 2022-12-30 11:23:07 +03:00
e378e55563
add referer_uri field for referral message 2022-12-20 13:27:30 +03:00
Ruslan Efanov
26c6cd6cbf add referer_uri field for referral message 2022-12-19 16:59:30 +03:00
23f3b3123d
allow specifying content-type while sending a file 2022-11-07 15:51:09 +03:00
081a4d5aa8 allow specifying content-type while sending a file 2022-11-07 15:48:47 +03:00
6c997fba17
Sender actions support 2022-07-29 15:46:11 +03:00
2af7845f3b update linter 2022-07-29 15:44:46 +03:00
5bdc2eb804 Sender actions support 2022-07-29 15:40:30 +03:00
tishmaria90
243bf4a2b2
Add mid field in read model (#17) 2022-02-10 11:59:44 +03:00
Alex Lushpai
ae68d46308
Add unmarshal error 2022-01-31 13:34:15 +03:00
Tyschitskaya Maria
d0069e73f8 json error text tests fix 2022-01-28 12:28:24 +03:00
Tyschitskaya Maria
76d0d601a0 unmarshal error text 2022-01-28 12:24:58 +03:00
Tyschitskaya Maria
c1e9fd594d buffer from decoder in unmarshal error 2022-01-27 17:18:26 +03:00
Tyschitskaya Maria
f24f4d512e ioutil in unmarshal error 2022-01-27 14:56:47 +03:00
Tyschitskaya Maria
cd6859b074 add unmarshal error 2022-01-26 18:20:09 +03:00
Alex Lushpai
1fc20f8bb5
Ad referral fields 2022-01-14 15:40:35 +03:00
Maria Tyschitskaya
efc2b29474 add ad referral fields 2022-01-13 14:50:17 +03:00
11 changed files with 295 additions and 44 deletions

View file

@ -19,17 +19,24 @@ jobs:
steps: steps:
- name: Check out code into the Go module directory - name: Check out code into the Go module directory
uses: actions/checkout@v2 uses: actions/checkout@v2
- name: Lint code with golangci-lint - name: Set up Go 1.17
uses: golangci/golangci-lint-action@v2 uses: actions/setup-go@v2
with: with:
version: v1.36 # TODO: Should migrate to 1.18 later
go-version: '1.17'
- name: Get dependencies
run: go mod tidy
- name: Lint code with golangci-lint
uses: golangci/golangci-lint-action@v3
with:
version: v1.50.1
only-new-issues: true only-new-issues: true
tests: tests:
name: Tests name: Tests
runs-on: ubuntu-latest runs-on: ubuntu-latest
strategy: strategy:
matrix: matrix:
go-version: ['1.11', '1.12', '1.13', '1.14', '1.15', '1.16', '1.17'] go-version: ['1.13', '1.14', '1.15', '1.16', '1.17']
steps: steps:
- name: Set up Go ${{ matrix.go-version }} - name: Set up Go ${{ matrix.go-version }}
uses: actions/setup-go@v2 uses: actions/setup-go@v2

View file

@ -6,20 +6,33 @@ output:
format: colored-line-number format: colored-line-number
sort-results: true sort-results: true
# Linters below do not support go1.18 yet because of generics.
# See https://github.com/golangci/golangci-lint/issues/2649
# - bodyclose
# - sqlclosecheck
linters: linters:
disable-all: true disable-all: true
enable: enable:
- deadcode - paralleltest
- tparallel
- asciicheck
- asasalint
- varnamelen
- reassign
- nilnil
- nilerr
- nakedret
- goprintffuncname
- typecheck
- errchkjson
- errcheck - errcheck
- gosimple - gosimple
- govet - govet
- ineffassign - ineffassign
- staticcheck - staticcheck
- structcheck
- unused - unused
- unparam - unparam
- varcheck
- bodyclose
- dogsled - dogsled
- dupl - dupl
- errorlint - errorlint
@ -32,20 +45,15 @@ linters:
- gocyclo - gocyclo
- godot - godot
- goimports - goimports
- golint - revive
- gomnd
- gosec - gosec
- ifshort
- interfacer
- lll - lll
- makezero - makezero
- maligned
- misspell - misspell
- nestif - nestif
- prealloc - prealloc
- predeclared - predeclared
- scopelint - exportloopref
- sqlclosecheck
- unconvert - unconvert
- whitespace - whitespace
@ -56,9 +64,11 @@ linters-settings:
enable: enable:
- assign - assign
- atomic - atomic
- atomicalign
- bools - bools
- buildtag - buildtag
- copylocks - copylocks
- fieldalignment
- httpresponse - httpresponse
- loopclosure - loopclosure
- lostcancel - lostcancel
@ -70,7 +80,6 @@ linters-settings:
- unmarshal - unmarshal
- unreachable - unreachable
- unsafeptr - unsafeptr
- unused
settings: settings:
printf: printf:
funcs: funcs:
@ -129,11 +138,13 @@ linters-settings:
threshold: 200 threshold: 200
errorlint: errorlint:
errorf: true errorf: true
asserts: false
comparison: false
exhaustive: exhaustive:
check-generated: false check-generated: false
default-signifies-exhaustive: false default-signifies-exhaustive: false
funlen: funlen:
lines: 60 lines: 90
statements: 40 statements: 40
gocognit: gocognit:
min-complexity: 25 min-complexity: 25
@ -143,8 +154,6 @@ linters-settings:
local-prefixes: github.com/retailcrm/messenger local-prefixes: github.com/retailcrm/messenger
lll: lll:
line-length: 120 line-length: 120
maligned:
suggest-new: true
misspell: misspell:
locale: US locale: US
nestif: nestif:
@ -152,28 +161,36 @@ linters-settings:
whitespace: whitespace:
multi-if: false multi-if: false
multi-func: false multi-func: false
varnamelen:
max-distance: 10
ignore-map-index-ok: true
ignore-type-assert-ok: true
ignore-chan-recv-ok: true
ignore-decls:
- t *testing.T
- e error
- i int
issues: issues:
exclude-rules: exclude-rules:
- path: _test\.go - path: _test\.go
linters: linters:
- gomnd
- lll - lll
- bodyclose
- errcheck - errcheck
- sqlclosecheck
- misspell - misspell
- ineffassign - ineffassign
- whitespace - whitespace
- makezero - makezero
- maligned
- ifshort
- errcheck - errcheck
- funlen - funlen
- goconst - goconst
- gocognit - gocognit
- gocyclo - gocyclo
- godot - godot
- unused
- errchkjson
- varnamelen
- path: \.go
text: "Error return value of `io.WriteString` is not checked"
exclude-use-default: true exclude-use-default: true
exclude-case-sensitive: false exclude-case-sensitive: false
max-issues-per-linter: 0 max-issues-per-linter: 0
@ -185,4 +202,4 @@ severity:
case-sensitive: false case-sensitive: false
service: service:
golangci-lint-version: 1.36.x golangci-lint-version: 1.50.x

View file

@ -24,3 +24,20 @@ const (
// status. // status.
AccountLinkingAction AccountLinkingAction
) )
// SenderAction is used to send a specific action (event) to the Facebook.
// The result of sending said action is supposed to give more interactivity to the bot.
type SenderAction string
const (
// MarkSeen marks message as seen.
MarkSeen SenderAction = "MARK_SEEN"
// TypingOn turns on "Bot is typing..." indicator.
TypingOn SenderAction = "TYPING_ON"
// TypingOff turns off typing indicator.
TypingOff SenderAction = "TYPING_OFF"
// React to the message.
React SenderAction = "REACT"
// Unreact to the message (remove reaction).
Unreact SenderAction = "UNREACT"
)

7
go.mod
View file

@ -1,8 +1,11 @@
module github.com/retailcrm/messenger module github.com/retailcrm/messenger
require (
github.com/stretchr/testify v1.2.2
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7
)
require ( require (
github.com/davecgh/go-spew v1.1.1 // indirect github.com/davecgh/go-spew v1.1.1 // indirect
github.com/pmezard/go-difflib v1.0.0 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect
github.com/stretchr/testify v1.2.2
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7
) )

View file

@ -66,6 +66,8 @@ type Read struct {
RawWatermark int64 `json:"watermark"` RawWatermark int64 `json:"watermark"`
// Seq is the sequence the message was sent in. // Seq is the sequence the message was sent in.
Seq int `json:"seq"` Seq int `json:"seq"`
// Mid is the ID of the message.
Mid string `json:"mid"`
} }
// IGMessageRead represents data with the read message ID. Present in the Instagram webhook. // IGMessageRead represents data with the read message ID. Present in the Instagram webhook.
@ -79,7 +81,7 @@ type IGMessageReaction struct {
// Mid is a message ID. // Mid is a message ID.
Mid string `json:"mid"` Mid string `json:"mid"`
// Action can be {react|unreact} // Action can be {react|unreact}
Action string `json:"action"` Action ReactionAction `json:"action"`
// Reaction is a reaction name. Optional. // Reaction is a reaction name. Optional.
Reaction string `json:"reaction,omitempty"` Reaction string `json:"reaction,omitempty"`
// Emoji is optional. // Emoji is optional.
@ -92,8 +94,10 @@ type IGMessageProduct struct {
ID string `json:"id,omitempty"` ID string `json:"id,omitempty"`
} }
// IGMessageReferral represents Instagram message referral with product ID. // IGMessageReferral represents Instagram message referral with ad data and product ID.
type IGMessageReferral struct { type IGMessageReferral struct {
// Ad data
Referral
// Product data. // Product data.
Product IGMessageProduct `json:"product,omitempty"` Product IGMessageProduct `json:"product,omitempty"`
} }

View file

@ -198,7 +198,7 @@ func (m *Messenger) ProfileByID(id int64, profileFields []string) (Profile, erro
err = json.Unmarshal(content, &p) err = json.Unmarshal(content, &p)
if err != nil { if err != nil {
return p, err return p, NewUnmarshalError(err).WithContent(content)
} }
if p == *new(Profile) { if p == *new(Profile) {
@ -508,6 +508,29 @@ func (m *Messenger) EnableChatExtension(homeURL HomeURL) error {
return checkFacebookError(resp.Body) return checkFacebookError(resp.Body)
} }
func (m *Messenger) SenderAction(to Recipient, action SenderAction) (QueryResponse, error) {
response := &Response{
token: m.token,
to: to,
sendAPIVersion: m.sendAPIVersion,
}
return response.SenderAction(action)
}
func (m *Messenger) InstagramReaction(
to Recipient,
mid string,
action ReactionAction,
reaction ...string,
) (QueryResponse, error) {
response := &Response{
token: m.token,
to: to,
sendAPIVersion: m.sendAPIVersion,
}
return response.InstagramReaction(mid, action, reaction...)
}
// classify determines what type of message a webhook event is. // classify determines what type of message a webhook event is.
func (m *Messenger) classify(info MessageInfo) Action { func (m *Messenger) classify(info MessageInfo) Action {
if info.Message != nil { if info.Message != nil {

View file

@ -35,6 +35,8 @@ type MessageInfo struct {
// Delivery is the contents of a message if it is a DeliveryAction. // Delivery is the contents of a message if it is a DeliveryAction.
// Nil if it is not a DeliveryAction. // Nil if it is not a DeliveryAction.
Delivery *Delivery `json:"delivery"` Delivery *Delivery `json:"delivery"`
// Reaction represents reaction to Instagram message.
Reaction *IGMessageReaction `json:"reaction,omitempty"`
PostBack *PostBack `json:"postback"` PostBack *PostBack `json:"postback"`
@ -78,6 +80,26 @@ type Referral struct {
Source string `json:"source"` Source string `json:"source"`
// The identifier dor the referral // The identifier dor the referral
Type string `json:"type"` Type string `json:"type"`
// ID of the ad
AdID string `json:"ad_id,omitempty"`
// The data containing information about the CTM ad, the user initiated the thread from.
AdsContextData AdsContextData `json:"ads_context_data,omitempty"`
// URI of the site from which the message was sent to the Facebook chat plugin.
RefererURI string `json:"referer_uri,omitempty"`
}
// AdsContextData represents data containing information about the CTM ad, the user initiated the thread from.
type AdsContextData struct {
// Title of the Ad
AdTitle string `json:"ad_title"`
// Url of the image from the Ad the user is interested
PhotoURL string `json:"photo_url,omitempty"`
// Thumbnail url of the video from the ad
VideoURL string `json:"video_url,omitempty"`
// ID of the post
PostID string `json:"post_id"`
// Product ID from the Ad the user is interested
ProductID string `json:"product_id,omitempty"`
} }
// Sender is who the message was sent from. // Sender is who the message was sent from.

View file

@ -87,9 +87,10 @@ func checkFacebookError(r io.Reader) error {
var err error var err error
qr := QueryResponse{} qr := QueryResponse{}
err = json.NewDecoder(r).Decode(&qr) decoder := json.NewDecoder(r)
err = decoder.Decode(&qr)
if err != nil { if err != nil {
return xerrors.Errorf("json unmarshal error: %w", err) return NewUnmarshalError(err).WithReader(decoder.Buffered())
} }
if qr.Error != nil { if qr.Error != nil {
return xerrors.Errorf("facebook error: %w", qr.Error) return xerrors.Errorf("facebook error: %w", qr.Error)
@ -100,9 +101,9 @@ func checkFacebookError(r io.Reader) error {
func getFacebookQueryResponse(r io.Reader) (QueryResponse, error) { func getFacebookQueryResponse(r io.Reader) (QueryResponse, error) {
qr := QueryResponse{} qr := QueryResponse{}
err := json.NewDecoder(r).Decode(&qr) decoder := json.NewDecoder(r)
if err != nil { if err := decoder.Decode(&qr); err != nil {
return qr, xerrors.Errorf("json unmarshal error: %w", err) return qr, NewUnmarshalError(err).WithReader(decoder.Buffered())
} }
if qr.Error != nil { if qr.Error != nil {
return qr, xerrors.Errorf("facebook error: %w", qr.Error) return qr, xerrors.Errorf("facebook error: %w", qr.Error)
@ -181,7 +182,7 @@ func (r *Response) Image(im image.Image) (QueryResponse, error) {
return qr, err return qr, err
} }
return r.AttachmentData(ImageAttachment, "meme.jpg", imageBytes) return r.AttachmentData(ImageAttachment, "meme.jpg", "image/jpeg", imageBytes)
} }
// Attachment sends an image, sound, video or a regular file to a chat. // Attachment sends an image, sound, video or a regular file to a chat.
@ -227,15 +228,14 @@ func createFormFile(filename string, w *multipart.Writer, contentType string) (i
} }
// AttachmentData sends an image, sound, video or a regular file to a chat via an io.Reader. // 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) { func (r *Response) AttachmentData(
dataType AttachmentType, filename string, contentType string, filedata io.Reader) (QueryResponse, error) {
var qr QueryResponse var qr QueryResponse
filedataBytes, err := ioutil.ReadAll(filedata) filedataBytes, err := ioutil.ReadAll(filedata)
if err != nil { if err != nil {
return qr, err return qr, err
} }
contentType := http.DetectContentType(filedataBytes[:512])
fmt.Println("Content-type detected:", contentType)
var body bytes.Buffer var body bytes.Buffer
multipartWriter := multipart.NewWriter(&body) multipartWriter := multipart.NewWriter(&body)
@ -351,8 +351,8 @@ func (r *Response) ListTemplate(elements *[]StructuredMessageElement, messagingT
return r.DispatchMessage(&m) return r.DispatchMessage(&m)
} }
// SenderAction sends a info about sender action. // SenderAction sends an info about sender action.
func (r *Response) SenderAction(action string) (QueryResponse, error) { func (r *Response) SenderAction(action SenderAction) (QueryResponse, error) {
m := SendSenderAction{ m := SendSenderAction{
Recipient: r.to, Recipient: r.to,
SenderAction: action, SenderAction: action,
@ -360,6 +360,21 @@ func (r *Response) SenderAction(action string) (QueryResponse, error) {
return r.DispatchMessage(&m) return r.DispatchMessage(&m)
} }
// InstagramReaction sends an info about Instagram reaction.
func (r *Response) InstagramReaction(mid string, action ReactionAction, reaction ...string) (QueryResponse, error) {
m := SendInstagramReaction{
Recipient: r.to,
SenderAction: action,
Payload: SenderInstagramReactionPayload{
MessageID: mid,
},
}
if len(reaction) > 0 {
m.Payload.Reaction = reaction[0]
}
return r.DispatchMessage(&m)
}
// DispatchMessage posts the message to messenger, return the error if there's any. // DispatchMessage posts the message to messenger, return the error if there's any.
func (r *Response) DispatchMessage(m interface{}) (QueryResponse, error) { func (r *Response) DispatchMessage(m interface{}) (QueryResponse, error) {
var res QueryResponse var res QueryResponse
@ -546,6 +561,29 @@ type StructuredMessageButton struct {
// SendSenderAction is the information about sender action. // SendSenderAction is the information about sender action.
type SendSenderAction struct { type SendSenderAction struct {
Recipient Recipient `json:"recipient"` Recipient Recipient `json:"recipient"`
SenderAction string `json:"sender_action"` SenderAction SenderAction `json:"sender_action"`
}
// ReactionAction contains info about reaction action type.
type ReactionAction string
const (
// ReactionActionReact is used when user added a reaction.
ReactionActionReact ReactionAction = "react"
// ReactionActionUnReact is used when user removed a reaction.
ReactionActionUnReact ReactionAction = "unreact"
)
// SendInstagramReaction is the information about sender action.
type SendInstagramReaction struct {
Recipient Recipient `json:"recipient"`
SenderAction ReactionAction `json:"sender_action"`
Payload SenderInstagramReactionPayload `json:"payload"`
}
// SenderInstagramReactionPayload contains target message ID and reaction name.
type SenderInstagramReactionPayload struct {
MessageID string `json:"message_id"`
Reaction string `json:"reaction"`
} }

View file

@ -1,7 +1,9 @@
package messenger package messenger
import ( import (
"bytes"
"encoding/json" "encoding/json"
"errors"
"testing" "testing"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
@ -15,3 +17,17 @@ func Test_MarshalStructuredMessageElement(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
assert.JSONEq(t, string(data), `{"image_url":"", "subtitle":"", "title": "Title"}`) assert.JSONEq(t, string(data), `{"image_url":"", "subtitle":"", "title": "Title"}`)
} }
func TestResponse_checkFacebookError_UnmarshalError(t *testing.T) {
r := bytes.NewReader([]byte("test error text"))
err := checkFacebookError(r)
assert.True(t, errors.Is(err, ErrUnmarshal))
assert.Contains(t, err.Error(), "test error text")
}
func TestResponse_getFacebookQueryResponse_UnmarshalError(t *testing.T) {
r := bytes.NewReader([]byte("test error text"))
_, err := getFacebookQueryResponse(r)
assert.True(t, errors.Is(err, ErrUnmarshal))
assert.Contains(t, err.Error(), "test error text")
}

47
unmarshal_error.go Normal file
View file

@ -0,0 +1,47 @@
package messenger
import (
"errors"
"fmt"
"io"
"io/ioutil"
)
var ErrUnmarshal = errors.New("unmarshal error")
type UnmarshalError struct {
Content []byte
ErrorText string
Err error
}
func (u *UnmarshalError) Error() string {
return fmt.Sprintf("can not unmarshal content: %s; error: %s", string(u.Content), u.ErrorText)
}
func (u *UnmarshalError) Unwrap() error {
return u.Err
}
func NewUnmarshalError(err error) *UnmarshalError {
return &UnmarshalError{
Err: ErrUnmarshal,
ErrorText: err.Error(),
}
}
func (u *UnmarshalError) WithReader(reader io.Reader) *UnmarshalError {
content, _ := ioutil.ReadAll(reader)
u.Content = content
return u
}
func (u *UnmarshalError) WithContent(content []byte) *UnmarshalError {
u.Content = content
return u
}
func (u *UnmarshalError) WithErr(err error) *UnmarshalError {
u.Err = err
return u
}

57
unmarshal_error_test.go Normal file
View file

@ -0,0 +1,57 @@
package messenger
import (
"bytes"
"errors"
"testing"
"github.com/stretchr/testify/assert"
)
func TestNewUnmarshalError(t *testing.T) {
err := errors.New("some error")
unmarshalError := NewUnmarshalError(err)
assert.True(t, errors.Is(unmarshalError, ErrUnmarshal))
}
func TestUnmarshalError_Error(t *testing.T) {
err := errors.New("some error")
content := []byte("test content")
actual := NewUnmarshalError(err).WithContent(content).Error()
expected := "can not unmarshal content: test content; error: some error"
assert.Equal(t, expected, actual)
}
func TestUnmarshalError_Unwrap(t *testing.T) {
err := errors.New("some error")
actual := NewUnmarshalError(err).Unwrap()
expected := ErrUnmarshal
assert.Equal(t, expected, actual)
}
func TestUnmarshalError_WithContent(t *testing.T) {
err := errors.New("some error")
content := []byte("test content")
actual := NewUnmarshalError(err).WithContent(content)
expected := &UnmarshalError{Err: ErrUnmarshal, Content: content, ErrorText: err.Error()}
assert.Equal(t, expected, actual)
}
func TestUnmarshalError_WithReader(t *testing.T) {
err := errors.New("some error")
content := []byte("test content")
reader := bytes.NewReader(content)
actual := NewUnmarshalError(err).WithReader(reader)
expected := &UnmarshalError{Err: ErrUnmarshal, Content: content, ErrorText: err.Error()}
assert.Equal(t, expected, actual)
}
func TestUnmarshalError_WithErr(t *testing.T) {
someError := errors.New("some error")
otherError := errors.New("other error")
actual := NewUnmarshalError(someError).WithErr(otherError)
expected := &UnmarshalError{Err: otherError, ErrorText: someError.Error()}
assert.Equal(t, expected, actual)
}