2020-02-16 15:26:55 +00:00
|
|
|
package irc
|
|
|
|
|
|
|
|
import (
|
2020-02-17 18:02:26 +00:00
|
|
|
"fmt"
|
2020-02-16 21:07:41 +00:00
|
|
|
_ "os"
|
2020-02-16 15:26:55 +00:00
|
|
|
"strings"
|
2020-02-17 18:02:26 +00:00
|
|
|
"time"
|
2020-02-16 15:26:55 +00:00
|
|
|
|
|
|
|
"github.com/lrstanley/girc"
|
2020-02-17 08:41:08 +00:00
|
|
|
|
|
|
|
. "git.deuxfleurs.fr/Deuxfleurs/easybridge/connector"
|
2020-02-16 15:26:55 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// User id format: nickname@server
|
2020-02-16 16:53:31 +00:00
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
// Room id format: #room_name@server
|
|
|
|
|
|
|
|
type IRC struct {
|
|
|
|
handler Handler
|
|
|
|
|
|
|
|
connected bool
|
2020-02-17 18:02:26 +00:00
|
|
|
timeout int
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-17 18:02:26 +00:00
|
|
|
nick string
|
|
|
|
name string
|
2020-02-16 15:26:55 +00:00
|
|
|
server string
|
2020-02-17 18:02:26 +00:00
|
|
|
conn *girc.Client
|
2020-05-04 16:54:45 +00:00
|
|
|
|
|
|
|
joinedRooms map[string]bool
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) SetHandler(h Handler) {
|
|
|
|
irc.handler = h
|
|
|
|
}
|
|
|
|
|
2020-02-17 18:02:26 +00:00
|
|
|
func (irc *IRC) Protocol() string {
|
2020-02-28 09:34:22 +00:00
|
|
|
return IRC_PROTOCOL
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) Configure(c Configuration) error {
|
2020-02-16 15:41:13 +00:00
|
|
|
if irc.conn != nil {
|
|
|
|
irc.Close()
|
|
|
|
}
|
|
|
|
|
2020-02-16 16:53:31 +00:00
|
|
|
var err error
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-16 16:53:31 +00:00
|
|
|
irc.nick, err = c.GetString("nick")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-02-16 16:53:31 +00:00
|
|
|
irc.server, err = c.GetString("server")
|
2020-02-16 15:26:55 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-02-16 16:53:31 +00:00
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
port, err := c.GetInt("port", 6667)
|
2020-02-16 16:53:31 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-16 18:30:49 +00:00
|
|
|
ssl, err := c.GetBool("ssl", false)
|
2020-02-16 16:53:31 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-05-04 16:46:25 +00:00
|
|
|
server_pass, _ := c.GetString("server_pass", "")
|
|
|
|
sasl_user, _ := c.GetString("sasl_user", "")
|
|
|
|
sasl_pass, _ := c.GetString("sasl_pass", "")
|
|
|
|
|
|
|
|
var sasl girc.SASLMech
|
|
|
|
if sasl_user != "" && sasl_pass != "" {
|
|
|
|
sasl = &girc.SASLPlain{
|
|
|
|
User: sasl_user,
|
|
|
|
Pass: sasl_pass,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
client := girc.New(girc.Config{
|
2020-05-04 16:46:25 +00:00
|
|
|
Server: irc.server,
|
|
|
|
ServerPass: server_pass,
|
|
|
|
Port: port,
|
|
|
|
Nick: irc.nick,
|
|
|
|
User: irc.nick,
|
|
|
|
SASL: sasl,
|
2020-02-16 21:07:41 +00:00
|
|
|
//Out: os.Stderr,
|
2020-02-16 16:53:31 +00:00
|
|
|
SSL: ssl,
|
2020-02-16 15:26:55 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
client.Handlers.Add(girc.CONNECTED, irc.ircConnected)
|
|
|
|
//client.Handlers.Add(girc.DISCONNECTED, irc.ircDisconnected)
|
|
|
|
//client.Handlers.Add(girc.NICK, irc.ircNick)
|
|
|
|
client.Handlers.Add(girc.PRIVMSG, irc.ircPrivmsg)
|
|
|
|
client.Handlers.Add(girc.JOIN, irc.ircJoin)
|
|
|
|
client.Handlers.Add(girc.PART, irc.ircPart)
|
|
|
|
client.Handlers.Add(girc.RPL_NAMREPLY, irc.ircNamreply)
|
2020-02-17 14:30:01 +00:00
|
|
|
client.Handlers.Add(girc.TOPIC, irc.ircTopic)
|
|
|
|
client.Handlers.Add(girc.RPL_TOPIC, irc.ircRplTopic)
|
2020-02-16 15:26:55 +00:00
|
|
|
|
2020-05-04 16:54:45 +00:00
|
|
|
irc.joinedRooms = make(map[string]bool)
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
irc.conn = client
|
|
|
|
go irc.connectLoop(client)
|
|
|
|
|
|
|
|
for i := 0; i < 42; i++ {
|
2020-02-17 18:02:26 +00:00
|
|
|
time.Sleep(time.Duration(1) * time.Second)
|
2020-02-16 15:26:55 +00:00
|
|
|
if irc.conn != client {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if irc.connected {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 16:53:31 +00:00
|
|
|
return fmt.Errorf("Failed to connect after 42s attempting")
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) User() UserID {
|
|
|
|
return UserID(irc.nick + "@" + irc.server)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) checkRoomId(id RoomID) (string, error) {
|
|
|
|
x := strings.Split(string(id), "@")
|
2020-02-17 20:19:28 +00:00
|
|
|
if len(x) == 1 {
|
|
|
|
return "", fmt.Errorf("Please write whole room ID with server: %s@%s", id, irc.server)
|
|
|
|
}
|
|
|
|
if x[0][0] != '#' || len(x) != 2 || x[1] != irc.server {
|
2020-02-16 15:26:55 +00:00
|
|
|
return "", fmt.Errorf("Invalid room ID: %s", id)
|
|
|
|
}
|
|
|
|
return x[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) checkUserId(id UserID) (string, error) {
|
|
|
|
x := strings.Split(string(id), "@")
|
2020-02-17 20:19:28 +00:00
|
|
|
if len(x) == 1 {
|
|
|
|
return "", fmt.Errorf("Please write whole user ID with server: %s@%s", id, irc.server)
|
|
|
|
}
|
|
|
|
if x[0][0] == '#' || len(x) != 2 || x[1] != irc.server {
|
2020-02-16 15:26:55 +00:00
|
|
|
return "", fmt.Errorf("Invalid user ID: %s", id)
|
|
|
|
}
|
|
|
|
return x[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) SetUserInfo(info *UserInfo) error {
|
|
|
|
return fmt.Errorf("Not implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) SetRoomInfo(roomId RoomID, info *RoomInfo) error {
|
2020-02-26 21:49:27 +00:00
|
|
|
if irc.conn == nil {
|
|
|
|
return fmt.Errorf("Not connected")
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
ch, err := irc.checkRoomId(roomId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-17 18:02:00 +00:00
|
|
|
if info.Topic != "" {
|
|
|
|
irc.conn.Cmd.Topic(ch, info.Topic)
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
if info.Name != "" && info.Name != ch {
|
|
|
|
return fmt.Errorf("May not change IRC room name to other than %s", ch)
|
|
|
|
}
|
2020-02-29 19:47:44 +00:00
|
|
|
if info.Picture.MediaObject != nil {
|
2020-02-16 15:26:55 +00:00
|
|
|
return fmt.Errorf("Room picture not supported on IRC")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) Join(roomId RoomID) error {
|
2020-02-26 21:49:27 +00:00
|
|
|
if irc.conn == nil {
|
|
|
|
return fmt.Errorf("Not connected")
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
ch, err := irc.checkRoomId(roomId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
irc.conn.Cmd.Join(ch)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:41:13 +00:00
|
|
|
func (irc *IRC) Invite(userId UserID, roomId RoomID) error {
|
2020-02-26 21:49:27 +00:00
|
|
|
if irc.conn == nil {
|
|
|
|
return fmt.Errorf("Not connected")
|
|
|
|
}
|
|
|
|
|
2020-02-17 20:19:28 +00:00
|
|
|
who, err := irc.checkUserId(userId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-17 20:04:21 +00:00
|
|
|
if roomId == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:41:13 +00:00
|
|
|
ch, err := irc.checkRoomId(roomId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
irc.conn.Cmd.Invite(ch, who)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
func (irc *IRC) Leave(roomId RoomID) {
|
2020-02-26 21:49:27 +00:00
|
|
|
if irc.conn == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
ch, err := irc.checkRoomId(roomId)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
irc.conn.Cmd.Part(ch)
|
|
|
|
}
|
|
|
|
|
2020-03-09 16:41:53 +00:00
|
|
|
func (irc *IRC) SearchForUsers(query string) ([]UserSearchResult, error) {
|
|
|
|
// TODO
|
|
|
|
return nil, fmt.Errorf("Not implemented")
|
|
|
|
}
|
|
|
|
|
2020-02-29 09:01:42 +00:00
|
|
|
func (irc *IRC) Send(event *Event) (string, error) {
|
2020-02-26 21:49:27 +00:00
|
|
|
if irc.conn == nil {
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", fmt.Errorf("Not connected")
|
2020-02-26 21:49:27 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 22:27:03 +00:00
|
|
|
// Workaround girc bug
|
|
|
|
if event.Text[0] == ':' {
|
|
|
|
event.Text = " " + event.Text
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
dest := ""
|
|
|
|
if event.Room != "" {
|
|
|
|
ch, err := irc.checkRoomId(event.Room)
|
|
|
|
if err != nil {
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", err
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
dest = ch
|
|
|
|
} else if event.Recipient != "" {
|
|
|
|
ui, err := irc.checkUserId(event.Recipient)
|
|
|
|
if err != nil {
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", err
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
dest = ui
|
|
|
|
} else {
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", fmt.Errorf("Invalid target")
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-02-21 17:08:40 +00:00
|
|
|
if event.Attachments != nil && len(event.Attachments) > 0 {
|
|
|
|
for _, at := range event.Attachments {
|
|
|
|
url := at.URL()
|
|
|
|
if url == "" {
|
|
|
|
// TODO find a way to send them using some hosing of some kind
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", fmt.Errorf("Attachment without URL sent to IRC")
|
2020-02-21 17:08:40 +00:00
|
|
|
} else {
|
|
|
|
irc.conn.Cmd.Message(dest, fmt.Sprintf("%s (%s, %dkb)",
|
|
|
|
url, at.Mimetype(), at.Size()/1024))
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if event.Type == EVENT_MESSAGE {
|
2020-02-16 16:53:31 +00:00
|
|
|
irc.conn.Cmd.Message(dest, event.Text)
|
2020-02-16 15:26:55 +00:00
|
|
|
} else if event.Type == EVENT_ACTION {
|
2020-02-16 16:53:31 +00:00
|
|
|
irc.conn.Cmd.Action(dest, event.Text)
|
2020-02-16 15:26:55 +00:00
|
|
|
} else {
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", fmt.Errorf("Invalid event type")
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
2020-02-29 09:01:42 +00:00
|
|
|
return "", nil
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
2020-10-04 19:20:22 +00:00
|
|
|
func (irc *IRC) UserCommand(cm string) {
|
|
|
|
irc.handler.SystemMessage("Command not supported.")
|
|
|
|
}
|
|
|
|
|
2020-02-16 15:26:55 +00:00
|
|
|
func (irc *IRC) Close() {
|
2020-02-16 16:53:31 +00:00
|
|
|
conn := irc.conn
|
2020-02-16 15:26:55 +00:00
|
|
|
irc.conn = nil
|
2020-02-26 22:01:34 +00:00
|
|
|
irc.connected = false
|
2020-02-26 21:49:27 +00:00
|
|
|
if conn != nil {
|
|
|
|
conn.Close()
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) connectLoop(c *girc.Client) {
|
|
|
|
irc.timeout = 10
|
|
|
|
for {
|
|
|
|
if irc.conn != c {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err := c.Connect(); err != nil {
|
|
|
|
irc.connected = false
|
2020-05-04 16:36:55 +00:00
|
|
|
irc.handler.SystemMessage(fmt.Sprintf("IRC failed to connect / disconnected (%s), reconnecting in %ds", err, irc.timeout))
|
2020-02-16 15:26:55 +00:00
|
|
|
time.Sleep(time.Duration(irc.timeout) * time.Second)
|
|
|
|
irc.timeout *= 2
|
2020-02-16 16:53:31 +00:00
|
|
|
if irc.timeout > 600 {
|
|
|
|
irc.timeout = 600
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
} else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircConnected(c *girc.Client, e girc.Event) {
|
2020-05-04 16:36:55 +00:00
|
|
|
irc.handler.SystemMessage("Connected to IRC.")
|
2020-02-16 15:26:55 +00:00
|
|
|
irc.timeout = 10
|
|
|
|
irc.connected = true
|
2020-05-04 16:54:45 +00:00
|
|
|
|
|
|
|
for room, joined := range irc.joinedRooms {
|
|
|
|
if joined {
|
|
|
|
irc.conn.Cmd.Join(room)
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircPrivmsg(c *girc.Client, e girc.Event) {
|
|
|
|
ev := &Event{
|
2020-02-17 18:02:26 +00:00
|
|
|
Type: EVENT_MESSAGE,
|
2020-02-16 15:26:55 +00:00
|
|
|
Author: UserID(e.Source.Name + "@" + irc.server),
|
2020-02-17 18:02:26 +00:00
|
|
|
Text: e.Last(),
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
if e.IsFromChannel() {
|
|
|
|
ev.Room = RoomID(e.Params[0] + "@" + irc.server)
|
|
|
|
}
|
|
|
|
if e.IsAction() {
|
|
|
|
ev.Type = EVENT_ACTION
|
|
|
|
}
|
|
|
|
irc.handler.Event(ev)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircJoin(c *girc.Client, e girc.Event) {
|
|
|
|
room := RoomID(e.Params[0] + "@" + irc.server)
|
|
|
|
if e.Source.Name == irc.nick {
|
|
|
|
irc.handler.Joined(room)
|
2020-05-04 16:54:45 +00:00
|
|
|
irc.joinedRooms[e.Params[0]] = true
|
2020-02-16 15:26:55 +00:00
|
|
|
} else {
|
2020-02-17 15:28:32 +00:00
|
|
|
user := UserID(e.Source.Name + "@" + irc.server)
|
2020-02-16 15:26:55 +00:00
|
|
|
ev := &Event{
|
2020-02-17 18:02:26 +00:00
|
|
|
Type: EVENT_JOIN,
|
2020-02-17 15:28:32 +00:00
|
|
|
Author: user,
|
2020-02-17 18:02:26 +00:00
|
|
|
Room: room,
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
irc.handler.Event(ev)
|
2020-02-17 15:28:32 +00:00
|
|
|
irc.handler.UserInfoUpdated(user, &UserInfo{
|
|
|
|
DisplayName: e.Source.Name,
|
|
|
|
})
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircPart(c *girc.Client, e girc.Event) {
|
|
|
|
room := RoomID(e.Params[0] + "@" + irc.server)
|
|
|
|
if e.Source.Name == irc.nick {
|
|
|
|
irc.handler.Left(room)
|
2020-05-04 16:54:45 +00:00
|
|
|
delete(irc.joinedRooms, e.Params[0])
|
2020-02-16 15:26:55 +00:00
|
|
|
} else {
|
2020-02-17 15:28:32 +00:00
|
|
|
user := UserID(e.Source.Name + "@" + irc.server)
|
2020-02-16 15:26:55 +00:00
|
|
|
ev := &Event{
|
2020-02-17 18:02:26 +00:00
|
|
|
Type: EVENT_LEAVE,
|
2020-02-17 15:28:32 +00:00
|
|
|
Author: user,
|
2020-02-17 18:02:26 +00:00
|
|
|
Room: room,
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
irc.handler.Event(ev)
|
2020-02-17 15:28:32 +00:00
|
|
|
irc.handler.UserInfoUpdated(user, &UserInfo{
|
|
|
|
DisplayName: e.Source.Name,
|
|
|
|
})
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircNamreply(c *girc.Client, e girc.Event) {
|
2020-02-16 15:41:13 +00:00
|
|
|
room := RoomID(e.Params[2] + "@" + irc.server)
|
|
|
|
names := strings.Split(e.Last(), " ")
|
|
|
|
for _, name := range names {
|
|
|
|
if name[0] == '@' {
|
|
|
|
name = name[1:]
|
|
|
|
}
|
|
|
|
src := girc.ParseSource(name)
|
|
|
|
if src.Name != irc.nick {
|
|
|
|
irc.handler.Event(&Event{
|
2020-02-17 18:02:26 +00:00
|
|
|
Type: EVENT_JOIN,
|
2020-02-16 15:41:13 +00:00
|
|
|
Author: UserID(src.Name + "@" + irc.server),
|
2020-02-17 18:02:26 +00:00
|
|
|
Room: room,
|
2020-02-16 15:41:13 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircTopic(c *girc.Client, e girc.Event) {
|
2020-02-17 14:30:01 +00:00
|
|
|
source := UserID(e.Source.Name + "@" + irc.server)
|
|
|
|
room := RoomID(e.Params[0] + "@" + irc.server)
|
|
|
|
topic := e.Last()
|
|
|
|
irc.handler.RoomInfoUpdated(room, source, &RoomInfo{
|
|
|
|
Topic: topic,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (irc *IRC) ircRplTopic(c *girc.Client, e girc.Event) {
|
2020-02-16 15:41:13 +00:00
|
|
|
room := RoomID(e.Params[1] + "@" + irc.server)
|
|
|
|
topic := e.Last()
|
2020-02-17 14:30:01 +00:00
|
|
|
irc.handler.RoomInfoUpdated(room, "", &RoomInfo{
|
|
|
|
Topic: topic,
|
2020-02-16 15:41:13 +00:00
|
|
|
})
|
2020-02-16 15:26:55 +00:00
|
|
|
}
|