Skip to content

Commit

Permalink
Merge ab971b4 into 21c6026
Browse files Browse the repository at this point in the history
  • Loading branch information
norkans7 committed Aug 4, 2017
2 parents 21c6026 + ab971b4 commit fca18bc
Show file tree
Hide file tree
Showing 3 changed files with 405 additions and 1 deletion.
10 changes: 10 additions & 0 deletions handlers/base.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package handlers

import (
"bytes"
"encoding/base64"
"encoding/json"
"fmt"
Expand Down Expand Up @@ -189,3 +190,12 @@ func DecodePossibleBase64(original string) string {

return decoded
}

// EncodeBase64 encodes the list of strings with Linux base64
func EncodeBase64(strList []string) string {
var strBuffer bytes.Buffer
for _, l := range strList {
strBuffer.WriteString(l)
}
return base64.StdEncoding.EncodeToString([]byte(strBuffer.String()))
}
231 changes: 230 additions & 1 deletion handlers/zenvia/zenvia.go
Original file line number Diff line number Diff line change
@@ -1,3 +1,232 @@
package zenvia

/* no logs */
import (
"bytes"
"encoding/json"
"fmt"
"net/http"
"strconv"
"strings"
"time"

"github.com/buger/jsonparser"
"github.com/nyaruka/courier"
"github.com/nyaruka/courier/handlers"
"github.com/nyaruka/courier/utils"
"github.com/pkg/errors"
)

var sendURL = "https://api-rest.zenvia360.com.br/services"

func init() {
courier.RegisterHandler(NewHandler())
}

type handler struct {
handlers.BaseHandler
}

// NewHandler returns a new Zenvia handler
func NewHandler() courier.ChannelHandler {
return &handler{handlers.NewBaseHandler(courier.ChannelType("ZV"), "Zenvia")}
}

// {
// "callbackMoRequest": {
// "id": "20690090",
// "mobile": "555191951711",
// "shortCode": "40001",
// "account": "zenvia.envio",
// "body": "Content of reply SMS",
// "received": "2014-08-26T12:27:08.488-03:00",
// "correlatedMessageSmsId": "hs765939061"
// }
// }
type messageRequest struct {
CallbackMoRequest struct {
ID string `validate:"required" json:"id"`
From string `validate:"required" json:"mobile"`
Text string `validate:"required" json:"body"`
Date string `validate:"required" json:"received"`
ExternalID string `validate:"required" json:"correlatedMessageSmsId"`
}
}

// {
// "callbackMtRequest": {
// "status": "03",
// "statusMessage": "Delivered",
// "statusDetail": "120",
// "statusDetailMessage": "Message received by mobile",
// "id": "hs765939216",
// "received": "2014-08-26T12:55:48.593-03:00",
// "mobileOperatorName": "Claro"
// }
// }
type statusRequest struct {
CallbackMtRequest struct {
StatusCode string `validate:"required" json:"status"`
ID string `validate:"required" json:"id"`
}
}

// {
// "sendSmsRequest": {
// "from": "Sender",
// "to": "555199999999",
// "schedule": "2014-08-22T14:55:00",
// "msg": "Test message.",
// "callbackOption": "NONE",
// "id": "002",
// "aggregateId": "1111"
// }
// }
type zvOutgoingMsg struct {
SendSmsRequest zvSendSmsRequest `json:"sendSmsRequest"`
}

type zvSendSmsRequest struct {
From string `validate:"required" json:"from"`
To string `validate:"required" json:"to"`
Schedule string `validate:"required" json:"schedule"`
Msg string `validate:"required" json:"msg"`
CallbackOption string `validate:"required" json:"callbackOption"`
ID string `validate:"required" json:"id"`
AggregateID string `validate:"required" json:"aggregateId"`
}

var statusMapping = map[string]courier.MsgStatusValue{
"00": courier.MsgSent,
"01": courier.MsgSent,
"02": courier.MsgSent,
"03": courier.MsgDelivered,
"04": courier.MsgErrored,
"05": courier.MsgErrored,
"06": courier.MsgErrored,
"07": courier.MsgErrored,
"08": courier.MsgErrored,
"09": courier.MsgErrored,
"10": courier.MsgErrored,
}

// Initialize is called by the engine once everything is loaded
func (h *handler) Initialize(s courier.Server) error {
h.SetServer(s)
err := s.AddReceiveMsgRoute(h, "POST", "receive", h.ReceiveMessage)
if err != nil {
return err
}
return s.AddUpdateStatusRoute(h, "POST", "status", h.StatusMessage)
}

// ReceiveMessage is our HTTP handler function for incoming messages
func (h *handler) ReceiveMessage(channel courier.Channel, w http.ResponseWriter, r *http.Request) ([]courier.Msg, error) {
// get our params
zvMsg := &messageRequest{}
err := handlers.DecodeAndValidateJSON(zvMsg, r)
if err != nil {
return nil, err
}

// create our date from the timestamp
// 2017-05-03T06:04:45.345-03:00
date, err := time.Parse("2006-01-02T15:04:05.000-07:00", zvMsg.CallbackMoRequest.Date)
if err != nil {
return nil, fmt.Errorf("invalid date format: %s", zvMsg.CallbackMoRequest.Date)
}

// create our URN
urn := courier.NewTelURNForChannel(zvMsg.CallbackMoRequest.From, channel)

// build our msg
msg := h.Backend().NewIncomingMsg(channel, urn, zvMsg.CallbackMoRequest.Text).WithExternalID(zvMsg.CallbackMoRequest.ExternalID).WithReceivedOn(date.UTC())

// and finally queue our message
err = h.Backend().WriteMsg(msg)
if err != nil {
return nil, err
}

return []courier.Msg{msg}, courier.WriteReceiveSuccess(w, r, msg)
}

// StatusMessage is our HTTP handler function for status updates
func (h *handler) StatusMessage(channel courier.Channel, w http.ResponseWriter, r *http.Request) ([]courier.MsgStatus, error) {
// get our params
zvStatus := &statusRequest{}
err := handlers.DecodeAndValidateJSON(zvStatus, r)
if err != nil {
return nil, err
}

msgStatus, found := statusMapping[zvStatus.CallbackMtRequest.StatusCode]
if !found {
msgStatus = courier.MsgErrored
}

// write our status
status := h.Backend().NewMsgStatusForExternalID(channel, zvStatus.CallbackMtRequest.ID, msgStatus)
err = h.Backend().WriteMsgStatus(status)
if err != nil {
return nil, err
}

return []courier.MsgStatus{status}, courier.WriteStatusSuccess(w, r, status)

}

// SendMsg sends the passed in message, returning any error
func (h *handler) SendMsg(msg courier.Msg) (courier.MsgStatus, error) {
username := msg.Channel().StringConfigForKey(courier.ConfigUsername, "")
if username == "" {
return nil, fmt.Errorf("no username set for Zenvia channel")
}

password := msg.Channel().StringConfigForKey(courier.ConfigPassword, "")
if password == "" {
return nil, fmt.Errorf("no password set for Zenvia channel")
}

encodedCreds := handlers.EncodeBase64([]string{username, ":", password})
authHeader := "Basic " + encodedCreds

zvMsg := zvOutgoingMsg{
SendSmsRequest: zvSendSmsRequest{
From: "Sender",
To: strings.TrimLeft(msg.URN().Path(), "+"),
Schedule: "",
Msg: courier.GetTextAndAttachments(msg),
ID: msg.ID().String(),
CallbackOption: strconv.Itoa(1),
AggregateID: "",
},
}

requestBody := new(bytes.Buffer)
json.NewEncoder(requestBody).Encode(zvMsg)

// build our request
req, err := http.NewRequest(http.MethodPost, sendURL, requestBody)
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Accept", "application/json")
req.Header.Set("Authorization", authHeader)
rr, err := utils.MakeHTTPRequest(req)

// record our status and log
status := h.Backend().NewMsgStatusForID(msg.Channel(), msg.ID(), courier.MsgErrored)
status.AddLog(courier.NewChannelLogFromRR(msg.Channel(), msg.ID(), rr))
if err != nil {
return status, err
}

// was this request successful?
responseMsgStatus, _ := jsonparser.GetString([]byte(rr.Body), "sendSmsResponse", "statusCode")
msgStatus, found := statusMapping[responseMsgStatus]
if msgStatus == courier.MsgErrored || !found {
return status, errors.Errorf("received non-success response from Zenvia '%s'", responseMsgStatus)
}
status.SetStatus(courier.MsgWired)

return status, nil

}
Loading

0 comments on commit fca18bc

Please sign in to comment.