2017-03-15 20:45:35 +03:00
|
|
|
package endpoint
|
|
|
|
|
|
|
|
import (
|
2018-04-18 00:56:43 +03:00
|
|
|
"net"
|
2017-03-15 20:45:35 +03:00
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"fmt"
|
2018-04-18 00:56:43 +03:00
|
|
|
|
2017-03-15 20:45:35 +03:00
|
|
|
"github.com/streadway/amqp"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
AMQPExpiresAfter = time.Second * 30
|
|
|
|
)
|
|
|
|
|
|
|
|
type AMQPEndpointConn struct {
|
|
|
|
mu sync.Mutex
|
|
|
|
ep Endpoint
|
|
|
|
conn *amqp.Connection
|
|
|
|
channel *amqp.Channel
|
|
|
|
ex bool
|
|
|
|
t time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
func (conn *AMQPEndpointConn) Expired() bool {
|
|
|
|
conn.mu.Lock()
|
|
|
|
defer conn.mu.Unlock()
|
|
|
|
if !conn.ex {
|
|
|
|
if time.Now().Sub(conn.t) > kafkaExpiresAfter {
|
|
|
|
conn.ex = true
|
|
|
|
conn.close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return conn.ex
|
|
|
|
}
|
|
|
|
|
|
|
|
func (conn *AMQPEndpointConn) close() {
|
|
|
|
if conn.conn != nil {
|
|
|
|
conn.conn.Close()
|
|
|
|
conn.conn = nil
|
|
|
|
conn.channel = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (conn *AMQPEndpointConn) Send(msg string) error {
|
|
|
|
conn.mu.Lock()
|
|
|
|
defer conn.mu.Unlock()
|
|
|
|
|
|
|
|
if conn.ex {
|
|
|
|
return errExpired
|
|
|
|
}
|
|
|
|
conn.t = time.Now()
|
|
|
|
|
|
|
|
if conn.conn == nil {
|
|
|
|
prefix := "amqp://"
|
|
|
|
if conn.ep.AMQP.SSL {
|
|
|
|
prefix = "amqps://"
|
|
|
|
}
|
|
|
|
|
2018-04-18 00:56:43 +03:00
|
|
|
var cfg amqp.Config
|
|
|
|
cfg.Dial = func(network, addr string) (net.Conn, error) {
|
|
|
|
return net.DialTimeout(network, addr, time.Second)
|
|
|
|
}
|
|
|
|
c, err := amqp.DialConfig(fmt.Sprintf("%s%s", prefix, conn.ep.AMQP.URI), cfg)
|
2017-03-15 20:45:35 +03:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
channel, err := c.Channel()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Declare new exchange
|
|
|
|
if err := channel.ExchangeDeclare(
|
|
|
|
conn.ep.AMQP.QueueName,
|
2018-04-18 00:56:43 +03:00
|
|
|
conn.ep.AMQP.Type,
|
|
|
|
conn.ep.AMQP.Durable,
|
|
|
|
conn.ep.AMQP.AutoDelete,
|
|
|
|
conn.ep.AMQP.Internal,
|
|
|
|
conn.ep.AMQP.NoWait,
|
2017-03-15 20:45:35 +03:00
|
|
|
nil,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create queue if queue don't exists
|
|
|
|
if _, err := channel.QueueDeclare(
|
|
|
|
conn.ep.AMQP.QueueName,
|
2018-04-18 00:56:43 +03:00
|
|
|
conn.ep.AMQP.Durable,
|
|
|
|
conn.ep.AMQP.AutoDelete,
|
2017-03-15 20:45:35 +03:00
|
|
|
false,
|
2018-04-18 00:56:43 +03:00
|
|
|
conn.ep.AMQP.NoWait,
|
2017-03-15 20:45:35 +03:00
|
|
|
nil,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Binding exchange to queue
|
|
|
|
if err := channel.QueueBind(
|
|
|
|
conn.ep.AMQP.QueueName,
|
|
|
|
conn.ep.AMQP.RouteKey,
|
|
|
|
conn.ep.AMQP.QueueName,
|
2018-04-18 00:56:43 +03:00
|
|
|
conn.ep.AMQP.NoWait,
|
2017-03-15 20:45:35 +03:00
|
|
|
nil,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
conn.conn = c
|
|
|
|
conn.channel = channel
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := conn.channel.Publish(
|
|
|
|
conn.ep.AMQP.QueueName,
|
|
|
|
conn.ep.AMQP.RouteKey,
|
2018-04-18 00:56:43 +03:00
|
|
|
conn.ep.AMQP.Mandatory,
|
|
|
|
conn.ep.AMQP.Immediate,
|
2017-03-15 20:45:35 +03:00
|
|
|
amqp.Publishing{
|
|
|
|
Headers: amqp.Table{},
|
|
|
|
ContentType: "application/json",
|
|
|
|
ContentEncoding: "",
|
|
|
|
Body: []byte(msg),
|
2018-04-18 00:56:43 +03:00
|
|
|
DeliveryMode: conn.ep.AMQP.DeliveryMode,
|
2017-03-15 20:45:35 +03:00
|
|
|
Priority: 0,
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func newAMQPEndpointConn(ep Endpoint) *AMQPEndpointConn {
|
|
|
|
return &AMQPEndpointConn{
|
|
|
|
ep: ep,
|
|
|
|
t: time.Now(),
|
|
|
|
}
|
|
|
|
}
|