mirror of
https://github.com/ergochat/ergo.git
synced 2024-12-22 18:52:41 +01:00
partially-working sqlite-based usermasks
This commit is contained in:
parent
832a5e1e19
commit
69cdad45ac
@ -45,5 +45,6 @@ func main() {
|
|||||||
irc.DEBUG_CHANNEL = config.Debug.Channel
|
irc.DEBUG_CHANNEL = config.Debug.Channel
|
||||||
irc.DEBUG_SERVER = config.Debug.Server
|
irc.DEBUG_SERVER = config.Debug.Server
|
||||||
|
|
||||||
|
log.Println(irc.SEM_VER, "running")
|
||||||
irc.NewServer(config).Run()
|
irc.NewServer(config).Run()
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ import (
|
|||||||
|
|
||||||
type Channel struct {
|
type Channel struct {
|
||||||
flags ChannelModeSet
|
flags ChannelModeSet
|
||||||
lists map[ChannelMode][]UserMask
|
lists map[ChannelMode]UserMaskSet
|
||||||
key string
|
key string
|
||||||
members MemberSet
|
members MemberSet
|
||||||
name string
|
name string
|
||||||
@ -26,10 +26,10 @@ func IsChannel(target string) bool {
|
|||||||
func NewChannel(s *Server, name string) *Channel {
|
func NewChannel(s *Server, name string) *Channel {
|
||||||
channel := &Channel{
|
channel := &Channel{
|
||||||
flags: make(ChannelModeSet),
|
flags: make(ChannelModeSet),
|
||||||
lists: map[ChannelMode][]UserMask{
|
lists: map[ChannelMode]UserMaskSet{
|
||||||
BanMask: []UserMask{},
|
BanMask: make(UserMaskSet),
|
||||||
ExceptMask: []UserMask{},
|
ExceptMask: make(UserMaskSet),
|
||||||
InviteMask: []UserMask{},
|
InviteMask: make(UserMaskSet),
|
||||||
},
|
},
|
||||||
members: make(MemberSet),
|
members: make(MemberSet),
|
||||||
name: strings.ToLower(name),
|
name: strings.ToLower(name),
|
||||||
@ -307,7 +307,7 @@ func (channel *Channel) applyMode(client *Client, change *ChannelModeChange) boo
|
|||||||
case BanMask, ExceptMask, InviteMask:
|
case BanMask, ExceptMask, InviteMask:
|
||||||
// TODO add/remove
|
// TODO add/remove
|
||||||
|
|
||||||
for _, mask := range channel.lists[change.mode] {
|
for mask := range channel.lists[change.mode] {
|
||||||
client.RplMaskList(change.mode, channel, mask)
|
client.RplMaskList(change.mode, channel, mask)
|
||||||
}
|
}
|
||||||
client.RplEndOfMaskList(change.mode, channel)
|
client.RplEndOfMaskList(change.mode, channel)
|
||||||
|
@ -656,7 +656,7 @@ func (msg *WhoisCommand) String() string {
|
|||||||
|
|
||||||
type WhoCommand struct {
|
type WhoCommand struct {
|
||||||
BaseCommand
|
BaseCommand
|
||||||
mask Mask
|
mask string
|
||||||
operatorOnly bool
|
operatorOnly bool
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -665,7 +665,7 @@ func NewWhoCommand(args []string) (editableCommand, error) {
|
|||||||
cmd := &WhoCommand{}
|
cmd := &WhoCommand{}
|
||||||
|
|
||||||
if len(args) > 0 {
|
if len(args) > 0 {
|
||||||
cmd.mask = Mask(args[0])
|
cmd.mask = args[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len(args) > 1) && (args[1] == "o") {
|
if (len(args) > 1) && (args[1] == "o") {
|
||||||
|
18
irc/reply.go
18
irc/reply.go
@ -196,6 +196,16 @@ func (target *Client) RplYoureOper() {
|
|||||||
":You are now an IRC operator")
|
":You are now an IRC operator")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (target *Client) RplWhois(client *Client) {
|
||||||
|
target.RplWhoisUser(client)
|
||||||
|
if client.flags[Operator] {
|
||||||
|
target.RplWhoisOperator(client)
|
||||||
|
}
|
||||||
|
target.RplWhoisIdle(client)
|
||||||
|
target.RplWhoisChannels(client)
|
||||||
|
target.RplEndOfWhois()
|
||||||
|
}
|
||||||
|
|
||||||
func (target *Client) RplWhoisUser(client *Client) {
|
func (target *Client) RplWhoisUser(client *Client) {
|
||||||
target.NumericReply(RPL_WHOISUSER,
|
target.NumericReply(RPL_WHOISUSER,
|
||||||
"%s %s %s * :%s", client.Nick(), client.username, client.hostname,
|
"%s %s %s * :%s", client.Nick(), client.username, client.hostname,
|
||||||
@ -258,7 +268,7 @@ func (target *Client) RplEndOfWho(name string) {
|
|||||||
"%s :End of WHO list", name)
|
"%s :End of WHO list", name)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (target *Client) RplMaskList(mode ChannelMode, channel *Channel, mask UserMask) {
|
func (target *Client) RplMaskList(mode ChannelMode, channel *Channel, mask string) {
|
||||||
switch mode {
|
switch mode {
|
||||||
case BanMask:
|
case BanMask:
|
||||||
target.RplBanList(channel, mask)
|
target.RplBanList(channel, mask)
|
||||||
@ -284,7 +294,7 @@ func (target *Client) RplEndOfMaskList(mode ChannelMode, channel *Channel) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (target *Client) RplBanList(channel *Channel, mask UserMask) {
|
func (target *Client) RplBanList(channel *Channel, mask string) {
|
||||||
target.NumericReply(RPL_BANLIST,
|
target.NumericReply(RPL_BANLIST,
|
||||||
"%s %s", channel, mask)
|
"%s %s", channel, mask)
|
||||||
}
|
}
|
||||||
@ -294,7 +304,7 @@ func (target *Client) RplEndOfBanList(channel *Channel) {
|
|||||||
"%s :End of channel ban list", channel)
|
"%s :End of channel ban list", channel)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (target *Client) RplExceptList(channel *Channel, mask UserMask) {
|
func (target *Client) RplExceptList(channel *Channel, mask string) {
|
||||||
target.NumericReply(RPL_EXCEPTLIST,
|
target.NumericReply(RPL_EXCEPTLIST,
|
||||||
"%s %s", channel, mask)
|
"%s %s", channel, mask)
|
||||||
}
|
}
|
||||||
@ -304,7 +314,7 @@ func (target *Client) RplEndOfExceptList(channel *Channel) {
|
|||||||
"%s :End of channel exception list", channel)
|
"%s :End of channel exception list", channel)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (target *Client) RplInviteList(channel *Channel, mask UserMask) {
|
func (target *Client) RplInviteList(channel *Channel, mask string) {
|
||||||
target.NumericReply(RPL_INVITELIST,
|
target.NumericReply(RPL_INVITELIST,
|
||||||
"%s %s", channel, mask)
|
"%s %s", channel, mask)
|
||||||
}
|
}
|
||||||
|
202
irc/server.go
202
irc/server.go
@ -3,6 +3,7 @@ package irc
|
|||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"database/sql"
|
"database/sql"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
@ -18,7 +19,7 @@ import (
|
|||||||
|
|
||||||
type Server struct {
|
type Server struct {
|
||||||
channels ChannelNameMap
|
channels ChannelNameMap
|
||||||
clients ClientNameMap
|
clients *ClientLookupSet
|
||||||
commands chan Command
|
commands chan Command
|
||||||
ctime time.Time
|
ctime time.Time
|
||||||
db *sql.DB
|
db *sql.DB
|
||||||
@ -35,7 +36,7 @@ type Server struct {
|
|||||||
func NewServer(config *Config) *Server {
|
func NewServer(config *Config) *Server {
|
||||||
server := &Server{
|
server := &Server{
|
||||||
channels: make(ChannelNameMap),
|
channels: make(ChannelNameMap),
|
||||||
clients: make(ClientNameMap),
|
clients: NewClientLookupSet(),
|
||||||
commands: make(chan Command, 16),
|
commands: make(chan Command, 16),
|
||||||
ctime: time.Now(),
|
ctime: time.Now(),
|
||||||
db: OpenDB(config.Server.Database),
|
db: OpenDB(config.Server.Database),
|
||||||
@ -136,7 +137,7 @@ func (server *Server) processCommand(cmd Command) {
|
|||||||
|
|
||||||
func (server *Server) Shutdown() {
|
func (server *Server) Shutdown() {
|
||||||
server.db.Close()
|
server.db.Close()
|
||||||
for _, client := range server.clients {
|
for _, client := range server.clients.byNick {
|
||||||
client.Reply(RplNotice(server, client, "shutting down"))
|
client.Reply(RplNotice(server, client, "shutting down"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -556,19 +557,14 @@ func (m *WhoisCommand) HandleServer(server *Server) {
|
|||||||
// TODO implement target query
|
// TODO implement target query
|
||||||
|
|
||||||
for _, mask := range m.masks {
|
for _, mask := range m.masks {
|
||||||
// TODO implement wildcard matching
|
matches := server.clients.FindAll(mask)
|
||||||
mclient := server.clients.Get(mask)
|
if len(matches) == 0 {
|
||||||
if mclient == nil {
|
|
||||||
client.ErrNoSuchNick(mask)
|
client.ErrNoSuchNick(mask)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
client.RplWhoisUser(mclient)
|
for mclient := range matches {
|
||||||
if mclient.flags[Operator] {
|
client.RplWhois(mclient)
|
||||||
client.RplWhoisOperator(mclient)
|
|
||||||
}
|
}
|
||||||
client.RplWhoisIdle(mclient)
|
|
||||||
client.RplWhoisChannels(mclient)
|
|
||||||
client.RplEndOfWhois()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -583,9 +579,9 @@ func (msg *ChannelModeCommand) HandleServer(server *Server) {
|
|||||||
channel.Mode(client, msg.changes)
|
channel.Mode(client, msg.changes)
|
||||||
}
|
}
|
||||||
|
|
||||||
func whoChannel(client *Client, channel *Channel) {
|
func whoChannel(client *Client, channel *Channel, friends ClientSet) {
|
||||||
for member := range channel.members {
|
for member := range channel.members {
|
||||||
if !client.flags[Invisible] {
|
if !client.flags[Invisible] || friends[client] {
|
||||||
client.RplWhoReply(channel, member)
|
client.RplWhoReply(channel, member)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -593,27 +589,21 @@ func whoChannel(client *Client, channel *Channel) {
|
|||||||
|
|
||||||
func (msg *WhoCommand) HandleServer(server *Server) {
|
func (msg *WhoCommand) HandleServer(server *Server) {
|
||||||
client := msg.Client()
|
client := msg.Client()
|
||||||
|
friends := client.Friends()
|
||||||
|
mask := msg.mask
|
||||||
|
|
||||||
// TODO implement wildcard matching
|
|
||||||
mask := string(msg.mask)
|
|
||||||
if mask == "" {
|
if mask == "" {
|
||||||
for _, channel := range server.channels {
|
for _, channel := range server.channels {
|
||||||
for member := range channel.members {
|
whoChannel(client, channel, friends)
|
||||||
if !client.flags[Invisible] {
|
|
||||||
client.RplWhoReply(channel, member)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else if IsChannel(mask) {
|
} else if IsChannel(mask) {
|
||||||
|
// TODO implement wildcard matching
|
||||||
channel := server.channels.Get(mask)
|
channel := server.channels.Get(mask)
|
||||||
if channel != nil {
|
if channel != nil {
|
||||||
for member := range channel.members {
|
whoChannel(client, channel, friends)
|
||||||
client.RplWhoReply(channel, member)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
mclient := server.clients.Get(mask)
|
for mclient := range server.clients.FindAll(mask) {
|
||||||
if mclient != nil {
|
|
||||||
client.RplWhoReply(nil, mclient)
|
client.RplWhoReply(nil, mclient)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -853,3 +843,163 @@ func (msg *KillCommand) HandleServer(server *Server) {
|
|||||||
quitMsg := fmt.Sprintf("KILLed by %s: %s", client.Nick(), msg.comment)
|
quitMsg := fmt.Sprintf("KILLed by %s: %s", client.Nick(), msg.comment)
|
||||||
target.Quit(quitMsg)
|
target.Quit(quitMsg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// keeping track of clients
|
||||||
|
//
|
||||||
|
|
||||||
|
type ClientLookupSet struct {
|
||||||
|
byNick map[string]*Client
|
||||||
|
db *ClientDB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClientLookupSet() *ClientLookupSet {
|
||||||
|
return &ClientLookupSet{
|
||||||
|
byNick: make(map[string]*Client),
|
||||||
|
db: NewClientDB(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrNickMissing = errors.New("nick missing")
|
||||||
|
ErrNicknameInUse = errors.New("nickname in use")
|
||||||
|
ErrNicknameMismatch = errors.New("nickname mismatch")
|
||||||
|
)
|
||||||
|
|
||||||
|
func (clients *ClientLookupSet) Get(nick string) *Client {
|
||||||
|
return clients.byNick[strings.ToLower(nick)]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (clients *ClientLookupSet) Add(client *Client) error {
|
||||||
|
if !client.HasNick() {
|
||||||
|
return ErrNickMissing
|
||||||
|
}
|
||||||
|
if clients.Get(client.nick) != nil {
|
||||||
|
return ErrNicknameInUse
|
||||||
|
}
|
||||||
|
clients.byNick[strings.ToLower(client.nick)] = client
|
||||||
|
clients.db.Add(client)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (clients *ClientLookupSet) Remove(client *Client) error {
|
||||||
|
if !client.HasNick() {
|
||||||
|
return ErrNickMissing
|
||||||
|
}
|
||||||
|
if clients.Get(client.nick) != client {
|
||||||
|
return ErrNicknameMismatch
|
||||||
|
}
|
||||||
|
delete(clients.byNick, strings.ToLower(client.nick))
|
||||||
|
clients.db.Remove(client)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ExpandUserHost(userhost string) (expanded string) {
|
||||||
|
expanded = userhost
|
||||||
|
// fill in missing wildcards for nicks
|
||||||
|
if !strings.Contains(expanded, "!") {
|
||||||
|
expanded += "!*"
|
||||||
|
}
|
||||||
|
if !strings.Contains(expanded, "@") {
|
||||||
|
expanded += "@*"
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (clients *ClientLookupSet) FindAll(userhost string) (set ClientSet) {
|
||||||
|
userhost = ExpandUserHost(userhost)
|
||||||
|
set = make(ClientSet)
|
||||||
|
rows, err := clients.db.db.Query(
|
||||||
|
`SELECT nickname FROM client
|
||||||
|
WHERE userhost LIKE ? ESCAPE '\'`,
|
||||||
|
QuoteLike(userhost))
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for rows.Next() {
|
||||||
|
var nickname string
|
||||||
|
err := rows.Scan(&nickname)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
client := clients.Get(nickname)
|
||||||
|
if client != nil {
|
||||||
|
set.Add(client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (clients *ClientLookupSet) Find(userhost string) *Client {
|
||||||
|
userhost = ExpandUserHost(userhost)
|
||||||
|
row := clients.db.db.QueryRow(
|
||||||
|
`SELECT nickname FROM client
|
||||||
|
WHERE userhost LIKE ? ESCAPE \
|
||||||
|
LIMIT 1`,
|
||||||
|
QuoteLike(userhost))
|
||||||
|
var nickname string
|
||||||
|
err := row.Scan(&nickname)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("ClientLookupSet.Find: ", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return clients.Get(nickname)
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// client db
|
||||||
|
//
|
||||||
|
|
||||||
|
type ClientDB struct {
|
||||||
|
db *sql.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClientDB() *ClientDB {
|
||||||
|
db := &ClientDB{
|
||||||
|
db: OpenDB(":memory:"),
|
||||||
|
}
|
||||||
|
_, err := db.db.Exec(`
|
||||||
|
CREATE TABLE client (
|
||||||
|
nickname TEXT NOT NULL UNIQUE,
|
||||||
|
userhost TEXT NOT NULL)`)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
_, err = db.db.Exec(`
|
||||||
|
CREATE UNIQUE INDEX nickname_index ON client (nickname)`)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
return db
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *ClientDB) Add(client *Client) {
|
||||||
|
_, err := db.db.Exec(`INSERT INTO client (nickname, userhost) VALUES (?, ?)`,
|
||||||
|
client.Nick(), client.UserHost())
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *ClientDB) Remove(client *Client) {
|
||||||
|
_, err := db.db.Exec(`DELETE FROM client WHERE nickname = ?`,
|
||||||
|
client.Nick())
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func QuoteLike(userhost string) (like string) {
|
||||||
|
like = userhost
|
||||||
|
// escape escape char
|
||||||
|
like = strings.Replace(like, `\`, `\\`, -1)
|
||||||
|
// escape meta-many
|
||||||
|
like = strings.Replace(like, `%`, `\%`, -1)
|
||||||
|
// escape meta-one
|
||||||
|
like = strings.Replace(like, `_`, `\_`, -1)
|
||||||
|
// swap meta-many
|
||||||
|
like = strings.Replace(like, `*`, `%`, -1)
|
||||||
|
// swap meta-one
|
||||||
|
like = strings.Replace(like, `?`, `_`, -1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
52
irc/types.go
52
irc/types.go
@ -1,7 +1,6 @@
|
|||||||
package irc
|
package irc
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
@ -10,8 +9,7 @@ import (
|
|||||||
// simple types
|
// simple types
|
||||||
//
|
//
|
||||||
|
|
||||||
// a string with wildcards
|
type UserMaskSet map[string]bool
|
||||||
type Mask string
|
|
||||||
|
|
||||||
// add, remove, list modes
|
// add, remove, list modes
|
||||||
type ModeOp rune
|
type ModeOp rune
|
||||||
@ -74,40 +72,6 @@ func (channels ChannelNameMap) Remove(channel *Channel) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type ClientNameMap map[string]*Client
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrNickMissing = errors.New("nick missing")
|
|
||||||
ErrNicknameInUse = errors.New("nickname in use")
|
|
||||||
ErrNicknameMismatch = errors.New("nickname mismatch")
|
|
||||||
)
|
|
||||||
|
|
||||||
func (clients ClientNameMap) Get(nick string) *Client {
|
|
||||||
return clients[strings.ToLower(nick)]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (clients ClientNameMap) Add(client *Client) error {
|
|
||||||
if !client.HasNick() {
|
|
||||||
return ErrNickMissing
|
|
||||||
}
|
|
||||||
if clients.Get(client.nick) != nil {
|
|
||||||
return ErrNicknameInUse
|
|
||||||
}
|
|
||||||
clients[strings.ToLower(client.nick)] = client
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (clients ClientNameMap) Remove(client *Client) error {
|
|
||||||
if !client.HasNick() {
|
|
||||||
return ErrNickMissing
|
|
||||||
}
|
|
||||||
if clients.Get(client.nick) != client {
|
|
||||||
return ErrNicknameMismatch
|
|
||||||
}
|
|
||||||
delete(clients, strings.ToLower(client.nick))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type ChannelModeSet map[ChannelMode]bool
|
type ChannelModeSet map[ChannelMode]bool
|
||||||
|
|
||||||
func (set ChannelModeSet) String() string {
|
func (set ChannelModeSet) String() string {
|
||||||
@ -209,17 +173,3 @@ type RegServerCommand interface {
|
|||||||
Command
|
Command
|
||||||
HandleRegServer(*Server)
|
HandleRegServer(*Server)
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
|
||||||
// structs
|
|
||||||
//
|
|
||||||
|
|
||||||
type UserMask struct {
|
|
||||||
nickname Mask
|
|
||||||
username Mask
|
|
||||||
hostname Mask
|
|
||||||
}
|
|
||||||
|
|
||||||
func (mask *UserMask) String() string {
|
|
||||||
return fmt.Sprintf("%s!%s@%s", mask.nickname, mask.username, mask.hostname)
|
|
||||||
}
|
|
||||||
|
Loading…
Reference in New Issue
Block a user