267 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package gateway
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"crypto/sha1" //nolint:gosec
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"net/http"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"regexp"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/42wim/matterbridge/bridge"
 | |
| 	"github.com/42wim/matterbridge/bridge/config"
 | |
| )
 | |
| 
 | |
| // handleEventFailure handles failures and reconnects bridges.
 | |
| func (r *Router) handleEventFailure(msg *config.Message) {
 | |
| 	if msg.Event != config.EventFailure {
 | |
| 		return
 | |
| 	}
 | |
| 	for _, gw := range r.Gateways {
 | |
| 		for _, br := range gw.Bridges {
 | |
| 			if msg.Account == br.Account {
 | |
| 				go gw.reconnectBridge(br)
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // handleEventGetChannelMembers handles channel members
 | |
| func (r *Router) handleEventGetChannelMembers(msg *config.Message) {
 | |
| 	if msg.Event != config.EventGetChannelMembers {
 | |
| 		return
 | |
| 	}
 | |
| 	for _, gw := range r.Gateways {
 | |
| 		for _, br := range gw.Bridges {
 | |
| 			if msg.Account == br.Account {
 | |
| 				cMembers := msg.Extra[config.EventGetChannelMembers][0].(config.ChannelMembers)
 | |
| 				r.logger.Debugf("Syncing channelmembers from %s", msg.Account)
 | |
| 				br.SetChannelMembers(&cMembers)
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // handleEventRejoinChannels handles rejoining of channels.
 | |
| func (r *Router) handleEventRejoinChannels(msg *config.Message) {
 | |
| 	if msg.Event != config.EventRejoinChannels {
 | |
| 		return
 | |
| 	}
 | |
| 	for _, gw := range r.Gateways {
 | |
| 		for _, br := range gw.Bridges {
 | |
| 			if msg.Account == br.Account {
 | |
| 				br.Joined = make(map[string]bool)
 | |
| 				if err := br.JoinChannels(); err != nil {
 | |
| 					r.logger.Errorf("channel join failed for %s: %s", msg.Account, err)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // handleFiles uploads or places all files on the given msg to the MediaServer and
 | |
| // adds the new URL of the file on the MediaServer onto the given msg.
 | |
| func (gw *Gateway) handleFiles(msg *config.Message) {
 | |
| 	reg := regexp.MustCompile("[^a-zA-Z0-9]+")
 | |
| 
 | |
| 	// If we don't have a attachfield or we don't have a mediaserver configured return
 | |
| 	if msg.Extra == nil ||
 | |
| 		(gw.BridgeValues().General.MediaServerUpload == "" &&
 | |
| 			gw.BridgeValues().General.MediaDownloadPath == "") {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// If we don't have files, nothing to upload.
 | |
| 	if len(msg.Extra["file"]) == 0 {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	for i, f := range msg.Extra["file"] {
 | |
| 		fi := f.(config.FileInfo)
 | |
| 		ext := filepath.Ext(fi.Name)
 | |
| 		fi.Name = fi.Name[0 : len(fi.Name)-len(ext)]
 | |
| 		fi.Name = reg.ReplaceAllString(fi.Name, "_")
 | |
| 		fi.Name += ext
 | |
| 
 | |
| 		sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
 | |
| 
 | |
| 		if gw.BridgeValues().General.MediaServerUpload != "" {
 | |
| 			// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth.
 | |
| 			if err := gw.handleFilesUpload(&fi); err != nil {
 | |
| 				gw.logger.Error(err)
 | |
| 				continue
 | |
| 			}
 | |
| 		} else {
 | |
| 			// Use MediaServerPath. Place the file on the current filesystem.
 | |
| 			if err := gw.handleFilesLocal(&fi); err != nil {
 | |
| 				gw.logger.Error(err)
 | |
| 				continue
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Download URL.
 | |
| 		durl := gw.BridgeValues().General.MediaServerDownload + "/" + sha1sum + "/" + fi.Name
 | |
| 
 | |
| 		gw.logger.Debugf("mediaserver download URL = %s", durl)
 | |
| 
 | |
| 		// We uploaded/placed the file successfully. Add the SHA and URL.
 | |
| 		extra := msg.Extra["file"][i].(config.FileInfo)
 | |
| 		extra.URL = durl
 | |
| 		extra.SHA = sha1sum
 | |
| 		msg.Extra["file"][i] = extra
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // handleFilesUpload uses MediaServerUpload configuration to upload the file.
 | |
| // Returns error on failure.
 | |
| func (gw *Gateway) handleFilesUpload(fi *config.FileInfo) error {
 | |
| 	client := &http.Client{
 | |
| 		Timeout: time.Second * 5,
 | |
| 	}
 | |
| 	// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth.
 | |
| 	sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
 | |
| 	url := gw.BridgeValues().General.MediaServerUpload + "/" + sha1sum + "/" + fi.Name
 | |
| 
 | |
| 	req, err := http.NewRequest("PUT", url, bytes.NewReader(*fi.Data))
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("mediaserver upload failed, could not create request: %#v", err)
 | |
| 	}
 | |
| 
 | |
| 	gw.logger.Debugf("mediaserver upload url: %s", url)
 | |
| 
 | |
| 	req.Header.Set("Content-Type", "binary/octet-stream")
 | |
| 	_, err = client.Do(req)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("mediaserver upload failed, could not Do request: %#v", err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // handleFilesLocal use MediaServerPath configuration, places the file on the current filesystem.
 | |
| // Returns error on failure.
 | |
| func (gw *Gateway) handleFilesLocal(fi *config.FileInfo) error {
 | |
| 	sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] //nolint:gosec
 | |
| 	dir := gw.BridgeValues().General.MediaDownloadPath + "/" + sha1sum
 | |
| 	err := os.Mkdir(dir, os.ModePerm)
 | |
| 	if err != nil && !os.IsExist(err) {
 | |
| 		return fmt.Errorf("mediaserver path failed, could not mkdir: %s %#v", err, err)
 | |
| 	}
 | |
| 
 | |
| 	path := dir + "/" + fi.Name
 | |
| 	gw.logger.Debugf("mediaserver path placing file: %s", path)
 | |
| 
 | |
| 	err = ioutil.WriteFile(path, *fi.Data, os.ModePerm)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("mediaserver path failed, could not writefile: %s %#v", err, err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // ignoreEvent returns true if we need to ignore this event for the specified destination bridge.
 | |
| func (gw *Gateway) ignoreEvent(event string, dest *bridge.Bridge) bool {
 | |
| 	switch event {
 | |
| 	case config.EventAvatarDownload:
 | |
| 		// Avatar downloads are only relevant for telegram and mattermost for now
 | |
| 		if dest.Protocol != "mattermost" && dest.Protocol != "telegram" {
 | |
| 			return true
 | |
| 		}
 | |
| 	case config.EventJoinLeave:
 | |
| 		// only relay join/part when configured
 | |
| 		if !dest.GetBool("ShowJoinPart") {
 | |
| 			return true
 | |
| 		}
 | |
| 	case config.EventTopicChange:
 | |
| 		// only relay topic change when used in some way on other side
 | |
| 		if dest.GetBool("ShowTopicChange") && dest.GetBool("SyncTopic") {
 | |
| 			return true
 | |
| 		}
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // handleMessage makes sure the message get sent to the correct bridge/channels.
 | |
| // Returns an array of msg ID's
 | |
| func (gw *Gateway) handleMessage(rmsg *config.Message, dest *bridge.Bridge) []*BrMsgID {
 | |
| 	var brMsgIDs []*BrMsgID
 | |
| 
 | |
| 	// if we have an attached file, or other info
 | |
| 	if rmsg.Extra != nil && len(rmsg.Extra[config.EventFileFailureSize]) != 0 && rmsg.Text == "" {
 | |
| 		return brMsgIDs
 | |
| 	}
 | |
| 
 | |
| 	if gw.ignoreEvent(rmsg.Event, dest) {
 | |
| 		return brMsgIDs
 | |
| 	}
 | |
| 
 | |
| 	// broadcast to every out channel (irc QUIT)
 | |
| 	if rmsg.Channel == "" && rmsg.Event != config.EventJoinLeave {
 | |
| 		gw.logger.Debug("empty channel")
 | |
| 		return brMsgIDs
 | |
| 	}
 | |
| 
 | |
| 	// Get the ID of the parent message in thread
 | |
| 	var canonicalParentMsgID string
 | |
| 	if rmsg.ParentID != "" && dest.GetBool("PreserveThreading") {
 | |
| 		canonicalParentMsgID = gw.FindCanonicalMsgID(rmsg.Protocol, rmsg.ParentID)
 | |
| 	}
 | |
| 
 | |
| 	channels := gw.getDestChannel(rmsg, *dest)
 | |
| 	for idx := range channels {
 | |
| 		channel := &channels[idx]
 | |
| 		msgID, err := gw.SendMessage(rmsg, dest, channel, canonicalParentMsgID)
 | |
| 		if err != nil {
 | |
| 			gw.logger.Errorf("SendMessage failed: %s", err)
 | |
| 			continue
 | |
| 		}
 | |
| 		if msgID == "" {
 | |
| 			continue
 | |
| 		}
 | |
| 		brMsgIDs = append(brMsgIDs, &BrMsgID{dest, dest.Protocol + " " + msgID, channel.ID})
 | |
| 	}
 | |
| 	return brMsgIDs
 | |
| }
 | |
| 
 | |
| func (gw *Gateway) handleExtractNicks(msg *config.Message) {
 | |
| 	var err error
 | |
| 	br := gw.Bridges[msg.Account]
 | |
| 	for _, outer := range br.GetStringSlice2D("ExtractNicks") {
 | |
| 		search := outer[0]
 | |
| 		replace := outer[1]
 | |
| 		msg.Username, msg.Text, err = extractNick(search, replace, msg.Username, msg.Text)
 | |
| 		if err != nil {
 | |
| 			gw.logger.Errorf("regexp in %s failed: %s", msg.Account, err)
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // extractNick searches for a username (based on "search" a regular expression).
 | |
| // if this matches it extracts a nick (based on "extract" another regular expression) from text
 | |
| // and replaces username with this result.
 | |
| // returns error if the regexp doesn't compile.
 | |
| func extractNick(search, extract, username, text string) (string, string, error) {
 | |
| 	re, err := regexp.Compile(search)
 | |
| 	if err != nil {
 | |
| 		return username, text, err
 | |
| 	}
 | |
| 	if re.MatchString(username) {
 | |
| 		re, err = regexp.Compile(extract)
 | |
| 		if err != nil {
 | |
| 			return username, text, err
 | |
| 		}
 | |
| 		res := re.FindAllStringSubmatch(text, 1)
 | |
| 		// only replace if we have exactly 1 match
 | |
| 		if len(res) > 0 && len(res[0]) == 2 {
 | |
| 			username = res[0][1]
 | |
| 			text = strings.Replace(text, res[0][0], "", 1)
 | |
| 		}
 | |
| 	}
 | |
| 	return username, text, nil
 | |
| }
 | 
