2017-08-28 01:13:45 +05:00
|
|
|
// OpenSAPS - Open Slack API server for everyone.
|
|
|
|
//
|
|
|
|
// Copyright (c) 2017, Stanislav N. aka pztrn.
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package slack
|
|
|
|
|
|
|
|
import (
|
2018-02-02 09:17:40 +05:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2019-12-15 22:17:01 +05:00
|
|
|
slackmessage "go.dev.pztrn.name/opensaps/slack/message"
|
2017-08-28 01:13:45 +05:00
|
|
|
)
|
|
|
|
|
2019-12-15 22:17:01 +05:00
|
|
|
type Handler struct{}
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2019-12-15 22:17:01 +05:00
|
|
|
func (sh Handler) ServeHTTP(respwriter http.ResponseWriter, req *http.Request) {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Str("method", req.Method).Str("host", req.Host).Str("path", req.URL.Path).Msg("Received HTTP request")
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
// We should catch only POST requests. Otherwise return HTTP 404.
|
|
|
|
if req.Method != "POST" {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Msg("Not a POST request, returning HTTP 404")
|
|
|
|
respwriter.WriteHeader(http.StatusNotFound)
|
2018-02-02 09:17:40 +05:00
|
|
|
fmt.Fprintf(respwriter, "NOT FOUND")
|
2019-12-15 22:17:01 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
return
|
|
|
|
}
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
body, _ := ioutil.ReadAll(req.Body)
|
|
|
|
req.Body.Close()
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Msgf("Received body: %s", string(body))
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
// Try to figure out where we should push received data.
|
|
|
|
cfg := c.Config.GetConfig()
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2021-11-21 16:16:10 +05:00
|
|
|
var sentToPusher bool
|
2019-12-15 22:17:01 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
for name, config := range cfg.Webhooks {
|
2019-12-15 22:17:01 +05:00
|
|
|
if strings.Contains(req.URL.Path, config.Slack.Random1) &&
|
|
|
|
strings.Contains(req.URL.Path, config.Slack.Random2) &&
|
|
|
|
strings.Contains(req.URL.Path, config.Slack.LongRandom) {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Msgf("Passed data belongs to '%s' and should go to '%s' pusher, protocol '%s'",
|
2019-12-15 22:17:01 +05:00
|
|
|
name, config.Remote.PushTo, config.Remote.Pusher)
|
2018-02-02 09:17:40 +05:00
|
|
|
// Parse message into SlackMessage structure.
|
|
|
|
if strings.Contains(string(body)[0:7], "payload") {
|
|
|
|
// We have HTTP form payload. It still should be a
|
|
|
|
// parseable JSON string, we just need to do some
|
|
|
|
// preparations.
|
|
|
|
// First - remove "payload=" from the beginning.
|
2019-12-15 22:17:01 +05:00
|
|
|
tempBody := string(body)
|
|
|
|
tempBody = strings.Replace(tempBody, "payload=", "", 1)
|
2018-02-02 09:17:40 +05:00
|
|
|
// Second - unescape data.
|
2019-12-15 22:17:01 +05:00
|
|
|
tempBody, err := url.QueryUnescape(tempBody)
|
2018-02-02 09:17:40 +05:00
|
|
|
if err != nil {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Error().Msg("Failed to decode body into parseable string!")
|
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
return
|
|
|
|
}
|
2017-09-13 11:44:25 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
// And finally - convert body back to bytes.
|
2019-12-15 22:17:01 +05:00
|
|
|
body = []byte(tempBody)
|
2018-02-02 09:17:40 +05:00
|
|
|
}
|
2019-12-15 22:17:01 +05:00
|
|
|
|
2021-11-21 16:16:10 +05:00
|
|
|
// nolint:exhaustivestruct
|
2018-02-02 09:17:40 +05:00
|
|
|
slackmsg := slackmessage.SlackMessage{}
|
2019-12-15 22:17:01 +05:00
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
err := json.Unmarshal(body, &slackmsg)
|
|
|
|
if err != nil {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Error().Err(err).Msg("Failed to decode JSON into SlackMessage struct")
|
|
|
|
|
2018-02-02 09:17:40 +05:00
|
|
|
return
|
|
|
|
}
|
2019-12-15 22:17:01 +05:00
|
|
|
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Msgf("Received message: %+v", slackmsg)
|
2018-02-02 09:17:40 +05:00
|
|
|
c.SendToPusher(config.Remote.Pusher, config.Remote.PushTo, slackmsg)
|
2019-12-15 22:17:01 +05:00
|
|
|
|
|
|
|
sentToPusher = true
|
2018-02-02 09:17:40 +05:00
|
|
|
}
|
|
|
|
}
|
2017-08-28 01:13:45 +05:00
|
|
|
|
2019-12-15 22:17:01 +05:00
|
|
|
if !sentToPusher {
|
2021-11-21 16:16:10 +05:00
|
|
|
c.Log.Debug().Msg("Don't know where to push data. Ignoring with HTTP 404")
|
|
|
|
respwriter.WriteHeader(http.StatusNotFound)
|
2018-02-02 09:17:40 +05:00
|
|
|
fmt.Fprintf(respwriter, "NOT FOUND")
|
|
|
|
}
|
2017-08-28 01:13:45 +05:00
|
|
|
}
|