4
0
mirror of https://github.com/cwinfo/matterbridge.git synced 2025-06-27 20:29:24 +00:00

Compare commits

...

9 Commits

Author SHA1 Message Date
Wim
57176dadd4 Support edited messages (telegram). See #141 2017-04-01 18:18:38 +02:00
Wim
dd449a8705 Remove debug info (irc) 2017-04-01 18:10:11 +02:00
Wim
587ad9f41d Remove space after nick (mattermost). Closes #142 2017-04-01 17:44:17 +02:00
Wim
a16ad8bf3b Reuse connection when using same bridge with another gateway. See #87 2017-04-01 17:24:19 +02:00
Wim
1e0490bd36 Merge branch 'channelinfo' 2017-03-28 23:58:22 +02:00
Wim
8afc641f0c Bump version 2017-03-28 23:58:07 +02:00
Wim
2e4d58cb92 Refactor 2017-03-28 23:56:58 +02:00
Wim
02d7e2db65 Release v0.10.3 2017-03-27 20:40:57 +02:00
Wim
f935c573e9 Allow bot tokens for now without warning (slack). Closes #140 2017-03-27 20:15:05 +02:00
12 changed files with 200 additions and 178 deletions

View File

@ -42,7 +42,7 @@ Accounts to one of the supported bridges
# Installing # Installing
## Binaries ## Binaries
Binaries can be found [here] (https://github.com/42wim/matterbridge/releases/) Binaries can be found [here] (https://github.com/42wim/matterbridge/releases/)
* Latest release [v0.10.2](https://github.com/42wim/matterbridge/releases/latest) * Latest release [v0.10.3](https://github.com/42wim/matterbridge/releases/latest)
## Building ## Building
Go 1.6+ is required. Make sure you have [Go](https://golang.org/doc/install) properly installed, including setting up your [GOPATH] (https://golang.org/doc/code.html#GOPATH) Go 1.6+ is required. Make sure you have [Go](https://golang.org/doc/install) properly installed, including setting up your [GOPATH] (https://golang.org/doc/code.html#GOPATH)

View File

@ -30,20 +30,20 @@ type Bridge struct {
Name string Name string
Account string Account string
Protocol string Protocol string
ChannelsIn map[string]config.ChannelOptions Channels map[string]config.ChannelInfo
ChannelsOut map[string]config.ChannelOptions Joined map[string]bool
} }
func New(cfg *config.Config, bridge *config.Bridge, c chan config.Message) *Bridge { func New(cfg *config.Config, bridge *config.Bridge, c chan config.Message) *Bridge {
b := new(Bridge) b := new(Bridge)
b.ChannelsIn = make(map[string]config.ChannelOptions) b.Channels = make(map[string]config.ChannelInfo)
b.ChannelsOut = make(map[string]config.ChannelOptions)
accInfo := strings.Split(bridge.Account, ".") accInfo := strings.Split(bridge.Account, ".")
protocol := accInfo[0] protocol := accInfo[0]
name := accInfo[1] name := accInfo[1]
b.Name = name b.Name = name
b.Protocol = protocol b.Protocol = protocol
b.Account = bridge.Account b.Account = bridge.Account
b.Joined = make(map[string]bool)
// override config from environment // override config from environment
config.OverrideCfgFromEnv(cfg, protocol, name) config.OverrideCfgFromEnv(cfg, protocol, name)
@ -83,33 +83,28 @@ func New(cfg *config.Config, bridge *config.Bridge, c chan config.Message) *Brid
} }
func (b *Bridge) JoinChannels() error { func (b *Bridge) JoinChannels() error {
exists := make(map[string]bool) err := b.joinChannels(b.Channels, b.Joined)
err := b.joinChannels(b.ChannelsIn, exists)
if err != nil {
return err
}
err = b.joinChannels(b.ChannelsOut, exists)
if err != nil { if err != nil {
return err return err
} }
return nil return nil
} }
func (b *Bridge) joinChannels(cMap map[string]config.ChannelOptions, exists map[string]bool) error { func (b *Bridge) joinChannels(channels map[string]config.ChannelInfo, exists map[string]bool) error {
mychannel := "" mychannel := ""
for channel, info := range cMap { for ID, channel := range channels {
if !exists[channel] { if !exists[ID] {
mychannel = channel mychannel = channel.Name
log.Infof("%s: joining %s", b.Account, channel) log.Infof("%s: joining %s (%s)", b.Account, channel.Name, ID)
if b.Protocol == "irc" && info.Key != "" { if b.Protocol == "irc" && channel.Options.Key != "" {
log.Debugf("using key %s for channel %s", info.Key, channel) log.Debugf("using key %s for channel %s", channel.Options.Key, channel.Name)
mychannel = mychannel + " " + info.Key mychannel = mychannel + " " + channel.Options.Key
} }
err := b.JoinChannel(mychannel) err := b.JoinChannel(channel.Name)
if err != nil { if err != nil {
return err return err
} }
exists[channel] = true exists[ID] = true
} }
} }
return nil return nil

View File

@ -25,6 +25,16 @@ type Message struct {
Timestamp time.Time Timestamp time.Time
} }
type ChannelInfo struct {
Name string
Account string
Direction string
ID string
GID map[string]bool
SameChannel map[string]bool
Options ChannelOptions
}
type Protocol struct { type Protocol struct {
BindAddress string // mattermost, slack BindAddress string // mattermost, slack
Buffer int // api Buffer int // api
@ -66,6 +76,7 @@ type Bridge struct {
Account string Account string
Channel string Channel string
Options ChannelOptions Options ChannelOptions
SameChannel bool
} }
type Gateway struct { type Gateway struct {

View File

@ -92,7 +92,7 @@ func (b *Birc) Connect() error {
} }
func (b *Birc) Disconnect() error { func (b *Birc) Disconnect() error {
b.i.Disconnect() //b.i.Disconnect()
close(b.Local) close(b.Local)
return nil return nil
} }

View File

@ -96,12 +96,7 @@ func (b *Bmattermost) Send(msg config.Message) error {
channel := msg.Channel channel := msg.Channel
if b.Config.PrefixMessagesWithNick { if b.Config.PrefixMessagesWithNick {
/*if IsMarkup(message) { message = nick + message
message = nick + "\n\n" + message
} else {
*/
message = nick + " " + message
//}
} }
if !b.Config.UseAPI { if !b.Config.UseAPI {
matterMessage := matterhook.OMessage{IconURL: b.Config.IconURL} matterMessage := matterhook.OMessage{IconURL: b.Config.IconURL}

View File

@ -73,6 +73,10 @@ func (b *Bslack) Disconnect() error {
func (b *Bslack) JoinChannel(channel string) error { func (b *Bslack) JoinChannel(channel string) error {
// we can only join channels using the API // we can only join channels using the API
if b.Config.UseAPI { if b.Config.UseAPI {
if strings.HasPrefix(b.Config.Token, "xoxb") {
// TODO check if bot has already joined channel
return nil
}
_, err := b.sc.JoinChannel(channel) _, err := b.sc.JoinChannel(channel)
if err != nil { if err != nil {
return err return err

View File

@ -80,28 +80,30 @@ func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) {
text := "" text := ""
channel := "" channel := ""
for update := range updates { for update := range updates {
var message *tgbotapi.Message
// handle channels // handle channels
if update.ChannelPost != nil { if update.ChannelPost != nil {
if update.ChannelPost.From != nil { message = update.ChannelPost
username = update.ChannelPost.From.FirstName
if username == "" {
username = update.ChannelPost.From.UserName
} }
} if update.EditedChannelPost != nil {
text = update.ChannelPost.Text message = update.EditedChannelPost
channel = strconv.FormatInt(update.ChannelPost.Chat.ID, 10)
} }
// handle groups // handle groups
if update.Message != nil { if update.Message != nil {
if update.Message.From != nil { message = update.Message
username = update.Message.From.FirstName }
if update.EditedMessage != nil {
message = update.EditedMessage
}
if message.From != nil {
username = message.From.FirstName
if username == "" { if username == "" {
username = update.Message.From.UserName username = message.From.UserName
} }
text = message.Text
channel = strconv.FormatInt(message.Chat.ID, 10)
} }
text = update.Message.Text
channel = strconv.FormatInt(update.Message.Chat.ID, 10)
}
if username == "" { if username == "" {
username = "unknown" username = "unknown"
} }

View File

@ -1,3 +1,13 @@
# v0.11.0-dev
## New features
* general: reusing the same account on multiple gateways now also reuses the connection.
This is particuarly useful for irc. See #87
* general: the Name is now REQUIRED and needs to be UNIQUE for each gateway configuration
# v0.10.3
## Bugfix
* slack: Allow bot tokens for now without warning (slack). Closes #140 (fixes user_is_bot message on channel join)
# v0.10.2 # v0.10.2
## New features ## New features
* general: gops agent added. Allows for more debugging. See #134 * general: gops agent added. Allows for more debugging. See #134

View File

@ -5,7 +5,6 @@ import (
"github.com/42wim/matterbridge/bridge" "github.com/42wim/matterbridge/bridge"
"github.com/42wim/matterbridge/bridge/config" "github.com/42wim/matterbridge/bridge/config"
log "github.com/Sirupsen/logrus" log "github.com/Sirupsen/logrus"
"reflect"
"strings" "strings"
"time" "time"
) )
@ -14,21 +13,21 @@ type Gateway struct {
*config.Config *config.Config
MyConfig *config.Gateway MyConfig *config.Gateway
Bridges map[string]*bridge.Bridge Bridges map[string]*bridge.Bridge
ChannelsOut map[string][]string Channels map[string]*config.ChannelInfo
ChannelsIn map[string][]string
ChannelOptions map[string]config.ChannelOptions ChannelOptions map[string]config.ChannelOptions
Names map[string]bool
Name string Name string
Message chan config.Message Message chan config.Message
DestChannelFunc func(msg *config.Message, dest string) []string DestChannelFunc func(msg *config.Message, dest bridge.Bridge) []config.ChannelInfo
} }
func New(cfg *config.Config, gateway *config.Gateway) *Gateway { func New(cfg *config.Config) *Gateway {
gw := &Gateway{} gw := &Gateway{}
gw.Name = gateway.Name
gw.Config = cfg gw.Config = cfg
gw.MyConfig = gateway gw.Channels = make(map[string]*config.ChannelInfo)
gw.Message = make(chan config.Message) gw.Message = make(chan config.Message)
gw.Bridges = make(map[string]*bridge.Bridge) gw.Bridges = make(map[string]*bridge.Bridge)
gw.Names = make(map[string]bool)
gw.DestChannelFunc = gw.getDestChannel gw.DestChannelFunc = gw.getDestChannel
return gw return gw
} }
@ -36,13 +35,17 @@ func New(cfg *config.Config, gateway *config.Gateway) *Gateway {
func (gw *Gateway) AddBridge(cfg *config.Bridge) error { func (gw *Gateway) AddBridge(cfg *config.Bridge) error {
for _, br := range gw.Bridges { for _, br := range gw.Bridges {
if br.Account == cfg.Account { if br.Account == cfg.Account {
gw.mapChannelsToBridge(br)
err := br.JoinChannels()
if err != nil {
return fmt.Errorf("Bridge %s failed to join channel: %v", br.Account, err)
}
return nil return nil
} }
} }
log.Infof("Starting bridge: %s ", cfg.Account) log.Infof("Starting bridge: %s ", cfg.Account)
br := bridge.New(gw.Config, cfg, gw.Message) br := bridge.New(gw.Config, cfg, gw.Message)
gw.mapChannelsToBridge(br, gw.ChannelsOut) gw.mapChannelsToBridge(br)
gw.mapChannelsToBridge(br, gw.ChannelsIn)
gw.Bridges[cfg.Account] = br gw.Bridges[cfg.Account] = br
err := br.Connect() err := br.Connect()
if err != nil { if err != nil {
@ -55,17 +58,17 @@ func (gw *Gateway) AddBridge(cfg *config.Bridge) error {
return nil return nil
} }
func (gw *Gateway) mapChannelsToBridge(br *bridge.Bridge, cMap map[string][]string) { func (gw *Gateway) AddConfig(cfg *config.Gateway) error {
for _, channel := range cMap[br.Account] { if gw.Names[cfg.Name] {
if _, ok := gw.ChannelOptions[br.Account+channel]; ok { return fmt.Errorf("Gateway with name %s already exists", cfg.Name)
br.ChannelsOut[channel] = gw.ChannelOptions[br.Account+channel]
} else {
br.ChannelsOut[channel] = config.ChannelOptions{}
} }
if cfg.Name == "" {
return fmt.Errorf("%s", "Gateway without name found")
} }
} log.Infof("Starting gateway: %s", cfg.Name)
gw.Names[cfg.Name] = true
func (gw *Gateway) Start() error { gw.Name = cfg.Name
gw.MyConfig = cfg
gw.mapChannels() gw.mapChannels()
for _, br := range append(gw.MyConfig.In, append(gw.MyConfig.InOut, gw.MyConfig.Out...)...) { for _, br := range append(gw.MyConfig.In, append(gw.MyConfig.InOut, gw.MyConfig.Out...)...) {
err := gw.AddBridge(&br) err := gw.AddBridge(&br)
@ -73,6 +76,18 @@ func (gw *Gateway) Start() error {
return err return err
} }
} }
return nil
}
func (gw *Gateway) mapChannelsToBridge(br *bridge.Bridge) {
for ID, channel := range gw.Channels {
if br.Account == channel.Account {
br.Channels[ID] = *channel
}
}
}
func (gw *Gateway) Start() error {
go gw.handleReceive() go gw.handleReceive()
return nil return nil
} }
@ -109,65 +124,68 @@ RECONNECT:
time.Sleep(time.Second * 60) time.Sleep(time.Second * 60)
goto RECONNECT goto RECONNECT
} }
br.Joined = make(map[string]bool)
br.JoinChannels() br.JoinChannels()
} }
func (gw *Gateway) mapChannels() error { func (gw *Gateway) mapChannels() error {
options := make(map[string]config.ChannelOptions) for _, br := range append(gw.MyConfig.Out, gw.MyConfig.InOut...) {
m := make(map[string][]string) ID := br.Channel + br.Account
for _, br := range gw.MyConfig.Out { _, ok := gw.Channels[ID]
m[br.Account] = append(m[br.Account], br.Channel) if !ok {
options[br.Account+br.Channel] = br.Options channel := &config.ChannelInfo{Name: br.Channel, Direction: "out", ID: ID, Options: br.Options, Account: br.Account,
GID: make(map[string]bool), SameChannel: make(map[string]bool)}
channel.GID[gw.Name] = true
channel.SameChannel[gw.Name] = br.SameChannel
gw.Channels[channel.ID] = channel
} }
gw.ChannelsOut = m gw.Channels[ID].GID[gw.Name] = true
m = nil gw.Channels[ID].SameChannel[gw.Name] = br.SameChannel
m = make(map[string][]string)
for _, br := range gw.MyConfig.In {
m[br.Account] = append(m[br.Account], br.Channel)
options[br.Account+br.Channel] = br.Options
} }
gw.ChannelsIn = m
for _, br := range gw.MyConfig.InOut { for _, br := range append(gw.MyConfig.In, gw.MyConfig.InOut...) {
gw.ChannelsIn[br.Account] = append(gw.ChannelsIn[br.Account], br.Channel) ID := br.Channel + br.Account
gw.ChannelsOut[br.Account] = append(gw.ChannelsOut[br.Account], br.Channel) _, ok := gw.Channels[ID]
options[br.Account+br.Channel] = br.Options if !ok {
channel := &config.ChannelInfo{Name: br.Channel, Direction: "in", ID: ID, Options: br.Options, Account: br.Account,
GID: make(map[string]bool), SameChannel: make(map[string]bool)}
channel.GID[gw.Name] = true
channel.SameChannel[gw.Name] = br.SameChannel
gw.Channels[channel.ID] = channel
}
gw.Channels[ID].GID[gw.Name] = true
gw.Channels[ID].SameChannel[gw.Name] = br.SameChannel
} }
gw.ChannelOptions = options
return nil return nil
} }
func (gw *Gateway) getDestChannel(msg *config.Message, dest string) []string { func (gw *Gateway) getDestChannel(msg *config.Message, dest bridge.Bridge) []config.ChannelInfo {
channels := gw.ChannelsIn[msg.Account] var channels []config.ChannelInfo
// broadcast to every out channel (irc QUIT) for _, channel := range gw.Channels {
if msg.Event == config.EVENT_JOIN_LEAVE && msg.Channel == "" { if _, ok := gw.Channels[getChannelID(*msg)]; !ok {
return gw.ChannelsOut[dest] continue
} }
for _, channel := range channels { if channel.Direction == "out" && channel.Account == dest.Account && gw.validGatewayDest(*msg, channel) {
if channel == msg.Channel { channels = append(channels, *channel)
return gw.ChannelsOut[dest]
} }
} }
return []string{} return channels
} }
func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) { func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) {
// only relay join/part when configged // broadcast to every out channel (irc QUIT)
if msg.Event == config.EVENT_JOIN_LEAVE && !gw.Bridges[dest.Account].Config.ShowJoinPart { if msg.Channel == "" && msg.Event != config.EVENT_JOIN_LEAVE {
return
}
originchannel := msg.Channel
channels := gw.DestChannelFunc(&msg, dest.Account)
for _, channel := range channels {
// do not send the message to the bridge we come from if also the channel is the same
if msg.Account == dest.Account && channel == originchannel {
continue
}
msg.Channel = channel
if msg.Channel == "" {
log.Debug("empty channel") log.Debug("empty channel")
return return
} }
log.Debugf("Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, originchannel, dest.Account, channel) originchannel := msg.Channel
for _, channel := range gw.DestChannelFunc(&msg, *dest) {
// do not send to ourself
if channel.ID == getChannelID(msg) {
continue
}
log.Debugf("Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, originchannel, dest.Account, channel.Name)
msg.Channel = channel.Name
gw.modifyUsername(&msg, dest) gw.modifyUsername(&msg, dest)
// for api we need originchannel as channel // for api we need originchannel as channel
if dest.Protocol == "api" { if dest.Protocol == "api" {
@ -194,21 +212,6 @@ func (gw *Gateway) ignoreMessage(msg *config.Message) bool {
return false return false
} }
func (gw *Gateway) modifyMessage(msg *config.Message, dest *bridge.Bridge) {
val := reflect.ValueOf(gw.Config).Elem()
for i := 0; i < val.NumField(); i++ {
typeField := val.Type().Field(i)
// look for the protocol map (both lowercase)
if strings.ToLower(typeField.Name) == dest.Protocol {
// get the Protocol struct from the map
protoCfg := val.Field(i).MapIndex(reflect.ValueOf(dest.Name))
//config.SetNickFormat(msg, protoCfg.Interface().(config.Protocol))
val.Field(i).SetMapIndex(reflect.ValueOf(dest.Name), protoCfg)
break
}
}
}
func (gw *Gateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) { func (gw *Gateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) {
br := gw.Bridges[msg.Account] br := gw.Bridges[msg.Account]
msg.Protocol = br.Protocol msg.Protocol = br.Protocol
@ -221,3 +224,29 @@ func (gw *Gateway) modifyUsername(msg *config.Message, dest *bridge.Bridge) {
nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1) nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1)
msg.Username = nick msg.Username = nick
} }
func getChannelID(msg config.Message) string {
return msg.Channel + msg.Account
}
func (gw *Gateway) validGatewayDest(msg config.Message, channel *config.ChannelInfo) bool {
GIDmap := gw.Channels[getChannelID(msg)].GID
// check if we are running a samechannelgateway.
// if it is and the channel name matches it's ok, otherwise we shouldn't use this channel.
for k, _ := range GIDmap {
if channel.SameChannel[k] == true {
if msg.Channel == channel.Name {
return true
} else {
return false
}
}
}
// check if we are in the correct gateway
for k, _ := range GIDmap {
if channel.GID[k] == true {
return true
}
}
return false
}

View File

@ -2,48 +2,27 @@ package samechannelgateway
import ( import (
"github.com/42wim/matterbridge/bridge/config" "github.com/42wim/matterbridge/bridge/config"
"github.com/42wim/matterbridge/gateway"
) )
type SameChannelGateway struct { type SameChannelGateway struct {
*config.Config *config.Config
MyConfig *config.SameChannelGateway
Channels []string
Name string
} }
func New(cfg *config.Config, gatewayCfg *config.SameChannelGateway) *SameChannelGateway { func New(cfg *config.Config) *SameChannelGateway {
return &SameChannelGateway{ return &SameChannelGateway{Config: cfg}
MyConfig: gatewayCfg,
Channels: gatewayCfg.Channels,
Name: gatewayCfg.Name,
Config: cfg}
} }
func (sgw *SameChannelGateway) Start() error { func (sgw *SameChannelGateway) GetConfig() []config.Gateway {
gw := gateway.New(sgw.Config, &config.Gateway{Name: sgw.Name}) var gwconfigs []config.Gateway
gw.DestChannelFunc = sgw.getDestChannel cfg := sgw.Config
for _, account := range sgw.MyConfig.Accounts { for _, gw := range cfg.SameChannelGateway {
for _, channel := range sgw.Channels { gwconfig := config.Gateway{Name: gw.Name, Enable: gw.Enable}
br := config.Bridge{Account: account, Channel: channel} for _, account := range gw.Accounts {
gw.MyConfig.InOut = append(gw.MyConfig.InOut, br) for _, channel := range gw.Channels {
gwconfig.InOut = append(gwconfig.InOut, config.Bridge{Account: account, Channel: channel, SameChannel: true})
} }
} }
return gw.Start() gwconfigs = append(gwconfigs, gwconfig)
} }
return gwconfigs
func (sgw *SameChannelGateway) validChannel(channel string) bool {
for _, c := range sgw.Channels {
if c == channel {
return true
}
}
return false
}
func (sgw *SameChannelGateway) getDestChannel(msg *config.Message, dest string) []string {
if sgw.validChannel(msg.Channel) {
return []string{msg.Channel}
}
return []string{}
} }

View File

@ -8,10 +8,11 @@ import (
"github.com/42wim/matterbridge/gateway/samechannel" "github.com/42wim/matterbridge/gateway/samechannel"
log "github.com/Sirupsen/logrus" log "github.com/Sirupsen/logrus"
"github.com/google/gops/agent" "github.com/google/gops/agent"
"strings"
) )
var ( var (
version = "0.10.2" version = "0.11.0-dev"
githash string githash string
) )
@ -39,31 +40,26 @@ func main() {
log.SetLevel(log.DebugLevel) log.SetLevel(log.DebugLevel)
} }
log.Printf("Running version %s %s", version, githash) log.Printf("Running version %s %s", version, githash)
if strings.Contains(version, "-dev") {
log.Println("WARNING: THIS IS A DEVELOPMENT VERSION. Things may break.")
}
cfg := config.NewConfig(*flagConfig) cfg := config.NewConfig(*flagConfig)
for _, gw := range cfg.SameChannelGateway {
if !gw.Enable {
continue
}
log.Printf("Starting samechannel gateway %#v", gw.Name)
g := samechannelgateway.New(cfg, &gw)
err := g.Start()
if err != nil {
log.Fatalf("Starting gateway failed %#v", err)
}
log.Printf("Started samechannel gateway %#v", gw.Name)
}
for _, gw := range cfg.Gateway { g := gateway.New(cfg)
sgw := samechannelgateway.New(cfg)
gwconfigs := sgw.GetConfig()
for _, gw := range append(gwconfigs, cfg.Gateway...) {
if !gw.Enable { if !gw.Enable {
continue continue
} }
log.Printf("Starting gateway %#v", gw.Name) err := g.AddConfig(&gw)
g := gateway.New(cfg, &gw) if err != nil {
log.Fatalf("Starting gateway failed: %s", err)
}
}
err := g.Start() err := g.Start()
if err != nil { if err != nil {
log.Fatalf("Starting gateway failed %#v", err) log.Fatalf("Starting gateway failed: %s", err)
}
log.Printf("Started gateway %#v", gw.Name)
} }
log.Printf("Gateway(s) started succesfully. Now relaying messages") log.Printf("Gateway(s) started succesfully. Now relaying messages")
select {} select {}

View File

@ -587,7 +587,7 @@ RemoteNickFormat="[{PROTOCOL}] <{NICK}> "
# #
[[gateway]] [[gateway]]
#OPTIONAL (not used for now) #REQUIRED and UNIQUE
name="gateway1" name="gateway1"
#Enable enables this gateway #Enable enables this gateway
##OPTIONAL (default false) ##OPTIONAL (default false)
@ -659,6 +659,7 @@ enable=true
#channel testing on slack and vice versa. (and for the channel testing2 and testing3) #channel testing on slack and vice versa. (and for the channel testing2 and testing3)
[[samechannelgateway]] [[samechannelgateway]]
name="samechannel1"
enable = false enable = false
accounts = [ "mattermost.work","slack.hobby" ] accounts = [ "mattermost.work","slack.hobby" ]
channels = [ "testing","testing2","testing3"] channels = [ "testing","testing2","testing3"]