originate.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. package ami
  2. import (
  3. "errors"
  4. "fmt"
  5. "log"
  6. "strconv"
  7. "sync"
  8. "time"
  9. "git.ali33.ru/fcg-xvii/go-tools/json"
  10. )
  11. func (s *Client) Originate(req *OriginateRequest) (*Originate, error) {
  12. if s.state != StateAuth {
  13. return nil, errors.New("AMI IS NOT AUTH")
  14. }
  15. req.uuid = time.Now().UnixNano()
  16. timeout := RequestTimeoutDefault
  17. if req.Timeout > timeout {
  18. timeout = req.Timeout + time.Millisecond*500
  19. }
  20. resp, check := s.Request(req.Request(), timeout)
  21. if !check {
  22. return nil, errors.New("Originate request timeout")
  23. }
  24. if resp.IsError() {
  25. return nil, fmt.Errorf("Originate error: %v", resp.ErrorMessage())
  26. }
  27. res := initOriginate(req, s)
  28. return res, nil
  29. }
  30. //////////////////////////////////////////////////////////////////
  31. type OriginateRequest struct {
  32. Channel string
  33. Context string
  34. Exten string
  35. Priority string
  36. Timeout time.Duration
  37. CallerID string
  38. Variable json.Map
  39. Account string
  40. Application string
  41. Data string
  42. uuid int64
  43. }
  44. func (s *OriginateRequest) Request() (res Request) {
  45. res = InitRequest("Originate")
  46. res.SetParam("Channel", s.Channel)
  47. res.SetParam("Context", s.Context)
  48. res.SetParam("Exten", s.Exten)
  49. if s.Timeout > 0 {
  50. res.SetParam("Timeout", fmt.Sprintf("%v", int(s.Timeout/time.Millisecond)))
  51. }
  52. res.SetParam("Priority", s.Priority)
  53. res.SetParam("CallerID", s.CallerID)
  54. res.SetParam("Account", s.Account)
  55. res.SetParam("Application", s.Application)
  56. res.SetParam("Data", s.Data)
  57. res.SetParam("Async", "true")
  58. res.SetParam("ChannelID", fmt.Sprint(s.uuid))
  59. res.SetVariables(s.Variable)
  60. return res
  61. }
  62. /////////////////////////////////////////////////////////////////
  63. func initOriginate(req *OriginateRequest, client *Client) *Originate {
  64. res := &Originate{
  65. OriginateRequest: req,
  66. eventChan: client.registerEventListener(req.uuid),
  67. locker: new(sync.RWMutex),
  68. client: client,
  69. userEventChan: make(chan Event, 1),
  70. }
  71. go res.listenEvents()
  72. return res
  73. }
  74. type Originate struct {
  75. *OriginateRequest
  76. eventChan <-chan Event
  77. userEventChan chan Event
  78. locker *sync.RWMutex
  79. finished bool
  80. client *Client
  81. responseReason byte
  82. hangupCause byte
  83. }
  84. func (s *Originate) listenEvents() {
  85. defer log.Println("AMI-LISTENER-ORIGINATE-CLOSED")
  86. for {
  87. e, ok := <-s.eventChan
  88. log.Println("E --------------------------->", e, ok)
  89. if !ok {
  90. s.finished = true
  91. close(s.userEventChan)
  92. return
  93. }
  94. s.locker.RLock()
  95. if s.userEventChan != nil {
  96. s.userEventChan <- e
  97. }
  98. s.locker.RUnlock()
  99. switch e.Name() {
  100. case "OriginateResponse":
  101. if reason, check := e.ActionData["Reason"]; check {
  102. reasonVal, _ := strconv.ParseInt(reason, 10, 32)
  103. s.responseReason = byte(reasonVal)
  104. }
  105. log.Println("RREASON", s.responseReason)
  106. if s.responseReason == 0 {
  107. s.finished = true
  108. close(s.userEventChan)
  109. return
  110. }
  111. //if s.responseReason != 4 {
  112. //s.client.removeEventListener(s.uuid)
  113. //}
  114. case "Hangup":
  115. if cause, check := e.ActionData["Cause"]; check {
  116. causeVal, _ := strconv.ParseInt(cause, 10, 32)
  117. s.hangupCause = byte(causeVal)
  118. }
  119. }
  120. }
  121. }
  122. func (s *Originate) IsFinished() bool {
  123. return s.finished
  124. }
  125. func (s *Originate) Events() (res <-chan Event) {
  126. return s.userEventChan
  127. }