You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

client_lookup_set.go 5.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. package irc
  2. import (
  3. "database/sql"
  4. "errors"
  5. "log"
  6. "regexp"
  7. "strings"
  8. )
  9. var (
  10. ErrNickMissing = errors.New("nick missing")
  11. ErrNicknameInUse = errors.New("nickname in use")
  12. ErrNicknameMismatch = errors.New("nickname mismatch")
  13. wildMaskExpr = regexp.MustCompile(`\*|\?`)
  14. likeQuoter = strings.NewReplacer(
  15. `\`, `\\`,
  16. `%`, `\%`,
  17. `_`, `\_`,
  18. `*`, `%`,
  19. `?`, `_`)
  20. )
  21. func HasWildcards(mask string) bool {
  22. return wildMaskExpr.MatchString(mask)
  23. }
  24. func ExpandUserHost(userhost Name) (expanded Name) {
  25. expanded = userhost
  26. // fill in missing wildcards for nicks
  27. if !strings.Contains(expanded.String(), "!") {
  28. expanded += "!*"
  29. }
  30. if !strings.Contains(expanded.String(), "@") {
  31. expanded += "@*"
  32. }
  33. return
  34. }
  35. func QuoteLike(userhost Name) string {
  36. return likeQuoter.Replace(userhost.String())
  37. }
  38. type ClientLookupSet struct {
  39. byNick map[Name]*Client
  40. db *ClientDB
  41. }
  42. func NewClientLookupSet() *ClientLookupSet {
  43. return &ClientLookupSet{
  44. byNick: make(map[Name]*Client),
  45. db: NewClientDB(),
  46. }
  47. }
  48. func (clients *ClientLookupSet) Get(nick Name) *Client {
  49. return clients.byNick[nick.ToLower()]
  50. }
  51. func (clients *ClientLookupSet) Add(client *Client) error {
  52. if !client.HasNick() {
  53. return ErrNickMissing
  54. }
  55. if clients.Get(client.nick) != nil {
  56. return ErrNicknameInUse
  57. }
  58. clients.byNick[client.Nick().ToLower()] = client
  59. clients.db.Add(client)
  60. return nil
  61. }
  62. func (clients *ClientLookupSet) Remove(client *Client) error {
  63. if !client.HasNick() {
  64. return ErrNickMissing
  65. }
  66. if clients.Get(client.nick) != client {
  67. return ErrNicknameMismatch
  68. }
  69. delete(clients.byNick, client.nick.ToLower())
  70. clients.db.Remove(client)
  71. return nil
  72. }
  73. func (clients *ClientLookupSet) FindAll(userhost Name) (set ClientSet) {
  74. userhost = ExpandUserHost(userhost)
  75. set = make(ClientSet)
  76. rows, err := clients.db.db.Query(
  77. `SELECT nickname FROM client WHERE userhost LIKE ? ESCAPE '\'`,
  78. QuoteLike(userhost))
  79. if err != nil {
  80. Log.error.Println("ClientLookupSet.FindAll.Query:", err)
  81. return
  82. }
  83. for rows.Next() {
  84. var sqlNickname string
  85. err := rows.Scan(&sqlNickname)
  86. if err != nil {
  87. Log.error.Println("ClientLookupSet.FindAll.Scan:", err)
  88. return
  89. }
  90. nickname := Name(sqlNickname)
  91. client := clients.Get(nickname)
  92. if client == nil {
  93. Log.error.Println("ClientLookupSet.FindAll: missing client:", nickname)
  94. continue
  95. }
  96. set.Add(client)
  97. }
  98. return
  99. }
  100. func (clients *ClientLookupSet) Find(userhost Name) *Client {
  101. userhost = ExpandUserHost(userhost)
  102. row := clients.db.db.QueryRow(
  103. `SELECT nickname FROM client WHERE userhost LIKE ? ESCAPE '\' LIMIT 1`,
  104. QuoteLike(userhost))
  105. var nickname Name
  106. err := row.Scan(&nickname)
  107. if err != nil {
  108. Log.error.Println("ClientLookupSet.Find:", err)
  109. return nil
  110. }
  111. return clients.Get(nickname)
  112. }
  113. //
  114. // client db
  115. //
  116. type ClientDB struct {
  117. db *sql.DB
  118. }
  119. func NewClientDB() *ClientDB {
  120. db := &ClientDB{
  121. db: OpenDB(":memory:"),
  122. }
  123. stmts := []string{
  124. `CREATE TABLE client (
  125. nickname TEXT NOT NULL COLLATE NOCASE UNIQUE,
  126. userhost TEXT NOT NULL COLLATE NOCASE,
  127. UNIQUE (nickname, userhost) ON CONFLICT REPLACE)`,
  128. `CREATE UNIQUE INDEX idx_nick ON client (nickname COLLATE NOCASE)`,
  129. `CREATE UNIQUE INDEX idx_uh ON client (userhost COLLATE NOCASE)`,
  130. }
  131. for _, stmt := range stmts {
  132. _, err := db.db.Exec(stmt)
  133. if err != nil {
  134. log.Fatal("NewClientDB: ", stmt, err)
  135. }
  136. }
  137. return db
  138. }
  139. func (db *ClientDB) Add(client *Client) {
  140. _, err := db.db.Exec(`INSERT INTO client (nickname, userhost) VALUES (?, ?)`,
  141. client.Nick().String(), client.UserHost().String())
  142. if err != nil {
  143. Log.error.Println("ClientDB.Add:", err)
  144. }
  145. }
  146. func (db *ClientDB) Remove(client *Client) {
  147. _, err := db.db.Exec(`DELETE FROM client WHERE nickname = ?`,
  148. client.Nick().String())
  149. if err != nil {
  150. Log.error.Println("ClientDB.Remove:", err)
  151. }
  152. }
  153. //
  154. // usermask to regexp
  155. //
  156. type UserMaskSet struct {
  157. masks map[Name]bool
  158. regexp *regexp.Regexp
  159. }
  160. func NewUserMaskSet() *UserMaskSet {
  161. return &UserMaskSet{
  162. masks: make(map[Name]bool),
  163. }
  164. }
  165. func (set *UserMaskSet) Add(mask Name) bool {
  166. if set.masks[mask] {
  167. return false
  168. }
  169. set.masks[mask] = true
  170. set.setRegexp()
  171. return true
  172. }
  173. func (set *UserMaskSet) AddAll(masks []Name) (added bool) {
  174. for _, mask := range masks {
  175. if !added && !set.masks[mask] {
  176. added = true
  177. }
  178. set.masks[mask] = true
  179. }
  180. set.setRegexp()
  181. return
  182. }
  183. func (set *UserMaskSet) Remove(mask Name) bool {
  184. if !set.masks[mask] {
  185. return false
  186. }
  187. delete(set.masks, mask)
  188. set.setRegexp()
  189. return true
  190. }
  191. func (set *UserMaskSet) Match(userhost Name) bool {
  192. if set.regexp == nil {
  193. return false
  194. }
  195. return set.regexp.MatchString(userhost.String())
  196. }
  197. func (set *UserMaskSet) String() string {
  198. masks := make([]string, len(set.masks))
  199. index := 0
  200. for mask := range set.masks {
  201. masks[index] = mask.String()
  202. index += 1
  203. }
  204. return strings.Join(masks, " ")
  205. }
  206. // Generate a regular expression from the set of user mask
  207. // strings. Masks are split at the two types of wildcards, `*` and
  208. // `?`. All the pieces are meta-escaped. `*` is replaced with `.*`,
  209. // the regexp equivalent. Likewise, `?` is replaced with `.`. The
  210. // parts are re-joined and finally all masks are joined into a big
  211. // or-expression.
  212. func (set *UserMaskSet) setRegexp() {
  213. if len(set.masks) == 0 {
  214. set.regexp = nil
  215. return
  216. }
  217. maskExprs := make([]string, len(set.masks))
  218. index := 0
  219. for mask := range set.masks {
  220. manyParts := strings.Split(mask.String(), "*")
  221. manyExprs := make([]string, len(manyParts))
  222. for mindex, manyPart := range manyParts {
  223. oneParts := strings.Split(manyPart, "?")
  224. oneExprs := make([]string, len(oneParts))
  225. for oindex, onePart := range oneParts {
  226. oneExprs[oindex] = regexp.QuoteMeta(onePart)
  227. }
  228. manyExprs[mindex] = strings.Join(oneExprs, ".")
  229. }
  230. maskExprs[index] = strings.Join(manyExprs, ".*")
  231. }
  232. expr := "^" + strings.Join(maskExprs, "|") + "$"
  233. set.regexp, _ = regexp.Compile(expr)
  234. }