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.
 

206 lines
5.2 KiB

package main
import (
"crypto/tls"
"flag"
"fmt"
"goqdpmbot/events_alerter"
"goqdpmbot/works"
"os"
"path"
"path/filepath"
// "runtime"
// "strconv"
"strings"
"sync"
"time"
"github.com/mattn/go-xmpp"
"gopkg.in/ini.v1"
)
type cfgStruct struct {
srv string
usr string
pwd string
dest string
notls bool
}
type commandMsg struct {
remote string
message string
}
func serverName(host string) string {
return strings.Split(host, ":")[0]
}
func cfgRead(CfgFile string) cfgStruct {
dir, dir_err := filepath.Abs(filepath.Dir(os.Args[0]))
var ret cfgStruct
if dir_err != nil {
fmt.Printf("fail to open dir: %v", dir_err)
}
cfg, cfg_err := ini.Load(path.Join(dir, CfgFile))
if cfg_err != nil {
fmt.Printf("Fail to read file: %v", cfg_err)
os.Exit(1)
}
cfg_notls, _ := cfg.Section("").Key("notls").Bool()
ret.srv = cfg.Section("").Key("server").String()
ret.usr = cfg.Section("").Key("username").String()
ret.pwd = cfg.Section("").Key("password").String()
ret.notls = cfg_notls
ret.dest = cfg.Section("").Key("dest").String()
return ret
}
func jabberClient(cfg cfgStruct) *xmpp.Client {
var server = flag.String("server", cfg.srv, "server")
var username = flag.String("username", cfg.usr, "username")
var password = flag.String("password", cfg.pwd, "password")
var status = flag.String("status", "xa", "status")
var statusMessage = flag.String("status-msg", "", "")
var notls = flag.Bool("notls", cfg.notls, "No TLS")
var debug = flag.Bool("debug", false, "debug output")
var session = flag.Bool("session", false, "use server session")
flag.Usage = func() {
fmt.Fprintf(os.Stderr, "usage: example [options]\n")
flag.PrintDefaults()
os.Exit(2)
}
flag.Parse()
if *username == "" || *password == "" {
if *debug && *username == "" && *password == "" {
fmt.Fprintf(os.Stderr, "no username or password were given; attempting ANONYMOUS auth\n")
} else if *username != "" || *password != "" {
flag.Usage()
}
}
if !*notls {
xmpp.DefaultConfig = tls.Config{
ServerName: serverName(*server),
InsecureSkipVerify: false,
}
}
var talk *xmpp.Client
var err error
options := xmpp.Options{Host: *server,
User: *username,
Password: *password,
NoTLS: *notls,
Debug: *debug,
Session: *session,
Status: *status,
StatusMessage: *statusMessage,
}
talk, err = options.NewClient()
if err != nil {
fmt.Println(err)
}
return talk
}
// читаем из джаббера и отправляем в канал
func jabberReader(talk *xmpp.Client, command chan<- commandMsg, in <-chan int) {
for {
chat, err := talk.Recv()
if err != nil {
fmt.Println(err)
}
switch v := chat.(type) {
case xmpp.Chat:
com := commandMsg{remote: v.Remote, message: v.Text}
command <- com
//case xmpp.Presence:
// com := commandMsg{remote: v.From, message: v.Show}
// command <- com
}
}
}
// отправляем ответы из канала
func jabberSender(talk *xmpp.Client, msgChan <-chan commandMsg) {
for {
msg := <-msgChan
remote := strings.Split(msg.remote, "/")[0]
talk.Send(xmpp.Chat{Remote: remote, Type: "chat", Text: msg.message})
}
}
// воркер
func worker(workerNum int, msgChan <-chan commandMsg, readerClose chan int, ansChan chan<- commandMsg) {
for {
message := <-msgChan
fmt.Println("Worker = ", workerNum, "message= ", message.message)
cmd := strings.Split(message.message, " ")
workMap := works.Works()
_, workExists := workMap[cmd[0]]
if workExists {
work, _ := workMap[cmd[0]]
ans, err := work.(func(...string) (string, error))(cmd[1:]...)
if err != nil {
ansChan <- commandMsg{remote: message.remote, message: err.Error()}
} else {
ansChan <- commandMsg{remote: message.remote, message: ans}
}
} else if len(message.message) > 0 {
ans := "неизвестная комманда"
ansChan <- commandMsg{remote: message.remote, message: ans}
}
}
}
//нопоминальщик
func reminder(readerClose chan int, ansChan chan<- commandMsg) {
for {
msgSl := events_alerter.BuildAlert()
fmt.Println("alerter works")
fmt.Println(msgSl)
for _, msgInt := range msgSl {
tmpMsg := commandMsg{remote: msgInt.Remote(), message: msgInt.Message()}
fmt.Println(tmpMsg)
ansChan <- tmpMsg
}
time.Sleep(60 * time.Second)
}
}
func main() {
cfg := cfgRead("goqdpmbot.ini") // читаем конфигу
goroutinesNum := 5 // количество потоков
wg := &sync.WaitGroup{} // ожидаем интеррапта
signal_channel := make(chan os.Signal, 1)
readerClose := make(chan int, 1) // для закрытия всего зоопарка воркеров
// каналы обмена инфой
comChan := make(chan commandMsg, 5)
ansChan := make(chan commandMsg, 5)
// запускаем клиента
talk := jabberClient(cfg)
go jabberReader(talk, comChan, readerClose)
go jabberSender(talk, ansChan)
// и воркеров
for i := 0; i < goroutinesNum; i++ {
go worker(i, comChan, readerClose, ansChan)
fmt.Println("worker started", i)
}
go reminder(readerClose, ansChan)
// ожидаем выхода
wg.Add(1)
go func() {
<-signal_channel
wg.Done()
}()
wg.Wait()
close(readerClose)
time.Sleep(1 * time.Second)
fmt.Println("interrupt")
talk.Close()
}

Powered by TurnKey Linux.