originate.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  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. waiter := make(chan struct{})
  72. go res.listenEvents(waiter)
  73. <-waiter
  74. return res
  75. }
  76. type Originate struct {
  77. *OriginateRequest
  78. eventChan <-chan Event
  79. userEventChan chan Event
  80. locker *sync.RWMutex
  81. finished bool
  82. client *Client
  83. responseReason byte
  84. hangupCause byte
  85. }
  86. func (s *Originate) listenEvents(waiter chan<- struct{}) {
  87. log.Println("AMI-LISTEN-EVENTS")
  88. close(waiter)
  89. defer log.Println("AMI-LISTENER-ORIGINATE-CLOSED")
  90. for {
  91. e, ok := <-s.eventChan
  92. log.Println("E --------------------------->", e, ok)
  93. if !ok {
  94. s.finished = true
  95. close(s.userEventChan)
  96. return
  97. }
  98. s.locker.RLock()
  99. if s.userEventChan != nil {
  100. s.userEventChan <- e
  101. }
  102. s.locker.RUnlock()
  103. switch e.Name() {
  104. case "OriginateResponse":
  105. if reason, check := e.ActionData["Reason"]; check {
  106. reasonVal, _ := strconv.ParseInt(reason, 10, 32)
  107. s.responseReason = byte(reasonVal)
  108. }
  109. log.Println("RREASON", s.responseReason)
  110. if s.responseReason == 0 {
  111. s.finished = true
  112. close(s.userEventChan)
  113. return
  114. }
  115. //if s.responseReason != 4 {
  116. //s.client.removeEventListener(s.uuid)
  117. //}
  118. case "Hangup":
  119. if cause, check := e.ActionData["Cause"]; check {
  120. causeVal, _ := strconv.ParseInt(cause, 10, 32)
  121. s.hangupCause = byte(causeVal)
  122. }
  123. }
  124. }
  125. }
  126. func (s *Originate) IsFinished() bool {
  127. return s.finished
  128. }
  129. func (s *Originate) Events() (res <-chan Event) {
  130. return s.userEventChan
  131. }