response.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package stream
  2. import (
  3. "errors"
  4. "sync"
  5. "github.com/gammazero/deque"
  6. )
  7. type StreamResponse[T any] struct {
  8. q deque.Deque[T]
  9. l *sync.Mutex
  10. sig chan bool
  11. closed bool
  12. max int
  13. listening bool
  14. onClose func()
  15. err error
  16. }
  17. func NewStreamResponse[T any](max int) *StreamResponse[T] {
  18. return &StreamResponse[T]{
  19. l: &sync.Mutex{},
  20. sig: make(chan bool),
  21. max: max,
  22. }
  23. }
  24. func (r *StreamResponse[T]) OnClose(f func()) {
  25. r.onClose = f
  26. }
  27. // Next returns true if there are more data to be read
  28. // and waits for the next data to be available
  29. // returns false if the stream is closed
  30. // NOTE: even if the stream is closed, it will return true if there is data available
  31. func (r *StreamResponse[T]) Next() bool {
  32. r.l.Lock()
  33. if r.closed && r.q.Len() == 0 && r.err == nil {
  34. r.l.Unlock()
  35. return false
  36. }
  37. if r.q.Len() > 0 || r.err != nil {
  38. r.l.Unlock()
  39. return true
  40. }
  41. r.listening = true
  42. defer func() {
  43. r.listening = false
  44. }()
  45. r.l.Unlock()
  46. return <-r.sig
  47. }
  48. // Read reads buffered data from the stream and
  49. // it returns error only if the buffer is empty or an error is written to the stream
  50. func (r *StreamResponse[T]) Read() (T, error) {
  51. r.l.Lock()
  52. defer r.l.Unlock()
  53. if r.q.Len() > 0 {
  54. data := r.q.PopFront()
  55. return data, nil
  56. } else {
  57. var data T
  58. if r.err != nil {
  59. err := r.err
  60. r.err = nil
  61. return data, err
  62. }
  63. return data, errors.New("no data available")
  64. }
  65. }
  66. // Write writes data to the stream
  67. // returns error if the buffer is full
  68. func (r *StreamResponse[T]) Write(data T) error {
  69. r.l.Lock()
  70. if r.closed {
  71. r.l.Unlock()
  72. return nil
  73. }
  74. if r.q.Len() >= r.max {
  75. r.l.Unlock()
  76. return errors.New("queue is full")
  77. }
  78. r.q.PushBack(data)
  79. if r.q.Len() == 1 {
  80. if r.listening {
  81. r.sig <- true
  82. }
  83. }
  84. r.l.Unlock()
  85. return nil
  86. }
  87. // Close closes the stream
  88. func (r *StreamResponse[T]) Close() {
  89. r.l.Lock()
  90. if r.closed {
  91. r.l.Unlock()
  92. return
  93. }
  94. r.closed = true
  95. r.l.Unlock()
  96. select {
  97. case r.sig <- false:
  98. default:
  99. }
  100. close(r.sig)
  101. if r.onClose != nil {
  102. r.onClose()
  103. }
  104. }
  105. func (r *StreamResponse[T]) IsClosed() bool {
  106. r.l.Lock()
  107. defer r.l.Unlock()
  108. return r.closed
  109. }
  110. func (r *StreamResponse[T]) Size() int {
  111. r.l.Lock()
  112. defer r.l.Unlock()
  113. return r.q.Len()
  114. }
  115. // WriteError writes an error to the stream
  116. func (r *StreamResponse[T]) WriteError(err error) {
  117. r.l.Lock()
  118. defer r.l.Unlock()
  119. r.err = err
  120. if r.q.Len() == 0 {
  121. if r.listening {
  122. r.sig <- true
  123. }
  124. }
  125. }