response.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  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. // Wrap wraps the stream with a new stream, and allows customized operations
  67. func (r *StreamResponse[T]) Wrap(fn func(T)) error {
  68. r.l.Lock()
  69. if r.closed {
  70. r.l.Unlock()
  71. return errors.New("stream is closed")
  72. }
  73. r.l.Unlock()
  74. for r.Next() {
  75. data, err := r.Read()
  76. if err != nil {
  77. return err
  78. }
  79. fn(data)
  80. }
  81. return nil
  82. }
  83. // Write writes data to the stream
  84. // returns error if the buffer is full
  85. func (r *StreamResponse[T]) Write(data T) error {
  86. r.l.Lock()
  87. if r.closed {
  88. r.l.Unlock()
  89. return nil
  90. }
  91. if r.q.Len() >= r.max {
  92. r.l.Unlock()
  93. return errors.New("queue is full")
  94. }
  95. r.q.PushBack(data)
  96. if r.q.Len() == 1 {
  97. if r.listening {
  98. r.sig <- true
  99. }
  100. }
  101. r.l.Unlock()
  102. return nil
  103. }
  104. // Close closes the stream
  105. func (r *StreamResponse[T]) Close() {
  106. r.l.Lock()
  107. if r.closed {
  108. r.l.Unlock()
  109. return
  110. }
  111. r.closed = true
  112. r.l.Unlock()
  113. select {
  114. case r.sig <- false:
  115. default:
  116. }
  117. close(r.sig)
  118. if r.onClose != nil {
  119. r.onClose()
  120. }
  121. }
  122. func (r *StreamResponse[T]) IsClosed() bool {
  123. r.l.Lock()
  124. defer r.l.Unlock()
  125. return r.closed
  126. }
  127. func (r *StreamResponse[T]) Size() int {
  128. r.l.Lock()
  129. defer r.l.Unlock()
  130. return r.q.Len()
  131. }
  132. // WriteError writes an error to the stream
  133. func (r *StreamResponse[T]) WriteError(err error) {
  134. r.l.Lock()
  135. defer r.l.Unlock()
  136. r.err = err
  137. if r.q.Len() == 0 {
  138. if r.listening {
  139. r.sig <- true
  140. }
  141. }
  142. }