Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. package irc
  2. import (
  3. "bufio"
  4. "bytes"
  5. "errors"
  6. "net"
  7. "unicode/utf8"
  8. "github.com/gorilla/websocket"
  9. "github.com/goshuirc/irc-go/ircmsg"
  10. "github.com/oragono/oragono/irc/utils"
  11. )
  12. const (
  13. maxReadQBytes = ircmsg.MaxlenTagsFromClient + MaxLineLen + 1024
  14. )
  15. var (
  16. crlf = []byte{'\r', '\n'}
  17. errReadQ = errors.New("ReadQ Exceeded")
  18. )
  19. // IRCConn abstracts away the distinction between a regular
  20. // net.Conn (which includes both raw TCP and TLS) and a websocket.
  21. // it doesn't expose the net.Conn, io.Reader, or io.Writer interfaces
  22. // because websockets are message-oriented, not stream-oriented, and
  23. // therefore this abstraction is message-oriented as well.
  24. type IRCConn interface {
  25. UnderlyingConn() *utils.WrappedConn
  26. // these take an IRC line or lines, correctly terminated with CRLF:
  27. WriteLine([]byte) error
  28. WriteLines([][]byte) error
  29. // this returns an IRC line, possibly terminated with CRLF, LF, or nothing:
  30. ReadLine() (line []byte, err error)
  31. Close() error
  32. }
  33. // IRCStreamConn is an IRCConn over a regular stream connection.
  34. type IRCStreamConn struct {
  35. conn *utils.WrappedConn
  36. reader *bufio.Reader
  37. }
  38. func NewIRCStreamConn(conn *utils.WrappedConn) *IRCStreamConn {
  39. return &IRCStreamConn{
  40. conn: conn,
  41. }
  42. }
  43. func (cc *IRCStreamConn) UnderlyingConn() *utils.WrappedConn {
  44. return cc.conn
  45. }
  46. func (cc *IRCStreamConn) WriteLine(buf []byte) (err error) {
  47. _, err = cc.conn.Write(buf)
  48. return
  49. }
  50. func (cc *IRCStreamConn) WriteLines(buffers [][]byte) (err error) {
  51. // on Linux, with a plaintext TCP or Unix domain socket,
  52. // the Go runtime will optimize this into a single writev(2) call:
  53. _, err = (*net.Buffers)(&buffers).WriteTo(cc.conn)
  54. return
  55. }
  56. func (cc *IRCStreamConn) ReadLine() (line []byte, err error) {
  57. // lazy initialize the reader in case the IP is banned
  58. if cc.reader == nil {
  59. cc.reader = bufio.NewReaderSize(cc.conn, maxReadQBytes)
  60. }
  61. var isPrefix bool
  62. line, isPrefix, err = cc.reader.ReadLine()
  63. if isPrefix {
  64. return nil, errReadQ
  65. }
  66. if globalUtf8EnforcementSetting && !utf8.Valid(line) {
  67. err = errInvalidUtf8
  68. }
  69. return
  70. }
  71. func (cc *IRCStreamConn) Close() (err error) {
  72. return cc.conn.Close()
  73. }
  74. // IRCWSConn is an IRCConn over a websocket.
  75. type IRCWSConn struct {
  76. conn *websocket.Conn
  77. }
  78. func NewIRCWSConn(conn *websocket.Conn) IRCWSConn {
  79. return IRCWSConn{conn: conn}
  80. }
  81. func (wc IRCWSConn) UnderlyingConn() *utils.WrappedConn {
  82. // just assume that the type is OK
  83. wConn, _ := wc.conn.UnderlyingConn().(*utils.WrappedConn)
  84. return wConn
  85. }
  86. func (wc IRCWSConn) WriteLine(buf []byte) (err error) {
  87. buf = bytes.TrimSuffix(buf, crlf)
  88. if !globalUtf8EnforcementSetting && !utf8.Valid(buf) {
  89. // there's not much we can do about this;
  90. // silently drop the message
  91. return nil
  92. }
  93. return wc.conn.WriteMessage(websocket.TextMessage, buf)
  94. }
  95. func (wc IRCWSConn) WriteLines(buffers [][]byte) (err error) {
  96. for _, buf := range buffers {
  97. err = wc.WriteLine(buf)
  98. if err != nil {
  99. return
  100. }
  101. }
  102. return
  103. }
  104. func (wc IRCWSConn) ReadLine() (line []byte, err error) {
  105. messageType, line, err := wc.conn.ReadMessage()
  106. if err == nil {
  107. if messageType == websocket.TextMessage {
  108. return line, nil
  109. } else {
  110. // for purposes of fakelag, treat non-text message as an empty line
  111. return nil, nil
  112. }
  113. } else if err == websocket.ErrReadLimit {
  114. return line, errReadQ
  115. } else {
  116. return line, err
  117. }
  118. }
  119. func (wc IRCWSConn) Close() (err error) {
  120. return wc.conn.Close()
  121. }