// Copyright 2009 Thomas Jager All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package irc import ( "bufio" "fmt" "net" "strings" "time" "tls" ) const ( VERSION = "cleanirc v1.0" ) var error_ bool func reader(irc *IRCConnection) { br := bufio.NewReader(irc.socket) for !error_ { msg, err := br.ReadString('\n') if err != nil { irc.Error <-err break } irc.lastMessage = time.Now() msg = msg[0 : len(msg)-2] //Remove \r\n event := &IRCEvent{Raw: msg} if msg[0] == ':' { if i := strings.Index(msg, " "); i > -1 { event.Source = msg[1:i] msg = msg[i+1 : len(msg)] } else { irc.log(fmt.Sprintf("Misformed msg from server: %#s\n", msg)) } if i, j := strings.Index(event.Source, "!"), strings.Index(event.Source, "@"); i > -1 && j > -1 { event.Nick = event.Source[0:i] event.User = event.Source[i+1 : j] event.Host = event.Source[j+1 : len(event.Source)] } } args := strings.SplitN(msg, " :", 2) if len(args) > 1 { event.Message = args[1] } args = strings.Split(args[0], " ") event.Code = strings.ToUpper(args[0]) if len(args) > 1 { event.Arguments = args[1:len(args)] } irc.RunCallbacks(event) } irc.syncreader <-true } func writer(irc *IRCConnection) { b, ok := <-irc.pwrite for !error_ && ok { if b == "" || irc.socket == nil { break } _, err := irc.socket.Write([]byte(b)) if err != nil { irc.Error <-err break } b, ok = <-irc.pwrite } irc.syncwriter <-true } //Pings the server if we have not recived any messages for 5 minutes func pinger(i *IRCConnection) { i.ticker = time.Tick(1 * time.Minute) //Tick every minute. i.ticker2 = time.Tick(15 * time.Minute) //Tick every 15 minutes. for { select { case <-i.ticker: //Ping if we haven't recived anything from the server within 4 minutes if time.Since(i.lastMessage) >= (4 * time.Minute) { i.SendRaw(fmt.Sprintf("PING %d", time.Now().UnixNano())) } case <-i.ticker2: //Ping every 15 minutes. i.SendRaw(fmt.Sprintf("PING %d", time.Now().UnixNano())) //Try to recapture nickname if it's not as configured. if i.nick != i.nickcurrent { i.nickcurrent = i.nick i.SendRaw(fmt.Sprintf("NICK %s", i.nick)) } } } } func (irc *IRCConnection) Cycle() { irc.SendRaw("QUIT") irc.Reconnect() } func (irc *IRCConnection) Quit() { irc.quitting = true irc.SendRaw("QUIT") } func (irc *IRCConnection) Join(channel string) { irc.pwrite <-fmt.Sprintf("JOIN %s\r\n", channel) } func (irc *IRCConnection) Part(channel string) { irc.pwrite <-fmt.Sprintf("PART %s\r\n", channel) } func (irc *IRCConnection) Notice(target, message string) { irc.pwrite <-fmt.Sprintf("NOTICE %s :%s\r\n", target, message) } func (irc *IRCConnection) Privmsg(target, message string) { irc.pwrite <-fmt.Sprintf("PRIVMSG %s :%s\r\n", target, message) } func (irc *IRCConnection) SendRaw(message string) { irc.log(fmt.Sprintf("--> %s\n", message)) irc.pwrite <-fmt.Sprintf("%s\r\n", message) } func (i *IRCConnection) Reconnect() error { close(i.pwrite) close(i.pread) <-i.syncreader <-i.syncwriter for { i.log(fmt.Sprintf("Reconnecting to %s\n", i.server)) var err error i.socket, err = net.Dial("tcp", i.server) if err == nil { break } i.log(fmt.Sprintf("Error: %s\n", err)) } error_ = false i.log(fmt.Sprintf("Connected to %s (%s)\n", i.server, i.socket.RemoteAddr())) go reader(i) go writer(i) i.pwrite <-fmt.Sprintf("NICK %s\r\n", i.nick) i.pwrite <-fmt.Sprintf("USER %s 0.0.0.0 0.0.0.0 :%s\r\n", i.user, i.user) return nil } func (i *IRCConnection) Loop() { for !i.quitting { e := <-i.Error if i.quitting { break } i.log(fmt.Sprintf("Error: %s\n", e)) error_ = true i.Reconnect() } close(i.pwrite) close(i.pread) <-i.syncreader <-i.syncwriter } func (i *IRCConnection) postConnect() error { i.pread = make(chan string, 100) i.pwrite = make(chan string, 100) i.Error = make(chan error, 10) i.syncreader = make(chan bool) i.syncwriter = make(chan bool) go reader(i) go writer(i) go pinger(i) if len(i.Password) > 0 { i.pwrite <-fmt.Sprintf("PASS %s\r\n", i.Password) } i.pwrite <-fmt.Sprintf("NICK %s\r\n", i.nick) i.pwrite <-fmt.Sprintf("USER %s 0.0.0.0 0.0.0.0 :%s\r\n", i.user, i.user) return nil } func (i *IRCConnection) Connect(server string) error { i.server = server i.log(fmt.Sprintf("Connecting to %s\n", i.server)) var err error i.socket, err = net.Dial("tcp", i.server) if err != nil { return err } i.log(fmt.Sprintf("Connected to %s (%s)\n", i.server, i.socket.RemoteAddr())) return i.postConnect() } func (i *IRCConnection) ConnectSSL(server string) error { i.server = server i.log(fmt.Sprintf("Connecting to %s over SSL\n", i.server)) var err error i.socket, err = tls.Dial("tcp", i.server, i.SSLConfig) if err != nil { return err } i.log(fmt.Sprintf("Connected to %s (%s) over SSL\n", i.server, i.socket.RemoteAddr())) return i.postConnect() } func (i *IRCConnection) log(msg string) { if i.Log != nil { i.Log <-msg } } func IRC(nick, user string) *IRCConnection { irc := new(IRCConnection) irc.registered = false irc.pread = make(chan string, 100) irc.pwrite = make(chan string, 100) irc.Error = make(chan error) irc.nick = nick irc.user = user irc.VerboseCallbackHandler = true irc.setupCallbacks() return irc }