package main import ( "crypto/tls" "fmt" "log" "net/http" "os" "time" "github.com/labstack/echo/v5" "github.com/lrstanley/girc" "github.com/pelletier/go-toml/v2" "github.com/pocketbase/pocketbase" "github.com/pocketbase/pocketbase/core" ) const ( reconnectTime = 30 ) type handlerWrapper struct { irc *girc.Client // irc chan *girc.Client config TomlConfig } type SMS struct { From string `json:"from"` Text string `json:"text"` SentStamp int64 `json:"sentStamp"` ReceivedStamp int64 `json:"receivedStamp"` Sim string `json:"sim"` } type TomlConfig struct { IrcServer string IrcPort int IrcNick string IrcSaslUser string IrcSaslPass string IrcChannel string } // curl -X 'POST' 'http://127.0.0.1:8090/sms' -H 'content-type: application/json; charset=utf-8' -d $'{"from":"1234567890","text":"Test"}' func (hw handlerWrapper) postHandler(context echo.Context) error { sms := new(SMS) if err := context.Bind(sms); err != nil { return context.String(http.StatusBadRequest, "bad request") } smsInfoReal := SMS{ From: sms.From, Text: sms.Text, SentStamp: sms.SentStamp, ReceivedStamp: sms.ReceivedStamp, Sim: sms.Sim, } for { if hw.irc != nil { if hw.irc.IsConnected() { break } } } hw.irc.Cmd.Message(hw.config.IrcChannel, fmt.Sprintf("From: %s, Text: %s", sms.From, sms.Text)) log.Println(smsInfoReal) return context.JSON(http.StatusOK, "OK") } func runIRC(appConfig TomlConfig, ircChan chan *girc.Client) { irc := girc.New(girc.Config{ Server: appConfig.IrcServer, Port: appConfig.IrcPort, Nick: appConfig.IrcNick, User: appConfig.IrcNick, Name: appConfig.IrcNick, SSL: true, TLSConfig: &tls.Config{InsecureSkipVerify: true}, }) saslUser := appConfig.IrcSaslUser saslPass := appConfig.IrcSaslPass if saslUser != "" && saslPass != "" { irc.Config.SASL = &girc.SASLPlain{ User: appConfig.IrcSaslUser, Pass: appConfig.IrcSaslPass, } } irc.Handlers.AddBg(girc.CONNECTED, func(c *girc.Client, e girc.Event) { c.Cmd.Join(appConfig.IrcChannel) }) // irc.Handlers.AddBg(girc.PRIVMSG, func(c *girc.Client, e girc.Event) {}) ircChan <- irc for { if err := irc.Connect(); err != nil { log.Println(err) log.Println("reconnecting in 30 seconds") time.Sleep(reconnectTime * time.Second) } else { return } } } func main() { var appConfig TomlConfig data, err := os.ReadFile("/opt/smswebhook/config.toml") if err != nil { log.Fatal(err) } err = toml.Unmarshal(data, &appConfig) if err != nil { log.Fatal(err) } log.Println(appConfig) app := pocketbase.New() ircChan := make(chan *girc.Client, 1) hw := handlerWrapper{irc: nil, config: appConfig} app.OnBeforeServe().Add(func(e *core.ServeEvent) error { go runIRC(appConfig, ircChan) hw.irc = <-ircChan e.Router.POST("/sms", hw.postHandler) return nil }) if err := app.Start(); err != nil { log.Fatal(err) } }