1
0
Fork 0
mirror of synced 2025-04-04 21:53:37 +03:00
This commit is contained in:
Pavel 2025-03-21 17:07:11 +03:00 committed by GitHub
commit 11f57aab2d
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
9 changed files with 122 additions and 81 deletions

View file

@ -18,32 +18,31 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Check out code into the Go module directory
uses: actions/checkout@v2
- name: Set up Go 1.17
uses: actions/setup-go@v2
uses: actions/checkout@v4
- name: Set up Go 1.24
uses: actions/setup-go@v5
with:
# TODO: Should migrate to 1.18 later
go-version: '1.17'
go-version: '1.24'
- name: Get dependencies
run: go mod tidy
- name: Lint code with golangci-lint
uses: golangci/golangci-lint-action@v3
with:
version: v1.50.1
version: v1.62.2
only-new-issues: true
tests:
name: Tests
runs-on: ubuntu-latest
strategy:
matrix:
go-version: ['1.13', '1.14', '1.15', '1.16', '1.17']
go-version: ['1.16', '1.17', '1.18', '1.19', '1.20', '1.21', '1.22', '1.23', '1.24']
steps:
- name: Set up Go ${{ matrix.go-version }}
uses: actions/setup-go@v2
uses: actions/setup-go@v5
with:
go-version: ${{ matrix.go-version }}
- name: Check out code into the Go module directory
uses: actions/checkout@v2
uses: actions/checkout@v4
- name: Get dependencies
run: go mod tidy
- name: Tests

View file

@ -1,31 +1,16 @@
run:
skip-dirs-use-default: true
allow-parallel-runners: true
modules-download-mode: readonly
output:
format: colored-line-number
formats:
- format: colored-line-number
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:
disable-all: true
enable:
- paralleltest
- tparallel
- asciicheck
- asasalint
- varnamelen
- reassign
- nilnil
- nilerr
- nakedret
- goprintffuncname
- typecheck
- errchkjson
- errcheck
- gosimple
- govet
@ -33,11 +18,12 @@ linters:
- staticcheck
- unused
- unparam
- bodyclose
- dogsled
- dupl
- errorlint
- exhaustive
- exportloopref
- copyloopvar
- funlen
- gocognit
- goconst
@ -46,16 +32,18 @@ linters:
- godot
- goimports
- revive
- mnd
- gosec
- lll
- makezero
- misspell
- nestif
- prealloc
- predeclared
- exportloopref
- sqlclosecheck
- unconvert
- whitespace
- unused
- testifylint
linters-settings:
govet:
@ -64,11 +52,9 @@ linters-settings:
enable:
- assign
- atomic
- atomicalign
- bools
- buildtag
- copylocks
- fieldalignment
- httpresponse
- loopclosure
- lostcancel
@ -80,6 +66,7 @@ linters-settings:
- unmarshal
- unreachable
- unsafeptr
- unused
settings:
printf:
funcs:
@ -138,20 +125,18 @@ linters-settings:
threshold: 200
errorlint:
errorf: true
asserts: false
comparison: false
exhaustive:
check-generated: false
default-signifies-exhaustive: false
funlen:
lines: 90
statements: 40
lines: 65
statements: 50
gocognit:
min-complexity: 25
gocyclo:
min-complexity: 25
goimports:
local-prefixes: github.com/retailcrm/messenger
local-prefixes: github.com/retailcrm/mg-transport-api-client-go
lll:
line-length: 120
misspell:
@ -161,25 +146,23 @@ linters-settings:
whitespace:
multi-if: 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:
exclude-rules:
- path: _test\.go
linters:
- dupl
- mnd
- lll
- bodyclose
- errcheck
- sqlclosecheck
- misspell
- ineffassign
- whitespace
- makezero
- maligned
- ifshort
- errcheck
- funlen
- goconst
@ -187,10 +170,6 @@ issues:
- gocyclo
- godot
- unused
- errchkjson
- varnamelen
- path: \.go
text: "Error return value of `io.WriteString` is not checked"
exclude-use-default: true
exclude-case-sensitive: false
max-issues-per-linter: 0
@ -202,4 +181,4 @@ severity:
case-sensitive: false
service:
golangci-lint-version: 1.50.x
golangci-lint-version: 1.62.x

View file

@ -48,7 +48,8 @@ func main() {
fmt.Println("Something went wrong!", err)
}
r.Text(fmt.Sprintf("Hello, %v!", p.FirstName), messenger.ResponseType, "")
r.Text(fmt.Sprintf("Hello, %v!", p.FirstName),
messenger.ResponseType, "", messenger.NotificationRegularType)
})
// Setup a handler to be triggered when a message is delivered

View file

@ -57,7 +57,8 @@ func main() {
fmt.Println("Something went wrong!", err)
}
r.Text(fmt.Sprintf("Hello, %v!", p.FirstName), messenger.ResponseType, "")
r.Text(fmt.Sprintf("Hello, %v!", p.FirstName), messenger.ResponseType,
"", messenger.NotificationRegularType)
})
addr := fmt.Sprintf("%s:%d", *host, *port)

View file

@ -84,7 +84,8 @@ func main() {
text = "You've been logged out of your account."
}
if _, err := r.Text(text, messenger.ResponseType, ""); err != nil {
if _, err := r.Text(text, messenger.ResponseType,
"", messenger.NotificationRegularType); err != nil {
log.Println("Failed to send account linking feedback")
}
})
@ -134,7 +135,8 @@ func logoutButton(r *messenger.Response) error {
// greeting salutes the user.
func greeting(p messenger.Profile, r *messenger.Response) error {
_, err := r.Text(fmt.Sprintf("Hello, %v!", p.FirstName), messenger.ResponseType, "")
_, err := r.Text(fmt.Sprintf("Hello, %v!", p.FirstName), messenger.ResponseType,
"", messenger.NotificationRegularType)
return err
}
@ -156,7 +158,7 @@ func help(p messenger.Profile, r *messenger.Response) error {
},
}
_, err := r.TextWithReplies(text, replies, messenger.ResponseType, "")
_, err := r.TextWithReplies(text, replies, messenger.ResponseType, "", messenger.NotificationRegularType)
return err
}

2
go.mod
View file

@ -5,6 +5,8 @@ require (
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7
)
go 1.16
require (
github.com/davecgh/go-spew v1.1.1 // indirect
github.com/pmezard/go-difflib v1.0.0 // indirect

View file

@ -15,20 +15,20 @@ import (
)
const (
// ProfileURL is the API endpoint used for retrieving profiles.
// Used in the form: https://graph.facebook.com/v2.6/<USER_ID>?fields=<PROFILE_FIELDS>&access_token=<PAGE_ACCESS_TOKEN>
ProfileURL = "https://graph.facebook.com/v2.6/"
// ProfileFields is a list of JSON field names which will be populated by the profile query.
ProfileFields = "first_name,last_name,profile_pic"
// ProfileURL is the API endpoint used for retrieving profiles.
// Used in the form: https://graph.facebook.com/v14.0/<USER_ID>?fields=<PROFILE_FIELDS>&access_token=<PAGE_ACCESS_TOKEN>
ProfileURL = "https://graph.facebook.com/v14.0/"
// SendSettingsURL is API endpoint for saving settings.
SendSettingsURL = "https://graph.facebook.com/v2.6/me/thread_settings"
SendSettingsURL = "https://graph.facebook.com/v11.0/me/thread_settings"
// MessengerProfileURL is the API endpoint where you set properties that define various aspects of the following Messenger Platform features.
// Used in the form https://graph.facebook.com/v2.6/me/messenger_profile?access_token=<PAGE_ACCESS_TOKEN>
// https://developers.facebook.com/docs/messenger-platform/reference/messenger-profile-api/
MessengerProfileURL = "https://graph.facebook.com/v2.6/me/messenger_profile"
MessengerProfileURL = "https://graph.facebook.com/v11.0/me/messenger_profile"
)
// Options are the settings used when creating a Messenger client.
@ -303,8 +303,8 @@ func (m *Messenger) handle(w http.ResponseWriter, r *http.Request) {
return
}
if rec.Object != "page" {
fmt.Println("Object is not page, undefined behavior. Got", rec.Object)
if rec.Object != "page" && rec.Object != "instagram" {
fmt.Println("Object is not page or instagram, undefined behavior. Got", rec.Object)
respond(w, http.StatusUnprocessableEntity)
return
}
@ -443,7 +443,7 @@ func (m *Messenger) Response(to int64) *Response {
}
// Send will send a textual message to a user. This user must have previously initiated a conversation with the bot.
func (m *Messenger) Send(to Recipient, message string, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
func (m *Messenger) Send(to Recipient, message string, messagingType MessagingType, metadata string, tags ...TagType) (QueryResponse, error) {
return m.SendWithReplies(to, message, nil, messagingType, metadata, tags...)
}
@ -458,14 +458,14 @@ func (m *Messenger) SendGeneralMessage(to Recipient, elements *[]StructuredMessa
}
// SendWithReplies sends a textual message to a user, but gives them the option of numerous quick response options.
func (m *Messenger) SendWithReplies(to Recipient, message string, replies []QuickReply, messagingType MessagingType, metadata string, tags ...string) (QueryResponse, error) {
func (m *Messenger) SendWithReplies(to Recipient, message string, replies []QuickReply, messagingType MessagingType, metadata string, tags ...TagType) (QueryResponse, error) {
response := &Response{
token: m.token,
to: to,
sendAPIVersion: m.sendAPIVersion,
}
return response.TextWithReplies(message, replies, messagingType, metadata, tags...)
return response.TextWithReplies(message, replies, messagingType, metadata, NotificationRegularType, tags...)
}
// Attachment sends an image, sound, video or a regular file to a given recipient.

View file

@ -9,4 +9,13 @@ type Profile struct {
Locale string `json:"locale"`
Timezone float64 `json:"timezone"`
Gender string `json:"gender"`
// instagram user profile
Username string `json:"username,omitempty"`
IsPrivate bool `json:"is_private,omitempty"`
// FollowCount int32 `json:"follow_count,omitempty"`
FollowedByCount int32 `json:"follower_count,omitempty"` // by the documentation followed_by_count
IsVerifiedUser bool `json:"is_verified_user"`
IsUserFollowBusiness bool `json:"is_user_follow_business"`
IsBusinessFollowUser bool `json:"is_business_follow_user"`
}

View file

@ -19,12 +19,14 @@ import (
// AttachmentType is attachment type.
type AttachmentType string
type MessagingType string
type NotificationType string
type TopElementStyle string
type ImageAspectRatio string
type TagType string
const (
// DefaultSendAPIVersion is a default Send API version
DefaultSendAPIVersion = "v2.11"
DefaultSendAPIVersion = "v11.0"
// SendMessageURL is API endpoint for sending messages.
SendMessageURL = "https://graph.facebook.com/%s/me/messages"
// ThreadControlURL is the API endpoint for passing thread control.
@ -50,6 +52,13 @@ const (
// NonPromotionalSubscriptionType is NON_PROMOTIONAL_SUBSCRIPTION messaging type.
NonPromotionalSubscriptionType MessagingType = "NON_PROMOTIONAL_SUBSCRIPTION"
// NotificationNoPushType is NO_PUSH notification type. No notification.
NotificationNoPushType NotificationType = "NO_PUSH"
// NotificationRegularType is REGULAR notification type (default). Sound or vibration when a message is received by a person.
NotificationRegularType NotificationType = "REGULAR"
// NotificationSilentPushType is SILENT_PUSH notification type. On-screen notification only.
NotificationSilentPushType NotificationType = "SILENT_PUSH"
// TopElementStyle is compact.
CompactTopElementStyle TopElementStyle = "compact"
// TopElementStyle is large.
@ -59,6 +68,21 @@ const (
HorizontalImageAspectRatio ImageAspectRatio = "horizontal"
// ImageAspectRatio is square.
SquareImageAspectRatio ImageAspectRatio = "square"
SenderActionMarkSeen SenderAction = "mark_seen"
SenderActionTypingOn SenderAction = "typing_on"
SenderActionTypingOff SenderAction = "typing_off"
// TagAccountUpdateType Tags the message you are sending to your customer as a non-recurring update to their application or account. Not available for Instagram Messaging API.
TagAccountUpdateType TagType = "ACCOUNT_UPDATE"
// TagConfirmedEventUpdateType Tags the message you are sending to your customer as a reminder fo an upcoming event or an update for an event in procgres for which the customer is registered. Not available for Instagram Messaging API.
TagConfirmedEventUpdateType TagType = "CONFIRMED_EVENT_UPDATE"
// TagCustomerFeedbackType Tags the message you are sending to your customer as a Customer Feedback Survey. Customer feedback messages must be sent within 7 days of the customer's last message. Not available for Instagram Messaging API.
TagCustomerFeedbackType TagType = "CUSTOMER_FEEDBACK"
// TagHumanAgentType When this tag is added to a message to a customer, it allows a human agent to respond to a person's message. Messages can be sent within 7 days of the person's. Human agent support is for issues that cannot be resolved within the standard 24 hour messaging window.
TagHumanAgentType TagType = "HUMAN_AGENT"
// TagPostPurchaseUpdateType Tags the message you are sending to your customer as an update for a recent purchase made by the customer. Not available for Instagram Messaging API.
TagPostPurchaseUpdateType TagType = "POST_PURCHASE_UPDATE"
)
// QueryResponse is the response sent back by Facebook when setting up things
@ -123,17 +147,31 @@ func (r *Response) SetToken(token string) {
r.token = token
}
// 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...)
func (r *Response) Text(
message string,
messagingType MessagingType,
metadata string,
notificationType NotificationType,
tags ...TagType,
) (QueryResponse, error) {
return r.TextWithReplies(message, nil, messagingType, metadata, notificationType, 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
// notificationType should be one of the following: "NO_PUSH","REGULAR" (default),"SILENT_PUSH"
// 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,
notificationType NotificationType,
tags ...TagType,
) (QueryResponse, error) {
var tag TagType
if len(tags) > 0 {
tag = tags[0]
}
@ -147,14 +185,22 @@ func (r *Response) TextWithReplies(message string, replies []QuickReply, messagi
QuickReplies: replies,
Metadata: metadata,
},
Tag: tag,
Tag: tag,
NotificationType: notificationType,
}
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
func (r *Response) AttachmentWithReplies(
attachment *StructuredMessageAttachment,
replies []QuickReply,
messagingType MessagingType,
metadata string,
notificationType NotificationType,
tags ...TagType,
) (QueryResponse, error) {
var tag TagType
if len(tags) > 0 {
tag = tags[0]
}
@ -167,7 +213,8 @@ func (r *Response) AttachmentWithReplies(attachment *StructuredMessageAttachment
QuickReplies: replies,
Metadata: metadata,
},
Tag: tag,
Tag: tag,
NotificationType: notificationType,
}
return r.DispatchMessage(&m)
}
@ -434,10 +481,11 @@ func (r *Response) PassThreadToInbox() error {
// SendMessage is the information sent in an API request to Facebook.
type SendMessage struct {
MessagingType MessagingType `json:"messaging_type"`
Recipient Recipient `json:"recipient"`
Message MessageData `json:"message"`
Tag string `json:"tag,omitempty"`
MessagingType MessagingType `json:"messaging_type"`
Recipient Recipient `json:"recipient"`
Message MessageData `json:"message"`
Tag TagType `json:"tag,omitempty"`
NotificationType NotificationType `json:"notification_type,omitempty"`
}
// MessageData is a message consisting of text or an attachment, with an additional selection of optional quick replies.