Compare commits
	
		
			87 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d2cfd235ef | ||
|   | f8bf7f8d76 | ||
|   | 8432330cb2 | ||
|   | 02577a2b5c | ||
|   | 73501739d5 | ||
|   | ba674af5d4 | ||
|   | 3c85d937c6 | ||
|   | c6fd65d1d7 | ||
|   | 0795906533 | ||
|   | a2b45bc799 | ||
|   | 757657f29c | ||
|   | 219c7659e1 | ||
|   | ae32bae791 | ||
|   | 57eba77561 | ||
|   | d5bc7c4343 | ||
|   | 32f57b7c26 | ||
|   | 692bb8faa7 | ||
|   | 455a0fc239 | ||
|   | b2cbd13251 | ||
|   | ce21ba1545 | ||
|   | c89085bf44 | ||
|   | 4254ed3c63 | ||
|   | 85564a35fd | ||
|   | 09713d40ba | ||
|   | 16d5aeac7c | ||
|   | e19ba5a06a | ||
|   | f7a5077d5d | ||
|   | f8dc24bc09 | ||
|   | e9419f10d3 | ||
|   | cded603c27 | ||
|   | d2ae3ebf9e | ||
|   | 730ccdd456 | ||
|   | 2f042ad915 | ||
|   | ba70691877 | ||
|   | ed11686a99 | ||
|   | 5c50d86908 | ||
|   | fea31753b0 | ||
|   | 0d64cd8bab | ||
|   | 9be0f8f000 | ||
|   | 78401214b0 | ||
|   | b2a07aba3a | ||
|   | 1e0bb3da95 | ||
|   | 59994da176 | ||
|   | 3d281b3316 | ||
|   | ea86849a58 | ||
|   | 399789811e | ||
|   | 8d117cb0a4 | ||
|   | 588b8e0303 | ||
|   | 1794922263 | ||
|   | 0ededb8863 | ||
|   | aa59bb1a41 | ||
|   | f2703979a4 | ||
|   | d2a1dc792f | ||
|   | 06d66a0b2b | ||
|   | 0e2522279e | ||
|   | 141a42a75b | ||
|   | a1bf37e457 | ||
|   | a20b7895a9 | ||
|   | 5666821e7b | ||
|   | 5132d8f097 | ||
|   | b81ff9c008 | ||
|   | 7e62bc4819 | ||
|   | d058be25ad | ||
|   | 1269be1d04 | ||
|   | 3b8837a16b | ||
|   | 32f478e4a0 | ||
|   | e2b50d6194 | ||
|   | 74e33b0a51 | ||
|   | 107969c09a | ||
|   | d379118772 | ||
|   | 291594b99c | ||
|   | f2cdda7278 | ||
|   | 6911458d15 | ||
|   | 6238effdc2 | ||
|   | 498377a230 | ||
|   | 3dd4ec57ff | ||
|   | e15b0e04b8 | ||
|   | 97b1fc813b | ||
|   | 917040b044 | ||
|   | 69646a160d | ||
|   | 54adb0509e | ||
|   | bd3a3b6eaf | ||
|   | 296428d53e | ||
|   | e0ca876de2 | ||
|   | a431a4fa04 | ||
|   | cc2bd03ec9 | ||
|   | 1fe81b7d1e | 
							
								
								
									
										20
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						| @@ -1,12 +1,13 @@ | |||||||
| language: go | language: go | ||||||
| go: | go: | ||||||
|     #- 1.7.x |     - 1.11.x | ||||||
|     - 1.10.x |  | ||||||
|       # - tip |  | ||||||
|  |  | ||||||
| # we have everything vendored | # we have everything vendored | ||||||
| install: true | install: true | ||||||
|  |  | ||||||
|  | git: | ||||||
|  |   depth: 200 | ||||||
|  |  | ||||||
| env: | env: | ||||||
|     - GOOS=linux GOARCH=amd64 |     - GOOS=linux GOARCH=amd64 | ||||||
|       #    - GOOS=windows GOARCH=amd64 |       #    - GOOS=windows GOARCH=amd64 | ||||||
| @@ -25,19 +26,22 @@ notifications: | |||||||
|  |  | ||||||
| before_script: | before_script: | ||||||
|   - MY_VERSION=$(git describe --tags) |   - MY_VERSION=$(git describe --tags) | ||||||
|   - GO_FILES=$(find . -iname '*.go' | grep -v /vendor/)  # All the .go files, excluding vendor/ | #  - GO_FILES=$(find . -iname '*.go' | grep -v /vendor/)  # All the .go files, excluding vendor/ | ||||||
|   - PKGS=$(go list ./... | grep -v /vendor/)             # All the import paths, excluding vendor/ |   - PKGS=$(go list ./... | grep -v /vendor/)             # All the import paths, excluding vendor/ | ||||||
| #  - go get github.com/golang/lint/golint                 # Linter | #  - go get github.com/golang/lint/golint                 # Linter | ||||||
|   - go get honnef.co/go/tools/cmd/megacheck              # Badass static analyzer/linter | #- go get honnef.co/go/tools/cmd/megacheck              # Badass static analyzer/linter | ||||||
|  |   - curl -sfL https://install.goreleaser.com/github.com/golangci/golangci-lint.sh | bash -s -- -b $GOPATH/bin v1.12.2 | ||||||
|  |  | ||||||
|  |  | ||||||
| # Anything in before_script: that returns a nonzero exit code will | # Anything in before_script: that returns a nonzero exit code will | ||||||
| # flunk the build and immediately stop. It's sorta like having | # flunk the build and immediately stop. It's sorta like having | ||||||
| # set -e enabled in bash.  | # set -e enabled in bash.  | ||||||
| script: | script: | ||||||
|  #- test -z $(gofmt -s -l $GO_FILES)  # Fail if a .go file hasn't been formatted with gofmt |   #- test -z "$(go fmt ./...)"         # Fail if a .go file hasn't been formatted with gofmt | ||||||
|   - go test -v -race $PKGS            # Run all the tests with the race detector enabled |   - go test -v -race $PKGS            # Run all the tests with the race detector enabled | ||||||
|  #  - go vet $PKGS                      # go vet is the official Go static analyzer |   #- go vet $PKGS                      # go vet is the official Go static analyzer | ||||||
|   - megacheck $PKGS                   # "go vet on steroids" + linter |   - golangci-lint run --enable-all -D lll -D errcheck -D gosec -D maligned -D prealloc -D gocyclo -D gochecknoglobals | ||||||
|  |   #- megacheck $PKGS                   # "go vet on steroids" + linter | ||||||
|   - /bin/bash ci/bintray.sh |   - /bin/bash ci/bintray.sh | ||||||
|   #- golint -set_exit_status $PKGS     # one last linter |   #- golint -set_exit_status $PKGS     # one last linter | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										127
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,21 +1,40 @@ | |||||||
|  | <div align="center"> | ||||||
|  |  | ||||||
| # matterbridge | # matterbridge | ||||||
| Click on one of the badges below to join the chat    |  | ||||||
|  |  | ||||||
| [](https://gitter.im/42wim/matterbridge) [](https://webchat.freenode.net/?channels=matterbridgechat) [](https://discord.gg/AkKPtrQ) [](https://riot.im/app/#/room/#matterbridge:matrix.org) [](https://join.slack.com/matterbridgechat/shared_invite/MjEwODMxNjU1NDMwLTE0OTk2MTU3NTMtMzZkZmRiNDZhOA) [](https://framateam.org/signup_user_complete/?id=tfqm33ggop8x3qgu4boeieta6e) [](https://inverse.chat) [](https://www.twitch.tv/matterbridge) [](https://matterbridge.zulipchat.com/register/) | <br /> | ||||||
|  |    **A simple chat bridge**<br /> | ||||||
|  |    Letting people be where they want to be.<br /> | ||||||
|  |    <sub>Bridges between a growing number of protocols. Click below to demo.</sub> | ||||||
|  |  | ||||||
| [](https://github.com/42wim/matterbridge/releases/latest) [](https://bintray.com/42wim/nightly/Matterbridge/_latestVersion) |    <sup> | ||||||
|  |  | ||||||
|  |    [Gitter][mb-gitter] | | ||||||
|  |    [IRC][mb-irc] | | ||||||
|  |       [Discord][mb-discord] | | ||||||
|  |       [Matrix][mb-matrix] | | ||||||
|  |       [Slack][mb-slack] | | ||||||
|  |       [Mattermost][mb-mattermost] | | ||||||
|  |       [XMPP][mb-xmpp] | | ||||||
|  |       [Twitch][mb-twitch] | | ||||||
|  |       [Zulip][mb-zulip] | | ||||||
|  |       And more... | ||||||
|  |    </sup> | ||||||
|  |  | ||||||
| Simple bridge between IRC, XMPP, Gitter, Mattermost, Slack, Discord, Telegram, Rocket.Chat, Hipchat(via xmpp), Matrix, Steam, ssh-chat and Zulip | ---- | ||||||
| Has a REST API.    | [](https://github.com/42wim/matterbridge/releases/latest) | ||||||
| Minecraft server chat support via [MatterLink](https://github.com/elytra/MatterLink) |    [](https://bintray.com/42wim/nightly/Matterbridge/_latestVersion) | ||||||
|  |    [](https://codeclimate.com/github/42wim/matterbridge/maintainability) | ||||||
|  |    [](https://codeclimate.com/github/42wim/matterbridge/test_coverage)<br /> | ||||||
|  |   <hr /> | ||||||
|  | </div> | ||||||
|  | <div align="right"><sup> | ||||||
|  |  | ||||||
| **Mattermost isn't required to run matterbridge. It bridges between any supported protocol.**    | **Note:** Matter<em>most</em> isn't required to run matter<em>bridge</em>.</sup></div> | ||||||
| (The name matterbridge is a remnant when it was only bridging mattermost) |  | ||||||
|  |  | ||||||
| # Table of Contents | ### Table of Contents | ||||||
|  * [Features](https://github.com/42wim/matterbridge/wiki/Features) |  * [Features](https://github.com/42wim/matterbridge/wiki/Features) | ||||||
|  |    * [API](#API) | ||||||
|  * [Requirements](#requirements) |  * [Requirements](#requirements) | ||||||
|  * [Screenshots](https://github.com/42wim/matterbridge/wiki/) |  * [Screenshots](https://github.com/42wim/matterbridge/wiki/) | ||||||
|  * [Installing](#installing) |  * [Installing](#installing) | ||||||
| @@ -23,31 +42,35 @@ Minecraft server chat support via [MatterLink](https://github.com/elytra/MatterL | |||||||
|    * [Building](#building) |    * [Building](#building) | ||||||
|  * [Configuration](#configuration) |  * [Configuration](#configuration) | ||||||
|    * [Howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) |    * [Howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) | ||||||
|    * [Examples](#examples)  |    * [Examples](#examples) | ||||||
|  * [Running](#running) |  * [Running](#running) | ||||||
|    * [Docker](#docker) |    * [Docker](#docker) | ||||||
|  * [Changelog](#changelog) |  * [Changelog](#changelog) | ||||||
|  * [FAQ](#faq) |  * [FAQ](#faq) | ||||||
|  |  * [Related projects](#related-projects) | ||||||
|  |  * [Articles](#articles) | ||||||
|  * [Thanks](#thanks) |  * [Thanks](#thanks) | ||||||
|  |  | ||||||
| # Features | ## Features | ||||||
| * [Support bridging between any protocols](https://github.com/42wim/matterbridge/wiki/Features#support-bridging-between-any-protocols) | * [Support bridging between any protocols](https://github.com/42wim/matterbridge/wiki/Features#support-bridging-between-any-protocols) | ||||||
| * [Support multiple gateways(bridges) for your protocols](https://github.com/42wim/matterbridge/wiki/Features#support-multiple-gatewaysbridges-for-your-protocols) | * [Support multiple gateways(bridges) for your protocols](https://github.com/42wim/matterbridge/wiki/Features#support-multiple-gatewaysbridges-for-your-protocols) | ||||||
| * [Message edits and deletes](https://github.com/42wim/matterbridge/wiki/Features#message-edits-and-deletes) | * [Message edits and deletes](https://github.com/42wim/matterbridge/wiki/Features#message-edits-and-deletes) | ||||||
|  | * Preserves threading when possible | ||||||
| * [Attachment / files handling](https://github.com/42wim/matterbridge/wiki/Features#attachment--files-handling) | * [Attachment / files handling](https://github.com/42wim/matterbridge/wiki/Features#attachment--files-handling) | ||||||
| * [Username and avatar spoofing](https://github.com/42wim/matterbridge/wiki/Features#username-and-avatar-spoofing) | * [Username and avatar spoofing](https://github.com/42wim/matterbridge/wiki/Features#username-and-avatar-spoofing) | ||||||
| * [Private groups](https://github.com/42wim/matterbridge/wiki/Features#private-groups) | * [Private groups](https://github.com/42wim/matterbridge/wiki/Features#private-groups) | ||||||
| * [API](https://github.com/42wim/matterbridge/wiki/Features#api) | * [API](https://github.com/42wim/matterbridge/wiki/Features#api) | ||||||
|  |  | ||||||
| ## API | ### API | ||||||
| The API is very basic at the moment and rather undocumented. | The API is very basic at the moment and rather undocumented. | ||||||
|  |  | ||||||
| Used by at least 2 projects. Feel free to make a PR to add your project to this list. | Used by at least 3 projects. Feel free to make a PR to add your project to this list. | ||||||
|  |  | ||||||
| * [MatterLink](https://github.com/elytra/MatterLink) (Matterbridge link for Minecraft Server chat) | * [MatterLink](https://github.com/elytra/MatterLink) (Matterbridge link for Minecraft Server chat) | ||||||
| * [pyCord](https://github.com/NikkyAI/pyCord) (crossplatform chatbot) | * [pyCord](https://github.com/NikkyAI/pyCord) (crossplatform chatbot) | ||||||
|  | * [Mattereddit](https://github.com/bonehurtingjuice/mattereddit) (Reddit chat support) | ||||||
|  |  | ||||||
| # Requirements | ## Requirements | ||||||
| Accounts to one of the supported bridges | Accounts to one of the supported bridges | ||||||
| * [Mattermost](https://github.com/mattermost/platform/) 3.8.x - 3.10.x, 4.x, 5.x | * [Mattermost](https://github.com/mattermost/platform/) 3.8.x - 3.10.x, 4.x, 5.x | ||||||
| * [IRC](http://www.mirc.com/servers.html) | * [IRC](http://www.mirc.com/servers.html) | ||||||
| @@ -64,18 +87,18 @@ Accounts to one of the supported bridges | |||||||
| * [Ssh-chat](https://github.com/shazow/ssh-chat) | * [Ssh-chat](https://github.com/shazow/ssh-chat) | ||||||
| * [Zulip](https://zulipchat.com) | * [Zulip](https://zulipchat.com) | ||||||
|  |  | ||||||
| # Screenshots | ## Screenshots | ||||||
| See https://github.com/42wim/matterbridge/wiki | See https://github.com/42wim/matterbridge/wiki | ||||||
|  |  | ||||||
| # Installing | ## Installing | ||||||
| ## Binaries | ### Binaries | ||||||
| * Latest stable release [v1.11.3](https://github.com/42wim/matterbridge/releases/latest) | * Latest stable release [v1.12.0](https://github.com/42wim/matterbridge/releases/latest) | ||||||
| * Development releases (follows master) can be downloaded [here](https://dl.bintray.com/42wim/nightly/)   | * Development releases (follows master) can be downloaded [here](https://dl.bintray.com/42wim/nightly/) | ||||||
|  |  | ||||||
| ## Building | ### Building | ||||||
| Go 1.8+ 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.8+ 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). | ||||||
|  |  | ||||||
| After Go is setup, download matterbridge to your $GOPATH directory.  | After Go is setup, download matterbridge to your $GOPATH directory. | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| cd $GOPATH | cd $GOPATH | ||||||
| @@ -89,16 +112,16 @@ $ ls bin/ | |||||||
| matterbridge | matterbridge | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| # Configuration | ## Configuration | ||||||
| ## Basic configuration | ### Basic configuration | ||||||
| See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration. | See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration. | ||||||
|  |  | ||||||
| ## Advanced configuration | ### Advanced configuration | ||||||
| * [matterbridge.toml.sample](https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for documentation and an example. | * [matterbridge.toml.sample](https://github.com/42wim/matterbridge/blob/master/matterbridge.toml.sample) for documentation and an example. | ||||||
|  |  | ||||||
| ## Examples  | ### Examples | ||||||
| ### Bridge mattermost (off-topic) - irc (#testing) | #### Bridge mattermost (off-topic) - irc (#testing) | ||||||
| ``` | ```toml | ||||||
| [irc] | [irc] | ||||||
|     [irc.freenode] |     [irc.freenode] | ||||||
|     Server="irc.freenode.net:6667" |     Server="irc.freenode.net:6667" | ||||||
| @@ -125,8 +148,8 @@ enable=true | |||||||
|     channel="off-topic" |     channel="off-topic" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Bridge slack (#general) - discord (general) | #### Bridge slack (#general) - discord (general) | ||||||
| ``` | ```toml | ||||||
| [slack] | [slack] | ||||||
| [slack.test] | [slack.test] | ||||||
| Token="yourslacktoken" | Token="yourslacktoken" | ||||||
| @@ -153,7 +176,7 @@ RemoteNickFormat="[{PROTOCOL}/{BRIDGE}] <{NICK}> " | |||||||
|     channel = "general" |     channel = "general" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| # Running | ## Running | ||||||
|  |  | ||||||
| See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration. | See [howto](https://github.com/42wim/matterbridge/wiki/How-to-create-your-config) for a step by step walkthrough for creating your configuration. | ||||||
|  |  | ||||||
| @@ -169,24 +192,42 @@ Usage of ./matterbridge: | |||||||
|         show version |         show version | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Docker | ### Docker | ||||||
| Create your matterbridge.toml file locally eg in ```/tmp/matterbridge.toml``` | Create your matterbridge.toml file locally eg in `/tmp/matterbridge.toml` | ||||||
| ``` | ``` | ||||||
| docker run -ti -v /tmp/matterbridge.toml:/matterbridge.toml 42wim/matterbridge | docker run -ti -v /tmp/matterbridge.toml:/matterbridge.toml 42wim/matterbridge | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| # Changelog | ## Changelog | ||||||
| See [changelog.md](https://github.com/42wim/matterbridge/blob/master/changelog.md) | See [changelog.md](https://github.com/42wim/matterbridge/blob/master/changelog.md) | ||||||
|  |  | ||||||
| # FAQ | ## FAQ | ||||||
|  |  | ||||||
| See [FAQ](https://github.com/42wim/matterbridge/wiki/FAQ) | See [FAQ](https://github.com/42wim/matterbridge/wiki/FAQ) | ||||||
|  |  | ||||||
| Want to tip ?  | Want to tip ? | ||||||
| * eth: 0xb3f9b5387c66ad6be892bcb7bbc67862f3abc16f | * eth: 0xb3f9b5387c66ad6be892bcb7bbc67862f3abc16f | ||||||
| * btc: 1N7cKHj5SfqBHBzDJ6kad4BzeqUBBS2zhs | * btc: 1N7cKHj5SfqBHBzDJ6kad4BzeqUBBS2zhs | ||||||
|  |  | ||||||
| # Thanks | ## Related projects | ||||||
|  | * [matterbridge-heroku](https://github.com/cadecairos/matterbridge-heroku) | ||||||
|  | * [matterbridge config viewer](https://github.com/patcon/matterbridge-heroku-viewer) | ||||||
|  | * [matterbridge autoconfig](https://github.com/patcon/matterbridge-autoconfig) | ||||||
|  | * [matterlink](https://github.com/elytra/MatterLink) | ||||||
|  | * [mattereddit](https://github.com/bonehurtingjuice/mattereddit) | ||||||
|  | * [pyCord](https://github.com/NikkyAI/pyCord) (crossplatform chatbot) | ||||||
|  | * [mattermost-plugin](https://github.com/matterbridge/mattermost-plugin) - Run matterbridge as a plugin in mattermost | ||||||
|  |  | ||||||
|  | ## Articles | ||||||
|  | * https://mattermost.com/blog/connect-irc-to-mattermost/ | ||||||
|  | * https://blog.valvin.fr/2016/09/17/mattermost-et-un-channel-irc-cest-possible/ | ||||||
|  | * https://blog.brightscout.com/top-10-mattermost-integrations/ | ||||||
|  | * http://bencey.co.nz/2018/09/17/bridge/ | ||||||
|  | * https://www.algoo.fr/blog/2018/01/19/recouvrez-votre-liberte-en-quittant-slack-pour-un-mattermost-auto-heberge/ | ||||||
|  | * https://kopano.com/blog/matterbridge-bridging-mattermost-chat/ | ||||||
|  | * https://www.stitcher.com/s/?eid=52382713 | ||||||
|  |  | ||||||
|  | ## Thanks | ||||||
| [](https://www.digitalocean.com/) for sponsoring demo/testing droplets. | [](https://www.digitalocean.com/) for sponsoring demo/testing droplets. | ||||||
|  |  | ||||||
| Matterbridge wouldn't exist without these libraries: | Matterbridge wouldn't exist without these libraries: | ||||||
| @@ -203,3 +244,15 @@ Matterbridge wouldn't exist without these libraries: | |||||||
| * telegram - https://github.com/go-telegram-bot-api/telegram-bot-api | * telegram - https://github.com/go-telegram-bot-api/telegram-bot-api | ||||||
| * xmpp - https://github.com/mattn/go-xmpp | * xmpp - https://github.com/mattn/go-xmpp | ||||||
| * zulip - https://github.com/ifo/gozulipbot | * zulip - https://github.com/ifo/gozulipbot | ||||||
|  |  | ||||||
|  | <!-- Links --> | ||||||
|  |  | ||||||
|  |    [mb-gitter]: https://gitter.im/42wim/matterbridge | ||||||
|  |    [mb-irc]: https://webchat.freenode.net/?channels=matterbridgechat | ||||||
|  |    [mb-discord]: https://discord.gg/AkKPtrQ | ||||||
|  |    [mb-matrix]: https://riot.im/app/#/room/#matterbridge:matrix.org | ||||||
|  |    [mb-slack]: https://join.slack.com/matterbridgechat/shared_invite/MjEwODMxNjU1NDMwLTE0OTk2MTU3NTMtMzZkZmRiNDZhOA | ||||||
|  |    [mb-mattermost]: https://framateam.org/signup_user_complete/?id=tfqm33ggop8x3qgu4boeieta6e | ||||||
|  |    [mb-xmpp]: https://inverse.chat/ | ||||||
|  |    [mb-twitch]: https://www.twitch.tv/matterbridge | ||||||
|  |    [mb-zulip]: https://matterbridge.zulipchat.com/register/ | ||||||
|   | |||||||
| @@ -13,13 +13,13 @@ import ( | |||||||
| 	"github.com/zfjagann/golang-ring" | 	"github.com/zfjagann/golang-ring" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Api struct { | type API struct { | ||||||
| 	Messages ring.Ring | 	Messages ring.Ring | ||||||
| 	sync.RWMutex | 	sync.RWMutex | ||||||
| 	*bridge.Config | 	*bridge.Config | ||||||
| } | } | ||||||
|  |  | ||||||
| type ApiMessage struct { | type Message struct { | ||||||
| 	Text     string `json:"text"` | 	Text     string `json:"text"` | ||||||
| 	Username string `json:"username"` | 	Username string `json:"username"` | ||||||
| 	UserID   string `json:"userid"` | 	UserID   string `json:"userid"` | ||||||
| @@ -28,17 +28,20 @@ type ApiMessage struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| func New(cfg *bridge.Config) bridge.Bridger { | func New(cfg *bridge.Config) bridge.Bridger { | ||||||
| 	b := &Api{Config: cfg} | 	b := &API{Config: cfg} | ||||||
| 	e := echo.New() | 	e := echo.New() | ||||||
| 	e.HideBanner = true | 	e.HideBanner = true | ||||||
| 	e.HidePort = true | 	e.HidePort = true | ||||||
| 	b.Messages = ring.Ring{} | 	b.Messages = ring.Ring{} | ||||||
| 	b.Messages.SetCapacity(b.GetInt("Buffer")) | 	if b.GetInt("Buffer") != 0 { | ||||||
|  | 		b.Messages.SetCapacity(b.GetInt("Buffer")) | ||||||
|  | 	} | ||||||
| 	if b.GetString("Token") != "" { | 	if b.GetString("Token") != "" { | ||||||
| 		e.Use(middleware.KeyAuth(func(key string, c echo.Context) (bool, error) { | 		e.Use(middleware.KeyAuth(func(key string, c echo.Context) (bool, error) { | ||||||
| 			return key == b.GetString("Token"), nil | 			return key == b.GetString("Token"), nil | ||||||
| 		})) | 		})) | ||||||
| 	} | 	} | ||||||
|  | 	e.GET("/api/health", b.handleHealthcheck) | ||||||
| 	e.GET("/api/messages", b.handleMessages) | 	e.GET("/api/messages", b.handleMessages) | ||||||
| 	e.GET("/api/stream", b.handleStream) | 	e.GET("/api/stream", b.handleStream) | ||||||
| 	e.POST("/api/message", b.handlePostMessage) | 	e.POST("/api/message", b.handlePostMessage) | ||||||
| @@ -52,30 +55,34 @@ func New(cfg *bridge.Config) bridge.Bridger { | |||||||
| 	return b | 	return b | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Api) Connect() error { | func (b *API) Connect() error { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| func (b *Api) Disconnect() error { | func (b *API) Disconnect() error { | ||||||
| 	return nil | 	return nil | ||||||
|  |  | ||||||
| } | } | ||||||
| func (b *Api) JoinChannel(channel config.ChannelInfo) error { | func (b *API) JoinChannel(channel config.ChannelInfo) error { | ||||||
| 	return nil | 	return nil | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Api) Send(msg config.Message) (string, error) { | func (b *API) Send(msg config.Message) (string, error) { | ||||||
| 	b.Lock() | 	b.Lock() | ||||||
| 	defer b.Unlock() | 	defer b.Unlock() | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
| 	b.Messages.Enqueue(&msg) | 	b.Messages.Enqueue(&msg) | ||||||
| 	return "", nil | 	return "", nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Api) handlePostMessage(c echo.Context) error { | func (b *API) handleHealthcheck(c echo.Context) error { | ||||||
|  | 	return c.String(http.StatusOK, "OK") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *API) handlePostMessage(c echo.Context) error { | ||||||
| 	message := config.Message{} | 	message := config.Message{} | ||||||
| 	if err := c.Bind(&message); err != nil { | 	if err := c.Bind(&message); err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -91,7 +98,7 @@ func (b *Api) handlePostMessage(c echo.Context) error { | |||||||
| 	return c.JSON(http.StatusOK, message) | 	return c.JSON(http.StatusOK, message) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Api) handleMessages(c echo.Context) error { | func (b *API) handleMessages(c echo.Context) error { | ||||||
| 	b.Lock() | 	b.Lock() | ||||||
| 	defer b.Unlock() | 	defer b.Unlock() | ||||||
| 	c.JSONPretty(http.StatusOK, b.Messages.Values(), " ") | 	c.JSONPretty(http.StatusOK, b.Messages.Values(), " ") | ||||||
| @@ -99,9 +106,17 @@ func (b *Api) handleMessages(c echo.Context) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Api) handleStream(c echo.Context) error { | func (b *API) handleStream(c echo.Context) error { | ||||||
| 	c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSON) | 	c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSON) | ||||||
| 	c.Response().WriteHeader(http.StatusOK) | 	c.Response().WriteHeader(http.StatusOK) | ||||||
|  | 	greet := config.Message{ | ||||||
|  | 		Event:     config.EventAPIConnected, | ||||||
|  | 		Timestamp: time.Now(), | ||||||
|  | 	} | ||||||
|  | 	if err := json.NewEncoder(c.Response()).Encode(greet); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	c.Response().Flush() | ||||||
| 	closeNotifier := c.Response().CloseNotify() | 	closeNotifier := c.Response().CloseNotify() | ||||||
| 	for { | 	for { | ||||||
| 		select { | 		select { | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ type Bridge struct { | |||||||
| 	Channels map[string]config.ChannelInfo | 	Channels map[string]config.ChannelInfo | ||||||
| 	Joined   map[string]bool | 	Joined   map[string]bool | ||||||
| 	Log      *log.Entry | 	Log      *log.Entry | ||||||
| 	Config   *config.Config | 	Config   config.Config | ||||||
| 	General  *config.Protocol | 	General  *config.Protocol | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -69,36 +69,41 @@ func (b *Bridge) joinChannels(channels map[string]config.ChannelInfo, exists map | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bridge) GetBool(key string) bool { | func (b *Bridge) GetBool(key string) bool { | ||||||
| 	if b.Config.GetBool(b.Account + "." + key) { | 	val, ok := b.Config.GetBool(b.Account + "." + key) | ||||||
| 		return b.Config.GetBool(b.Account + "." + key) | 	if !ok { | ||||||
|  | 		val, _ = b.Config.GetBool("general." + key) | ||||||
| 	} | 	} | ||||||
| 	return b.Config.GetBool("general." + key) | 	return val | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bridge) GetInt(key string) int { | func (b *Bridge) GetInt(key string) int { | ||||||
| 	if b.Config.GetInt(b.Account+"."+key) != 0 { | 	val, ok := b.Config.GetInt(b.Account + "." + key) | ||||||
| 		return b.Config.GetInt(b.Account + "." + key) | 	if !ok { | ||||||
|  | 		val, _ = b.Config.GetInt("general." + key) | ||||||
| 	} | 	} | ||||||
| 	return b.Config.GetInt("general." + key) | 	return val | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bridge) GetString(key string) string { | func (b *Bridge) GetString(key string) string { | ||||||
| 	if b.Config.GetString(b.Account+"."+key) != "" { | 	val, ok := b.Config.GetString(b.Account + "." + key) | ||||||
| 		return b.Config.GetString(b.Account + "." + key) | 	if !ok { | ||||||
|  | 		val, _ = b.Config.GetString("general." + key) | ||||||
| 	} | 	} | ||||||
| 	return b.Config.GetString("general." + key) | 	return val | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bridge) GetStringSlice(key string) []string { | func (b *Bridge) GetStringSlice(key string) []string { | ||||||
| 	if len(b.Config.GetStringSlice(b.Account+"."+key)) != 0 { | 	val, ok := b.Config.GetStringSlice(b.Account + "." + key) | ||||||
| 		return b.Config.GetStringSlice(b.Account + "." + key) | 	if !ok { | ||||||
|  | 		val, _ = b.Config.GetStringSlice("general." + key) | ||||||
| 	} | 	} | ||||||
| 	return b.Config.GetStringSlice("general." + key) | 	return val | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bridge) GetStringSlice2D(key string) [][]string { | func (b *Bridge) GetStringSlice2D(key string) [][]string { | ||||||
| 	if len(b.Config.GetStringSlice2D(b.Account+"."+key)) != 0 { | 	val, ok := b.Config.GetStringSlice2D(b.Account + "." + key) | ||||||
| 		return b.Config.GetStringSlice2D(b.Account + "." + key) | 	if !ok { | ||||||
|  | 		val, _ = b.Config.GetStringSlice2D("general." + key) | ||||||
| 	} | 	} | ||||||
| 	return b.Config.GetStringSlice2D("general." + key) | 	return val | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ package config | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"os" | 	"io/ioutil" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -14,14 +14,16 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
| 	EVENT_JOIN_LEAVE        = "join_leave" | 	EventJoinLeave       = "join_leave" | ||||||
| 	EVENT_TOPIC_CHANGE      = "topic_change" | 	EventTopicChange     = "topic_change" | ||||||
| 	EVENT_FAILURE           = "failure" | 	EventFailure         = "failure" | ||||||
| 	EVENT_FILE_FAILURE_SIZE = "file_failure_size" | 	EventFileFailureSize = "file_failure_size" | ||||||
| 	EVENT_AVATAR_DOWNLOAD   = "avatar_download" | 	EventAvatarDownload  = "avatar_download" | ||||||
| 	EVENT_REJOIN_CHANNELS   = "rejoin_channels" | 	EventRejoinChannels  = "rejoin_channels" | ||||||
| 	EVENT_USER_ACTION       = "user_action" | 	EventUserAction      = "user_action" | ||||||
| 	EVENT_MSG_DELETE        = "msg_delete" | 	EventMsgDelete       = "msg_delete" | ||||||
|  | 	EventAPIConnected    = "api_connected" | ||||||
|  | 	EventUserTyping      = "user_typing" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Message struct { | type Message struct { | ||||||
| @@ -34,6 +36,7 @@ type Message struct { | |||||||
| 	Event     string    `json:"event"` | 	Event     string    `json:"event"` | ||||||
| 	Protocol  string    `json:"protocol"` | 	Protocol  string    `json:"protocol"` | ||||||
| 	Gateway   string    `json:"gateway"` | 	Gateway   string    `json:"gateway"` | ||||||
|  | 	ParentID  string    `json:"parent_id"` | ||||||
| 	Timestamp time.Time `json:"timestamp"` | 	Timestamp time.Time `json:"timestamp"` | ||||||
| 	ID        string    `json:"id"` | 	ID        string    `json:"id"` | ||||||
| 	Extra     map[string][]interface{} | 	Extra     map[string][]interface{} | ||||||
| @@ -97,6 +100,7 @@ type Protocol struct { | |||||||
| 	NoTLS                  bool       // mattermost | 	NoTLS                  bool       // mattermost | ||||||
| 	Password               string     // IRC,mattermost,XMPP,matrix | 	Password               string     // IRC,mattermost,XMPP,matrix | ||||||
| 	PrefixMessagesWithNick bool       // mattemost, slack | 	PrefixMessagesWithNick bool       // mattemost, slack | ||||||
|  | 	PreserveThreading      bool       // slack | ||||||
| 	Protocol               string     // all protocols | 	Protocol               string     // all protocols | ||||||
| 	QuoteDisable           bool       // telegram | 	QuoteDisable           bool       // telegram | ||||||
| 	QuoteFormat            string     // telegram | 	QuoteFormat            string     // telegram | ||||||
| @@ -107,6 +111,7 @@ type Protocol struct { | |||||||
| 	Server                 string     // IRC,mattermost,XMPP,discord | 	Server                 string     // IRC,mattermost,XMPP,discord | ||||||
| 	ShowJoinPart           bool       // all protocols | 	ShowJoinPart           bool       // all protocols | ||||||
| 	ShowTopicChange        bool       // slack | 	ShowTopicChange        bool       // slack | ||||||
|  | 	ShowUserTyping         bool       // slack | ||||||
| 	ShowEmbeds             bool       // discord | 	ShowEmbeds             bool       // discord | ||||||
| 	SkipTLSVerify          bool       // IRC, mattermost | 	SkipTLSVerify          bool       // IRC, mattermost | ||||||
| 	StripNick              bool       // all protocols | 	StripNick              bool       // all protocols | ||||||
| @@ -152,113 +157,129 @@ type SameChannelGateway struct { | |||||||
| 	Accounts []string | 	Accounts []string | ||||||
| } | } | ||||||
|  |  | ||||||
| type ConfigValues struct { | type BridgeValues struct { | ||||||
| 	Api                map[string]Protocol | 	API                map[string]Protocol | ||||||
| 	Irc                map[string]Protocol | 	IRC                map[string]Protocol | ||||||
| 	Mattermost         map[string]Protocol | 	Mattermost         map[string]Protocol | ||||||
| 	Matrix             map[string]Protocol | 	Matrix             map[string]Protocol | ||||||
| 	Slack              map[string]Protocol | 	Slack              map[string]Protocol | ||||||
|  | 	SlackLegacy        map[string]Protocol | ||||||
| 	Steam              map[string]Protocol | 	Steam              map[string]Protocol | ||||||
| 	Gitter             map[string]Protocol | 	Gitter             map[string]Protocol | ||||||
| 	Xmpp               map[string]Protocol | 	XMPP               map[string]Protocol | ||||||
| 	Discord            map[string]Protocol | 	Discord            map[string]Protocol | ||||||
| 	Telegram           map[string]Protocol | 	Telegram           map[string]Protocol | ||||||
| 	Rocketchat         map[string]Protocol | 	Rocketchat         map[string]Protocol | ||||||
| 	Sshchat            map[string]Protocol | 	SSHChat            map[string]Protocol | ||||||
| 	Zulip              map[string]Protocol | 	Zulip              map[string]Protocol | ||||||
| 	General            Protocol | 	General            Protocol | ||||||
| 	Gateway            []Gateway | 	Gateway            []Gateway | ||||||
| 	SameChannelGateway []SameChannelGateway | 	SameChannelGateway []SameChannelGateway | ||||||
| } | } | ||||||
|  |  | ||||||
| type Config struct { | type Config interface { | ||||||
| 	v *viper.Viper | 	BridgeValues() *BridgeValues | ||||||
| 	*ConfigValues | 	GetBool(key string) (bool, bool) | ||||||
| 	sync.RWMutex | 	GetInt(key string) (int, bool) | ||||||
|  | 	GetString(key string) (string, bool) | ||||||
|  | 	GetStringSlice(key string) ([]string, bool) | ||||||
|  | 	GetStringSlice2D(key string) ([][]string, bool) | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewConfig(cfgfile string) *Config { | type config struct { | ||||||
|  | 	v *viper.Viper | ||||||
|  | 	sync.RWMutex | ||||||
|  |  | ||||||
|  | 	cv *BridgeValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewConfig(cfgfile string) Config { | ||||||
| 	log.SetFormatter(&prefixed.TextFormatter{PrefixPadding: 13, DisableColors: true, FullTimestamp: false}) | 	log.SetFormatter(&prefixed.TextFormatter{PrefixPadding: 13, DisableColors: true, FullTimestamp: false}) | ||||||
| 	flog := log.WithFields(log.Fields{"prefix": "config"}) | 	flog := log.WithFields(log.Fields{"prefix": "config"}) | ||||||
| 	var cfg ConfigValues |  | ||||||
| 	viper.SetConfigType("toml") |  | ||||||
| 	viper.SetConfigFile(cfgfile) | 	viper.SetConfigFile(cfgfile) | ||||||
| 	viper.SetEnvPrefix("matterbridge") | 	input, err := getFileContents(cfgfile) | ||||||
| 	viper.AddConfigPath(".") |  | ||||||
| 	viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_")) |  | ||||||
| 	viper.AutomaticEnv() |  | ||||||
| 	f, err := os.Open(cfgfile) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Fatal(err) | 		log.Fatal(err) | ||||||
| 	} | 	} | ||||||
| 	err = viper.ReadConfig(f) | 	mycfg := newConfigFromString(input) | ||||||
| 	if err != nil { | 	if mycfg.cv.General.MediaDownloadSize == 0 { | ||||||
| 		log.Fatal(err) | 		mycfg.cv.General.MediaDownloadSize = 1000000 | ||||||
| 	} |  | ||||||
| 	err = viper.Unmarshal(&cfg) |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Fatal("blah", err) |  | ||||||
| 	} |  | ||||||
| 	mycfg := new(Config) |  | ||||||
| 	mycfg.v = viper.GetViper() |  | ||||||
| 	if cfg.General.MediaDownloadSize == 0 { |  | ||||||
| 		cfg.General.MediaDownloadSize = 1000000 |  | ||||||
| 	} | 	} | ||||||
| 	viper.WatchConfig() | 	viper.WatchConfig() | ||||||
| 	viper.OnConfigChange(func(e fsnotify.Event) { | 	viper.OnConfigChange(func(e fsnotify.Event) { | ||||||
| 		flog.Println("Config file changed:", e.Name) | 		flog.Println("Config file changed:", e.Name) | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	mycfg.ConfigValues = &cfg |  | ||||||
| 	return mycfg | 	return mycfg | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewConfigFromString(input []byte) *Config { | func getFileContents(filename string) ([]byte, error) { | ||||||
| 	var cfg ConfigValues | 	input, err := ioutil.ReadFile(filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal(err) | ||||||
|  | 		return []byte(nil), err | ||||||
|  | 	} | ||||||
|  | 	return input, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewConfigFromString(input []byte) Config { | ||||||
|  | 	return newConfigFromString(input) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func newConfigFromString(input []byte) *config { | ||||||
| 	viper.SetConfigType("toml") | 	viper.SetConfigType("toml") | ||||||
|  | 	viper.SetEnvPrefix("matterbridge") | ||||||
|  | 	viper.AddConfigPath(".") | ||||||
|  | 	viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_", "-", "_")) | ||||||
|  | 	viper.AutomaticEnv() | ||||||
| 	err := viper.ReadConfig(bytes.NewBuffer(input)) | 	err := viper.ReadConfig(bytes.NewBuffer(input)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Fatal(err) | 		log.Fatal(err) | ||||||
| 	} | 	} | ||||||
| 	err = viper.Unmarshal(&cfg) |  | ||||||
|  | 	cfg := &BridgeValues{} | ||||||
|  | 	err = viper.Unmarshal(cfg) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Fatal(err) | 		log.Fatal(err) | ||||||
| 	} | 	} | ||||||
| 	mycfg := new(Config) | 	return &config{ | ||||||
| 	mycfg.v = viper.GetViper() | 		v:  viper.GetViper(), | ||||||
| 	mycfg.ConfigValues = &cfg | 		cv: cfg, | ||||||
| 	return mycfg | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Config) GetBool(key string) bool { | func (c *config) BridgeValues() *BridgeValues { | ||||||
|  | 	return c.cv | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *config) GetBool(key string) (bool, bool) { | ||||||
| 	c.RLock() | 	c.RLock() | ||||||
| 	defer c.RUnlock() | 	defer c.RUnlock() | ||||||
| 	//	log.Debugf("getting bool %s = %#v", key, c.v.GetBool(key)) | 	//	log.Debugf("getting bool %s = %#v", key, c.v.GetBool(key)) | ||||||
| 	return c.v.GetBool(key) | 	return c.v.GetBool(key), c.v.IsSet(key) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Config) GetInt(key string) int { | func (c *config) GetInt(key string) (int, bool) { | ||||||
| 	c.RLock() | 	c.RLock() | ||||||
| 	defer c.RUnlock() | 	defer c.RUnlock() | ||||||
| 	//	log.Debugf("getting int %s = %d", key, c.v.GetInt(key)) | 	//	log.Debugf("getting int %s = %d", key, c.v.GetInt(key)) | ||||||
| 	return c.v.GetInt(key) | 	return c.v.GetInt(key), c.v.IsSet(key) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Config) GetString(key string) string { | func (c *config) GetString(key string) (string, bool) { | ||||||
| 	c.RLock() | 	c.RLock() | ||||||
| 	defer c.RUnlock() | 	defer c.RUnlock() | ||||||
| 	//	log.Debugf("getting String %s = %s", key, c.v.GetString(key)) | 	//	log.Debugf("getting String %s = %s", key, c.v.GetString(key)) | ||||||
| 	return c.v.GetString(key) | 	return c.v.GetString(key), c.v.IsSet(key) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Config) GetStringSlice(key string) []string { | func (c *config) GetStringSlice(key string) ([]string, bool) { | ||||||
| 	c.RLock() | 	c.RLock() | ||||||
| 	defer c.RUnlock() | 	defer c.RUnlock() | ||||||
| 	// log.Debugf("getting StringSlice %s = %#v", key, c.v.GetStringSlice(key)) | 	// log.Debugf("getting StringSlice %s = %#v", key, c.v.GetStringSlice(key)) | ||||||
| 	return c.v.GetStringSlice(key) | 	return c.v.GetStringSlice(key), c.v.IsSet(key) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Config) GetStringSlice2D(key string) [][]string { | func (c *config) GetStringSlice2D(key string) ([][]string, bool) { | ||||||
| 	c.RLock() | 	c.RLock() | ||||||
| 	defer c.RUnlock() | 	defer c.RUnlock() | ||||||
| 	result := [][]string{} | 	result := [][]string{} | ||||||
| @@ -270,9 +291,9 @@ func (c *Config) GetStringSlice2D(key string) [][]string { | |||||||
| 			} | 			} | ||||||
| 			result = append(result, result2) | 			result = append(result, result2) | ||||||
| 		} | 		} | ||||||
| 		return result | 		return result, true | ||||||
| 	} | 	} | ||||||
| 	return result | 	return result, false | ||||||
| } | } | ||||||
|  |  | ||||||
| func GetIconURL(msg *Message, iconURL string) string { | func GetIconURL(msg *Message, iconURL string) string { | ||||||
| @@ -284,3 +305,45 @@ func GetIconURL(msg *Message, iconURL string) string { | |||||||
| 	iconURL = strings.Replace(iconURL, "{PROTOCOL}", protocol, -1) | 	iconURL = strings.Replace(iconURL, "{PROTOCOL}", protocol, -1) | ||||||
| 	return iconURL | 	return iconURL | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type TestConfig struct { | ||||||
|  | 	Config | ||||||
|  |  | ||||||
|  | 	Overrides map[string]interface{} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetBool(key string) (bool, bool) { | ||||||
|  | 	val, ok := c.Overrides[key] | ||||||
|  | 	if ok { | ||||||
|  | 		return val.(bool), true | ||||||
|  | 	} | ||||||
|  | 	return c.Config.GetBool(key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetInt(key string) (int, bool) { | ||||||
|  | 	if val, ok := c.Overrides[key]; ok { | ||||||
|  | 		return val.(int), true | ||||||
|  | 	} | ||||||
|  | 	return c.Config.GetInt(key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetString(key string) (string, bool) { | ||||||
|  | 	if val, ok := c.Overrides[key]; ok { | ||||||
|  | 		return val.(string), true | ||||||
|  | 	} | ||||||
|  | 	return c.Config.GetString(key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetStringSlice(key string) ([]string, bool) { | ||||||
|  | 	if val, ok := c.Overrides[key]; ok { | ||||||
|  | 		return val.([]string), true | ||||||
|  | 	} | ||||||
|  | 	return c.Config.GetStringSlice(key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetStringSlice2D(key string) ([][]string, bool) { | ||||||
|  | 	if val, ok := c.Overrides[key]; ok { | ||||||
|  | 		return val.([][]string), true | ||||||
|  | 	} | ||||||
|  | 	return c.Config.GetStringSlice2D(key) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package bdiscord | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"regexp" | 	"regexp" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -10,7 +11,7 @@ import ( | |||||||
| 	"github.com/42wim/matterbridge/bridge" | 	"github.com/42wim/matterbridge/bridge" | ||||||
| 	"github.com/42wim/matterbridge/bridge/config" | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
| 	"github.com/42wim/matterbridge/bridge/helper" | 	"github.com/42wim/matterbridge/bridge/helper" | ||||||
| 	"github.com/matterbridge/discordgo" | 	"github.com/bwmarrin/discordgo" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const MessageLength = 1950 | const MessageLength = 1950 | ||||||
| @@ -21,6 +22,7 @@ type Bdiscord struct { | |||||||
| 	Nick           string | 	Nick           string | ||||||
| 	UseChannelID   bool | 	UseChannelID   bool | ||||||
| 	userMemberMap  map[string]*discordgo.Member | 	userMemberMap  map[string]*discordgo.Member | ||||||
|  | 	nickMemberMap  map[string]*discordgo.Member | ||||||
| 	guildID        string | 	guildID        string | ||||||
| 	webhookID      string | 	webhookID      string | ||||||
| 	webhookToken   string | 	webhookToken   string | ||||||
| @@ -32,6 +34,7 @@ type Bdiscord struct { | |||||||
| func New(cfg *bridge.Config) bridge.Bridger { | func New(cfg *bridge.Config) bridge.Bridger { | ||||||
| 	b := &Bdiscord{Config: cfg} | 	b := &Bdiscord{Config: cfg} | ||||||
| 	b.userMemberMap = make(map[string]*discordgo.Member) | 	b.userMemberMap = make(map[string]*discordgo.Member) | ||||||
|  | 	b.nickMemberMap = make(map[string]*discordgo.Member) | ||||||
| 	b.channelInfoMap = make(map[string]*config.ChannelInfo) | 	b.channelInfoMap = make(map[string]*config.ChannelInfo) | ||||||
| 	if b.GetString("WebhookURL") != "" { | 	if b.GetString("WebhookURL") != "" { | ||||||
| 		b.Log.Debug("Configuring Discord Incoming Webhook") | 		b.Log.Debug("Configuring Discord Incoming Webhook") | ||||||
| @@ -73,9 +76,10 @@ func (b *Bdiscord) Connect() error { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	serverName := strings.Replace(b.GetString("Server"), "ID:", "", -1) | ||||||
| 	b.Nick = userinfo.Username | 	b.Nick = userinfo.Username | ||||||
| 	for _, guild := range guilds { | 	for _, guild := range guilds { | ||||||
| 		if guild.Name == b.GetString("Server") { | 		if guild.Name == serverName || guild.ID == serverName { | ||||||
| 			b.Channels, err = b.c.GuildChannels(guild.ID) | 			b.Channels, err = b.c.GuildChannels(guild.ID) | ||||||
| 			b.guildID = guild.ID | 			b.guildID = guild.ID | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| @@ -86,6 +90,21 @@ func (b *Bdiscord) Connect() error { | |||||||
| 	for _, channel := range b.Channels { | 	for _, channel := range b.Channels { | ||||||
| 		b.Log.Debugf("found channel %#v", channel) | 		b.Log.Debugf("found channel %#v", channel) | ||||||
| 	} | 	} | ||||||
|  | 	// obtaining guild members and initializing nickname mapping | ||||||
|  | 	b.Lock() | ||||||
|  | 	defer b.Unlock() | ||||||
|  | 	members, err := b.c.GuildMembers(b.guildID, "", 1000) | ||||||
|  | 	if err != nil { | ||||||
|  | 		b.Log.Error("Error obtaining guild members", err) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	for _, member := range members { | ||||||
|  | 		b.userMemberMap[member.User.ID] = member | ||||||
|  | 		b.nickMemberMap[member.User.Username] = member | ||||||
|  | 		if member.Nick != "" { | ||||||
|  | 			b.nickMemberMap[member.Nick] = member | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -111,7 +130,7 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Make a action /me of the message | 	// Make a action /me of the message | ||||||
| 	if msg.Event == config.EVENT_USER_ACTION { | 	if msg.Event == config.EventUserAction { | ||||||
| 		msg.Text = "_" + msg.Text + "_" | 		msg.Text = "_" + msg.Text + "_" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -129,7 +148,7 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| 	// Use webhook to send the message | 	// Use webhook to send the message | ||||||
| 	if wID != "" { | 	if wID != "" { | ||||||
| 		// skip events | 		// skip events | ||||||
| 		if msg.Event != "" && msg.Event != config.EVENT_JOIN_LEAVE && msg.Event != config.EVENT_TOPIC_CHANGE { | 		if msg.Event != "" && msg.Event != config.EventJoinLeave && msg.Event != config.EventTopicChange { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
| 		b.Log.Debugf("Broadcasting using Webhook") | 		b.Log.Debugf("Broadcasting using Webhook") | ||||||
| @@ -145,6 +164,11 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		msg.Text = helper.ClipMessage(msg.Text, MessageLength) | 		msg.Text = helper.ClipMessage(msg.Text, MessageLength) | ||||||
|  | 		msg.Text = b.replaceUserMentions(msg.Text) | ||||||
|  | 		// discord username must be [0..32] max | ||||||
|  | 		if len(msg.Username) > 32 { | ||||||
|  | 			msg.Username = msg.Username[0:32] | ||||||
|  | 		} | ||||||
| 		err := b.c.WebhookExecute( | 		err := b.c.WebhookExecute( | ||||||
| 			wID, | 			wID, | ||||||
| 			wToken, | 			wToken, | ||||||
| @@ -160,7 +184,7 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| 	b.Log.Debugf("Broadcasting using token (API)") | 	b.Log.Debugf("Broadcasting using token (API)") | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
| @@ -181,6 +205,8 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	msg.Text = helper.ClipMessage(msg.Text, MessageLength) | 	msg.Text = helper.ClipMessage(msg.Text, MessageLength) | ||||||
|  | 	msg.Text = b.replaceUserMentions(msg.Text) | ||||||
|  |  | ||||||
| 	// Edit message | 	// Edit message | ||||||
| 	if msg.ID != "" { | 	if msg.ID != "" { | ||||||
| 		_, err := b.c.ChannelMessageEdit(channelID, msg.ID, msg.Username+msg.Text) | 		_, err := b.c.ChannelMessageEdit(channelID, msg.ID, msg.Username+msg.Text) | ||||||
| @@ -196,7 +222,7 @@ func (b *Bdiscord) Send(msg config.Message) (string, error) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bdiscord) messageDelete(s *discordgo.Session, m *discordgo.MessageDelete) { | func (b *Bdiscord) messageDelete(s *discordgo.Session, m *discordgo.MessageDelete) { | ||||||
| 	rmsg := config.Message{Account: b.Account, ID: m.ID, Event: config.EVENT_MSG_DELETE, Text: config.EVENT_MSG_DELETE} | 	rmsg := config.Message{Account: b.Account, ID: m.ID, Event: config.EventMsgDelete, Text: config.EventMsgDelete} | ||||||
| 	rmsg.Channel = b.getChannelName(m.ChannelID) | 	rmsg.Channel = b.getChannelName(m.ChannelID) | ||||||
| 	if b.UseChannelID { | 	if b.UseChannelID { | ||||||
| 		rmsg.Channel = "ID:" + m.ChannelID | 		rmsg.Channel = "ID:" + m.ChannelID | ||||||
| @@ -213,7 +239,7 @@ func (b *Bdiscord) messageUpdate(s *discordgo.Session, m *discordgo.MessageUpdat | |||||||
| 	// only when message is actually edited | 	// only when message is actually edited | ||||||
| 	if m.Message.EditedTimestamp != "" { | 	if m.Message.EditedTimestamp != "" { | ||||||
| 		b.Log.Debugf("Sending edit message") | 		b.Log.Debugf("Sending edit message") | ||||||
| 		m.Content = m.Content + b.GetString("EditSuffix") | 		m.Content += b.GetString("EditSuffix") | ||||||
| 		b.messageCreate(s, (*discordgo.MessageCreate)(m)) | 		b.messageCreate(s, (*discordgo.MessageCreate)(m)) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @@ -279,7 +305,7 @@ func (b *Bdiscord) messageCreate(s *discordgo.Session, m *discordgo.MessageCreat | |||||||
| 	var ok bool | 	var ok bool | ||||||
| 	rmsg.Text, ok = b.replaceAction(rmsg.Text) | 	rmsg.Text, ok = b.replaceAction(rmsg.Text) | ||||||
| 	if ok { | 	if ok { | ||||||
| 		rmsg.Event = config.EVENT_USER_ACTION | 		rmsg.Event = config.EventUserAction | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	b.Log.Debugf("<= Sending message from %s on %s to gateway", m.Author.Username, b.Account) | 	b.Log.Debugf("<= Sending message from %s on %s to gateway", m.Author.Username, b.Account) | ||||||
| @@ -293,6 +319,10 @@ func (b *Bdiscord) memberUpdate(s *discordgo.Session, m *discordgo.GuildMemberUp | |||||||
| 		b.Log.Debugf("%s: memberupdate: user %s (nick %s) changes nick to %s", b.Account, m.Member.User.Username, b.userMemberMap[m.Member.User.ID].Nick, m.Member.Nick) | 		b.Log.Debugf("%s: memberupdate: user %s (nick %s) changes nick to %s", b.Account, m.Member.User.Username, b.userMemberMap[m.Member.User.ID].Nick, m.Member.Nick) | ||||||
| 	} | 	} | ||||||
| 	b.userMemberMap[m.Member.User.ID] = m.Member | 	b.userMemberMap[m.Member.User.ID] = m.Member | ||||||
|  | 	b.nickMemberMap[m.Member.User.Username] = m.Member | ||||||
|  | 	if m.Member.Nick != "" { | ||||||
|  | 		b.nickMemberMap[m.Member.Nick] = m.Member | ||||||
|  | 	} | ||||||
| 	b.Unlock() | 	b.Unlock() | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -323,6 +353,18 @@ func (b *Bdiscord) getNick(user *discordgo.User) string { | |||||||
| 	return user.Username | 	return user.Username | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *Bdiscord) getGuildMemberByNick(nick string) (*discordgo.Member, error) { | ||||||
|  | 	b.Lock() | ||||||
|  | 	defer b.Unlock() | ||||||
|  | 	if _, ok := b.nickMemberMap[nick]; ok { | ||||||
|  | 		if b.nickMemberMap[nick] != nil { | ||||||
|  | 			return b.nickMemberMap[nick], nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil, errors.New("Couldn't find guild member with nick " + nick) // This will most likely get ignored by the caller | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *Bdiscord) getChannelID(name string) string { | func (b *Bdiscord) getChannelID(name string) string { | ||||||
| 	idcheck := strings.Split(name, "ID:") | 	idcheck := strings.Split(name, "ID:") | ||||||
| 	if len(idcheck) > 1 { | 	if len(idcheck) > 1 { | ||||||
| @@ -364,6 +406,34 @@ func (b *Bdiscord) replaceChannelMentions(text string) string { | |||||||
| 	return text | 	return text | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *Bdiscord) replaceUserMentions(text string) string { | ||||||
|  | 	re := regexp.MustCompile("@[^@]{1,32}") | ||||||
|  | 	text = re.ReplaceAllStringFunc(text, func(m string) string { | ||||||
|  | 		mention := strings.TrimSpace(m[1:]) | ||||||
|  | 		var member *discordgo.Member | ||||||
|  | 		var err error | ||||||
|  | 		for { | ||||||
|  | 			b.Log.Debugf("Testing mention: '%s'", mention) | ||||||
|  | 			member, err = b.getGuildMemberByNick(mention) | ||||||
|  | 			if err != nil { | ||||||
|  | 				lastSpace := strings.LastIndex(mention, " ") | ||||||
|  | 				if lastSpace == -1 { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 				mention = strings.TrimSpace(mention[0:lastSpace]) | ||||||
|  | 			} else { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if err != nil { | ||||||
|  | 			return m | ||||||
|  | 		} | ||||||
|  | 		return strings.Replace(m, "@"+mention, member.User.Mention(), -1) | ||||||
|  | 	}) | ||||||
|  | 	b.Log.Debugf("Message with mention replaced: %s", text) | ||||||
|  | 	return text | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *Bdiscord) replaceAction(text string) (string, bool) { | func (b *Bdiscord) replaceAction(text string) (string, bool) { | ||||||
| 	if strings.HasPrefix(text, "_") && strings.HasSuffix(text, "_") { | 	if strings.HasPrefix(text, "_") && strings.HasSuffix(text, "_") { | ||||||
| 		return strings.Replace(text, "_", "", -1), true | 		return strings.Replace(text, "_", "", -1), true | ||||||
| @@ -423,9 +493,16 @@ func (b *Bdiscord) handleUploadFile(msg *config.Message, channelID string) (stri | |||||||
| 	var err error | 	var err error | ||||||
| 	for _, f := range msg.Extra["file"] { | 	for _, f := range msg.Extra["file"] { | ||||||
| 		fi := f.(config.FileInfo) | 		fi := f.(config.FileInfo) | ||||||
| 		files := []*discordgo.File{} | 		file := discordgo.File{ | ||||||
| 		files = append(files, &discordgo.File{fi.Name, "", bytes.NewReader(*fi.Data)}) | 			Name:        fi.Name, | ||||||
| 		_, err = b.c.ChannelMessageSendComplex(channelID, &discordgo.MessageSend{Content: msg.Username + fi.Comment, Files: files}) | 			ContentType: "", | ||||||
|  | 			Reader:      bytes.NewReader(*fi.Data), | ||||||
|  | 		} | ||||||
|  | 		m := discordgo.MessageSend{ | ||||||
|  | 			Content: msg.Username + fi.Comment, | ||||||
|  | 			Files:   []*discordgo.File{&file}, | ||||||
|  | 		} | ||||||
|  | 		_, err = b.c.ChannelMessageSendComplex(channelID, &m) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return "", fmt.Errorf("file upload failed: %#v", err) | 			return "", fmt.Errorf("file upload failed: %#v", err) | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ func (b *Bgitter) JoinChannel(channel config.ChannelInfo) error { | |||||||
| 						Account: b.Account, Avatar: b.getAvatar(ev.Message.From.Username), UserID: ev.Message.From.ID, | 						Account: b.Account, Avatar: b.getAvatar(ev.Message.From.Username), UserID: ev.Message.From.ID, | ||||||
| 						ID: ev.Message.ID} | 						ID: ev.Message.ID} | ||||||
| 					if strings.HasPrefix(ev.Message.Text, "@"+ev.Message.From.Username) { | 					if strings.HasPrefix(ev.Message.Text, "@"+ev.Message.From.Username) { | ||||||
| 						rmsg.Event = config.EVENT_USER_ACTION | 						rmsg.Event = config.EventUserAction | ||||||
| 						rmsg.Text = strings.Replace(rmsg.Text, "@"+ev.Message.From.Username+" ", "", -1) | 						rmsg.Text = strings.Replace(rmsg.Text, "@"+ev.Message.From.Username+" ", "", -1) | ||||||
| 					} | 					} | ||||||
| 					b.Log.Debugf("<= Message is %#v", rmsg) | 					b.Log.Debugf("<= Message is %#v", rmsg) | ||||||
| @@ -100,7 +100,7 @@ func (b *Bgitter) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -40,29 +40,52 @@ func DownloadFileAuth(url string, auth string) (*[]byte, error) { | |||||||
| 	return &data, nil | 	return &data, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func SplitStringLength(input string, length int) string { | // GetSubLines splits messages in newline-delimited lines. If maxLineLength is | ||||||
| 	a := []rune(input) | // specified as non-zero GetSubLines will and also clip long lines to the | ||||||
| 	str := "" | // maximum length and insert a warning marker that the line was clipped. | ||||||
| 	for i, r := range a { | // | ||||||
| 		str = str + string(r) | // TODO: The current implementation has the inconvenient that it disregards | ||||||
| 		if i > 0 && (i+1)%length == 0 { | // word boundaries when splitting but this is hard to solve without potentially | ||||||
| 			str += "\n" | // breaking formatting and other stylistic effects. | ||||||
|  | func GetSubLines(message string, maxLineLength int) []string { | ||||||
|  | 	const clippingMessage = " <clipped message>" | ||||||
|  |  | ||||||
|  | 	var lines []string | ||||||
|  | 	for _, line := range strings.Split(strings.TrimSpace(message), "\n") { | ||||||
|  | 		if maxLineLength == 0 || len([]byte(line)) <= maxLineLength { | ||||||
|  | 			lines = append(lines, line) | ||||||
|  | 			continue | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		// !!! WARNING !!! | ||||||
|  | 		// Before touching the splitting logic below please ensure that you PROPERLY | ||||||
|  | 		// understand how strings, runes and range loops over strings work in Go. | ||||||
|  | 		// A good place to start is to read https://blog.golang.org/strings. :-) | ||||||
|  | 		var splitStart int | ||||||
|  | 		var startOfPreviousRune int | ||||||
|  | 		for i := range line { | ||||||
|  | 			if i-splitStart > maxLineLength-len([]byte(clippingMessage)) { | ||||||
|  | 				lines = append(lines, line[splitStart:startOfPreviousRune]+clippingMessage) | ||||||
|  | 				splitStart = startOfPreviousRune | ||||||
|  | 			} | ||||||
|  | 			startOfPreviousRune = i | ||||||
|  | 		} | ||||||
|  | 		// This last append is safe to do without looking at the remaining byte-length | ||||||
|  | 		// as we assume that the byte-length of the last rune will never exceed that of | ||||||
|  | 		// the byte-length of the clipping message. | ||||||
|  | 		lines = append(lines, line[splitStart:]) | ||||||
| 	} | 	} | ||||||
| 	return str | 	return lines | ||||||
| } | } | ||||||
|  |  | ||||||
| // handle all the stuff we put into extra | // handle all the stuff we put into extra | ||||||
| func HandleExtra(msg *config.Message, general *config.Protocol) []config.Message { | func HandleExtra(msg *config.Message, general *config.Protocol) []config.Message { | ||||||
| 	extra := msg.Extra | 	extra := msg.Extra | ||||||
| 	rmsg := []config.Message{} | 	rmsg := []config.Message{} | ||||||
| 	if len(extra[config.EVENT_FILE_FAILURE_SIZE]) > 0 { | 	for _, f := range extra[config.EventFileFailureSize] { | ||||||
| 		for _, f := range extra[config.EVENT_FILE_FAILURE_SIZE] { | 		fi := f.(config.FileInfo) | ||||||
| 			fi := f.(config.FileInfo) | 		text := fmt.Sprintf("file %s too big to download (%#v > allowed size: %#v)", fi.Name, fi.Size, general.MediaDownloadSize) | ||||||
| 			text := fmt.Sprintf("file %s too big to download (%#v > allowed size: %#v)", fi.Name, fi.Size, general.MediaDownloadSize) | 		rmsg = append(rmsg, config.Message{Text: text, Username: "<system> ", Channel: msg.Channel, Account: msg.Account}) | ||||||
| 			rmsg = append(rmsg, config.Message{Text: text, Username: "<system> ", Channel: msg.Channel, Account: msg.Account}) |  | ||||||
| 		} |  | ||||||
| 		return rmsg |  | ||||||
| 	} | 	} | ||||||
| 	return rmsg | 	return rmsg | ||||||
| } | } | ||||||
| @@ -90,7 +113,7 @@ func HandleDownloadSize(flog *log.Entry, msg *config.Message, name string, size | |||||||
| 	} | 	} | ||||||
| 	flog.Debugf("Trying to download %#v with size %#v", name, size) | 	flog.Debugf("Trying to download %#v with size %#v", name, size) | ||||||
| 	if int(size) > general.MediaDownloadSize { | 	if int(size) > general.MediaDownloadSize { | ||||||
| 		msg.Event = config.EVENT_FILE_FAILURE_SIZE | 		msg.Event = config.EventFileFailureSize | ||||||
| 		msg.Extra[msg.Event] = append(msg.Extra[msg.Event], config.FileInfo{Name: name, Comment: msg.Text, Size: size}) | 		msg.Extra[msg.Event] = append(msg.Extra[msg.Event], config.FileInfo{Name: name, Comment: msg.Text, Size: size}) | ||||||
| 		return fmt.Errorf("File %#v to large to download (%#v). MediaDownloadSize is %#v", name, size, general.MediaDownloadSize) | 		return fmt.Errorf("File %#v to large to download (%#v). MediaDownloadSize is %#v", name, size, general.MediaDownloadSize) | ||||||
| 	} | 	} | ||||||
| @@ -100,7 +123,7 @@ func HandleDownloadSize(flog *log.Entry, msg *config.Message, name string, size | |||||||
| func HandleDownloadData(flog *log.Entry, msg *config.Message, name, comment, url string, data *[]byte, general *config.Protocol) { | func HandleDownloadData(flog *log.Entry, msg *config.Message, name, comment, url string, data *[]byte, general *config.Protocol) { | ||||||
| 	var avatar bool | 	var avatar bool | ||||||
| 	flog.Debugf("Download OK %#v %#v", name, len(*data)) | 	flog.Debugf("Download OK %#v %#v", name, len(*data)) | ||||||
| 	if msg.Event == config.EVENT_AVATAR_DOWNLOAD { | 	if msg.Event == config.EventAvatarDownload { | ||||||
| 		avatar = true | 		avatar = true | ||||||
| 	} | 	} | ||||||
| 	msg.Extra["file"] = append(msg.Extra["file"], config.FileInfo{Name: name, Data: data, URL: url, Comment: comment, Avatar: avatar}) | 	msg.Extra["file"] = append(msg.Extra["file"], config.FileInfo{Name: name, Data: data, URL: url, Comment: comment, Avatar: avatar}) | ||||||
|   | |||||||
							
								
								
									
										105
									
								
								bridge/helper/helper_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,105 @@ | |||||||
|  | package helper | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | const testLineLength = 64 | ||||||
|  |  | ||||||
|  | var ( | ||||||
|  | 	lineSplittingTestCases = map[string]struct { | ||||||
|  | 		input          string | ||||||
|  | 		splitOutput    []string | ||||||
|  | 		nonSplitOutput []string | ||||||
|  | 	}{ | ||||||
|  | 		"Short single-line message": { | ||||||
|  | 			input:          "short", | ||||||
|  | 			splitOutput:    []string{"short"}, | ||||||
|  | 			nonSplitOutput: []string{"short"}, | ||||||
|  | 		}, | ||||||
|  | 		"Long single-line message": { | ||||||
|  | 			input: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.", | ||||||
|  | 			splitOutput: []string{ | ||||||
|  | 				"Lorem ipsum dolor sit amet, consectetur adipis <clipped message>", | ||||||
|  | 				"cing elit, sed do eiusmod tempor incididunt ut <clipped message>", | ||||||
|  | 				" labore et dolore magna aliqua.", | ||||||
|  | 			}, | ||||||
|  | 			nonSplitOutput: []string{"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua."}, | ||||||
|  | 		}, | ||||||
|  | 		"Short multi-line message": { | ||||||
|  | 			input: "I\ncan't\nget\nno\nsatisfaction!", | ||||||
|  | 			splitOutput: []string{ | ||||||
|  | 				"I", | ||||||
|  | 				"can't", | ||||||
|  | 				"get", | ||||||
|  | 				"no", | ||||||
|  | 				"satisfaction!", | ||||||
|  | 			}, | ||||||
|  | 			nonSplitOutput: []string{ | ||||||
|  | 				"I", | ||||||
|  | 				"can't", | ||||||
|  | 				"get", | ||||||
|  | 				"no", | ||||||
|  | 				"satisfaction!", | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		"Long multi-line message": { | ||||||
|  | 			input: "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.\n" + | ||||||
|  | 				"Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.\n" + | ||||||
|  | 				"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.\n" + | ||||||
|  | 				"Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.", | ||||||
|  | 			splitOutput: []string{ | ||||||
|  | 				"Lorem ipsum dolor sit amet, consectetur adipis <clipped message>", | ||||||
|  | 				"cing elit, sed do eiusmod tempor incididunt ut <clipped message>", | ||||||
|  | 				" labore et dolore magna aliqua.", | ||||||
|  | 				"Ut enim ad minim veniam, quis nostrud exercita <clipped message>", | ||||||
|  | 				"tion ullamco laboris nisi ut aliquip ex ea com <clipped message>", | ||||||
|  | 				"modo consequat.", | ||||||
|  | 				"Duis aute irure dolor in reprehenderit in volu <clipped message>", | ||||||
|  | 				"ptate velit esse cillum dolore eu fugiat nulla <clipped message>", | ||||||
|  | 				" pariatur.", | ||||||
|  | 				"Excepteur sint occaecat cupidatat non proident <clipped message>", | ||||||
|  | 				", sunt in culpa qui officia deserunt mollit an <clipped message>", | ||||||
|  | 				"im id est laborum.", | ||||||
|  | 			}, | ||||||
|  | 			nonSplitOutput: []string{ | ||||||
|  | 				"Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.", | ||||||
|  | 				"Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.", | ||||||
|  | 				"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.", | ||||||
|  | 				"Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.", | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		"Message ending with new-line.": { | ||||||
|  | 			input:          "Newline ending\n", | ||||||
|  | 			splitOutput:    []string{"Newline ending"}, | ||||||
|  | 			nonSplitOutput: []string{"Newline ending"}, | ||||||
|  | 		}, | ||||||
|  | 		"Long message containing UTF-8 multi-byte runes": { | ||||||
|  | 			input: "不布人個我此而及單石業喜資富下我河下日沒一我臺空達的常景便物沒為……子大我別名解成?生賣的全直黑,我自我結毛分洲了世當,是政福那是東;斯說", | ||||||
|  | 			splitOutput: []string{ | ||||||
|  | 				"不布人個我此而及單石業喜資富下 <clipped message>", | ||||||
|  | 				"我河下日沒一我臺空達的常景便物 <clipped message>", | ||||||
|  | 				"沒為……子大我別名解成?生賣的 <clipped message>", | ||||||
|  | 				"全直黑,我自我結毛分洲了世當, <clipped message>", | ||||||
|  | 				"是政福那是東;斯說", | ||||||
|  | 			}, | ||||||
|  | 			nonSplitOutput: []string{"不布人個我此而及單石業喜資富下我河下日沒一我臺空達的常景便物沒為……子大我別名解成?生賣的全直黑,我自我結毛分洲了世當,是政福那是東;斯說"}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestGetSubLines(t *testing.T) { | ||||||
|  | 	for testname, testcase := range lineSplittingTestCases { | ||||||
|  | 		splitLines := GetSubLines(testcase.input, testLineLength) | ||||||
|  | 		assert.Equalf(t, testcase.splitOutput, splitLines, "'%s' testcase should give expected lines with splitting.", testname) | ||||||
|  | 		for _, splitLine := range splitLines { | ||||||
|  | 			byteLength := len([]byte(splitLine)) | ||||||
|  | 			assert.True(t, byteLength <= testLineLength, "Splitted line '%s' of testcase '%s' should not exceed the maximum byte-length (%d vs. %d).", splitLine, testcase, byteLength, testLineLength) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		nonSplitLines := GetSubLines(testcase.input, 0) | ||||||
|  | 		assert.Equalf(t, testcase.nonSplitOutput, nonSplitLines, "'%s' testcase should give expected lines without splitting.", testname) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -1,61 +0,0 @@ | |||||||
| package birc |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"strings" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| /* |  | ||||||
| func tableformatter(nicks []string, nicksPerRow int, continued bool) string { |  | ||||||
| 	result := "|IRC users" |  | ||||||
| 	if continued { |  | ||||||
| 		result = "|(continued)" |  | ||||||
| 	} |  | ||||||
| 	for i := 0; i < 2; i++ { |  | ||||||
| 		for j := 1; j <= nicksPerRow && j <= len(nicks); j++ { |  | ||||||
| 			if i == 0 { |  | ||||||
| 				result += "|" |  | ||||||
| 			} else { |  | ||||||
| 				result += ":-|" |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		result += "\r\n|" |  | ||||||
| 	} |  | ||||||
| 	result += nicks[0] + "|" |  | ||||||
| 	for i := 1; i < len(nicks); i++ { |  | ||||||
| 		if i%nicksPerRow == 0 { |  | ||||||
| 			result += "\r\n|" + nicks[i] + "|" |  | ||||||
| 		} else { |  | ||||||
| 			result += nicks[i] + "|" |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return result |  | ||||||
| } |  | ||||||
| */ |  | ||||||
|  |  | ||||||
| func plainformatter(nicks []string, nicksPerRow int) string { |  | ||||||
| 	return strings.Join(nicks, ", ") + " currently on IRC" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func IsMarkup(message string) bool { |  | ||||||
| 	switch message[0] { |  | ||||||
| 	case '|': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '#': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '_': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '*': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '~': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '-': |  | ||||||
| 		fallthrough |  | ||||||
| 	case ':': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '>': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '=': |  | ||||||
| 		return true |  | ||||||
| 	} |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
| @@ -13,7 +13,6 @@ import ( | |||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 	"unicode/utf8" |  | ||||||
|  |  | ||||||
| 	"github.com/42wim/matterbridge/bridge" | 	"github.com/42wim/matterbridge/bridge" | ||||||
| 	"github.com/42wim/matterbridge/bridge/config" | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
| @@ -21,17 +20,20 @@ import ( | |||||||
| 	"github.com/dfordsoft/golib/ic" | 	"github.com/dfordsoft/golib/ic" | ||||||
| 	"github.com/lrstanley/girc" | 	"github.com/lrstanley/girc" | ||||||
| 	"github.com/paulrosania/go-charset/charset" | 	"github.com/paulrosania/go-charset/charset" | ||||||
| 	_ "github.com/paulrosania/go-charset/data" |  | ||||||
| 	"github.com/saintfish/chardet" | 	"github.com/saintfish/chardet" | ||||||
|  |  | ||||||
|  | 	// We need to import the 'data' package as an implicit dependency. | ||||||
|  | 	// See: https://godoc.org/github.com/paulrosania/go-charset/charset | ||||||
|  | 	_ "github.com/paulrosania/go-charset/data" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Birc struct { | type Birc struct { | ||||||
| 	i                                         *girc.Client | 	i                                         *girc.Client | ||||||
| 	Nick                                      string | 	Nick                                      string | ||||||
| 	names                                     map[string][]string | 	names                                     map[string][]string | ||||||
| 	connected                                 chan struct{} | 	connected                                 chan error | ||||||
| 	Local                                     chan config.Message // local queue for flood control | 	Local                                     chan config.Message // local queue for flood control | ||||||
| 	FirstConnection                           bool | 	FirstConnection, authDone                 bool | ||||||
| 	MessageDelay, MessageQueue, MessageLength int | 	MessageDelay, MessageQueue, MessageLength int | ||||||
|  |  | ||||||
| 	*bridge.Config | 	*bridge.Config | ||||||
| @@ -42,7 +44,7 @@ func New(cfg *bridge.Config) bridge.Bridger { | |||||||
| 	b.Config = cfg | 	b.Config = cfg | ||||||
| 	b.Nick = b.GetString("Nick") | 	b.Nick = b.GetString("Nick") | ||||||
| 	b.names = make(map[string][]string) | 	b.names = make(map[string][]string) | ||||||
| 	b.connected = make(chan struct{}) | 	b.connected = make(chan error) | ||||||
| 	if b.GetInt("MessageDelay") == 0 { | 	if b.GetInt("MessageDelay") == 0 { | ||||||
| 		b.MessageDelay = 1300 | 		b.MessageDelay = 1300 | ||||||
| 	} else { | 	} else { | ||||||
| @@ -63,8 +65,7 @@ func New(cfg *bridge.Config) bridge.Bridger { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) Command(msg *config.Message) string { | func (b *Birc) Command(msg *config.Message) string { | ||||||
| 	switch msg.Text { | 	if msg.Text == "!users" { | ||||||
| 	case "!users": |  | ||||||
| 		b.i.Handlers.Add(girc.RPL_NAMREPLY, b.storeNames) | 		b.i.Handlers.Add(girc.RPL_NAMREPLY, b.storeNames) | ||||||
| 		b.i.Handlers.Add(girc.RPL_ENDOFNAMES, b.endNames) | 		b.i.Handlers.Add(girc.RPL_ENDOFNAMES, b.endNames) | ||||||
| 		b.i.Cmd.SendRaw("NAMES " + msg.Channel) | 		b.i.Cmd.SendRaw("NAMES " + msg.Channel) | ||||||
| @@ -106,38 +107,44 @@ func (b *Birc) Connect() error { | |||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	if b.GetBool("UseSASL") { | 	if b.GetBool("UseSASL") { | ||||||
| 		i.Config.SASL = &girc.SASLPlain{b.GetString("NickServNick"), b.GetString("NickServPassword")} | 		i.Config.SASL = &girc.SASLPlain{ | ||||||
|  | 			User: b.GetString("NickServNick"), | ||||||
|  | 			Pass: b.GetString("NickServPassword"), | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	i.Handlers.Add(girc.RPL_WELCOME, b.handleNewConnection) | 	i.Handlers.Add(girc.RPL_WELCOME, b.handleNewConnection) | ||||||
| 	i.Handlers.Add(girc.RPL_ENDOFMOTD, b.handleOtherAuth) | 	i.Handlers.Add(girc.RPL_ENDOFMOTD, b.handleOtherAuth) | ||||||
| 	i.Handlers.Add(girc.ALL_EVENTS, b.handleOther) | 	i.Handlers.Add(girc.ALL_EVENTS, b.handleOther) | ||||||
|  |  | ||||||
| 	go func() { | 	go func() { | ||||||
| 		for { | 		for { | ||||||
| 			if err := i.Connect(); err != nil { | 			if err := i.Connect(); err != nil { | ||||||
| 				b.Log.Errorf("disconnect: error: %s", err) | 				b.Log.Errorf("disconnect: error: %s", err) | ||||||
|  | 				if b.FirstConnection { | ||||||
|  | 					b.connected <- err | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				b.Log.Info("disconnect: client requested quit") | 				b.Log.Info("disconnect: client requested quit") | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			b.Log.Info("reconnecting in 30 seconds...") | 			b.Log.Info("reconnecting in 30 seconds...") | ||||||
| 			time.Sleep(30 * time.Second) | 			time.Sleep(30 * time.Second) | ||||||
| 			i.Handlers.Clear(girc.RPL_WELCOME) | 			i.Handlers.Clear(girc.RPL_WELCOME) | ||||||
| 			i.Handlers.Add(girc.RPL_WELCOME, func(client *girc.Client, event girc.Event) { | 			i.Handlers.Add(girc.RPL_WELCOME, func(client *girc.Client, event girc.Event) { | ||||||
| 				b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EVENT_REJOIN_CHANNELS} | 				b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EventRejoinChannels} | ||||||
| 				// set our correct nick on reconnect if necessary | 				// set our correct nick on reconnect if necessary | ||||||
| 				b.Nick = event.Source.Name | 				b.Nick = event.Source.Name | ||||||
| 			}) | 			}) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
| 	b.i = i | 	b.i = i | ||||||
| 	select { | 	err = <-b.connected | ||||||
| 	case <-b.connected: | 	if err != nil { | ||||||
| 		b.Log.Info("Connection succeeded") | 		return fmt.Errorf("connection failed %s", err) | ||||||
| 	case <-time.After(time.Second * 30): |  | ||||||
| 		return fmt.Errorf("connection timed out") |  | ||||||
| 	} | 	} | ||||||
| 	//i.Debug = false | 	b.Log.Info("Connection succeeded") | ||||||
|  | 	b.FirstConnection = false | ||||||
| 	if b.GetInt("DebugLevel") == 0 { | 	if b.GetInt("DebugLevel") == 0 { | ||||||
| 		i.Handlers.Clear(girc.ALL_EVENTS) | 		i.Handlers.Clear(girc.ALL_EVENTS) | ||||||
| 	} | 	} | ||||||
| @@ -152,18 +159,26 @@ func (b *Birc) Disconnect() error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) JoinChannel(channel config.ChannelInfo) error { | func (b *Birc) JoinChannel(channel config.ChannelInfo) error { | ||||||
|  | 	// need to check if we have nickserv auth done before joining channels | ||||||
|  | 	for { | ||||||
|  | 		if b.authDone { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		time.Sleep(time.Second) | ||||||
|  | 	} | ||||||
| 	if channel.Options.Key != "" { | 	if channel.Options.Key != "" { | ||||||
| 		b.Log.Debugf("using key %s for channel %s", channel.Options.Key, channel.Name) | 		b.Log.Debugf("using key %s for channel %s", channel.Options.Key, channel.Name) | ||||||
| 		b.i.Cmd.JoinKey(channel.Name, channel.Options.Key) | 		b.i.Cmd.JoinKey(channel.Name, channel.Options.Key) | ||||||
| 	} else { | 	} else { | ||||||
| 		b.i.Cmd.Join(channel.Name) | 		b.i.Cmd.Join(channel.Name) | ||||||
| 	} | 	} | ||||||
|  | 	b.authDone = false | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) Send(msg config.Message) (string, error) { | func (b *Birc) Send(msg config.Message) (string, error) { | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -220,25 +235,23 @@ func (b *Birc) Send(msg config.Message) (string, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// split long messages on messageLength, to avoid clipped messages #281 | 	var msgLines []string | ||||||
| 	if b.GetBool("MessageSplit") { | 	if b.GetBool("MessageSplit") { | ||||||
| 		msg.Text = helper.SplitStringLength(msg.Text, b.MessageLength) | 		msgLines = helper.GetSubLines(msg.Text, b.MessageLength) | ||||||
|  | 	} else { | ||||||
|  | 		msgLines = helper.GetSubLines(msg.Text, 0) | ||||||
| 	} | 	} | ||||||
| 	for _, text := range strings.Split(msg.Text, "\n") { | 	for i := range msgLines { | ||||||
| 		if len(text) > b.MessageLength { | 		if len(b.Local) >= b.MessageQueue { | ||||||
| 			text = text[:b.MessageLength-len(" <message clipped>")] |  | ||||||
| 			if r, size := utf8.DecodeLastRuneInString(text); r == utf8.RuneError { |  | ||||||
| 				text = text[:len(text)-size] |  | ||||||
| 			} |  | ||||||
| 			text += " <message clipped>" |  | ||||||
| 		} |  | ||||||
| 		if len(b.Local) < b.MessageQueue { |  | ||||||
| 			if len(b.Local) == b.MessageQueue-1 { |  | ||||||
| 				text = text + " <message clipped>" |  | ||||||
| 			} |  | ||||||
| 			b.Local <- config.Message{Text: text, Username: msg.Username, Channel: msg.Channel, Event: msg.Event} |  | ||||||
| 		} else { |  | ||||||
| 			b.Log.Debugf("flooding, dropping message (queue at %d)", len(b.Local)) | 			b.Log.Debugf("flooding, dropping message (queue at %d)", len(b.Local)) | ||||||
|  | 			return "", nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		b.Local <- config.Message{ | ||||||
|  | 			Text:     msgLines[i], | ||||||
|  | 			Username: msg.Username, | ||||||
|  | 			Channel:  msg.Channel, | ||||||
|  | 			Event:    msg.Event, | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return "", nil | 	return "", nil | ||||||
| @@ -255,7 +268,7 @@ func (b *Birc) doSend() { | |||||||
| 			colorCode := checksum%14 + 2 // quick fix - prevent white or black color codes | 			colorCode := checksum%14 + 2 // quick fix - prevent white or black color codes | ||||||
| 			username = fmt.Sprintf("\x03%02d%s\x0F", colorCode, msg.Username) | 			username = fmt.Sprintf("\x03%02d%s\x0F", colorCode, msg.Username) | ||||||
| 		} | 		} | ||||||
| 		if msg.Event == config.EVENT_USER_ACTION { | 		if msg.Event == config.EventUserAction { | ||||||
| 			b.i.Cmd.Action(msg.Channel, username+msg.Text) | 			b.i.Cmd.Action(msg.Channel, username+msg.Text) | ||||||
| 		} else { | 		} else { | ||||||
| 			b.Log.Debugf("Sending to channel %s", msg.Channel) | 			b.Log.Debugf("Sending to channel %s", msg.Channel) | ||||||
| @@ -268,14 +281,12 @@ func (b *Birc) endNames(client *girc.Client, event girc.Event) { | |||||||
| 	channel := event.Params[1] | 	channel := event.Params[1] | ||||||
| 	sort.Strings(b.names[channel]) | 	sort.Strings(b.names[channel]) | ||||||
| 	maxNamesPerPost := (300 / b.nicksPerRow()) * b.nicksPerRow() | 	maxNamesPerPost := (300 / b.nicksPerRow()) * b.nicksPerRow() | ||||||
| 	continued := false |  | ||||||
| 	for len(b.names[channel]) > maxNamesPerPost { | 	for len(b.names[channel]) > maxNamesPerPost { | ||||||
| 		b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel][0:maxNamesPerPost], continued), | 		b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel][0:maxNamesPerPost]), | ||||||
| 			Channel: channel, Account: b.Account} | 			Channel: channel, Account: b.Account} | ||||||
| 		b.names[channel] = b.names[channel][maxNamesPerPost:] | 		b.names[channel] = b.names[channel][maxNamesPerPost:] | ||||||
| 		continued = true |  | ||||||
| 	} | 	} | ||||||
| 	b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel], continued), | 	b.Remote <- config.Message{Username: b.Nick, Text: b.formatnicks(b.names[channel]), | ||||||
| 		Channel: channel, Account: b.Account} | 		Channel: channel, Account: b.Account} | ||||||
| 	b.names[channel] = nil | 	b.names[channel] = nil | ||||||
| 	b.i.Handlers.Clear(girc.RPL_NAMREPLY) | 	b.i.Handlers.Clear(girc.RPL_NAMREPLY) | ||||||
| @@ -287,7 +298,6 @@ func (b *Birc) handleNewConnection(client *girc.Client, event girc.Event) { | |||||||
| 	i := b.i | 	i := b.i | ||||||
| 	b.Nick = event.Params[0] | 	b.Nick = event.Params[0] | ||||||
|  |  | ||||||
| 	i.Handlers.Add(girc.RPL_ENDOFMOTD, b.handleOtherAuth) |  | ||||||
| 	i.Handlers.Add("PRIVMSG", b.handlePrivMsg) | 	i.Handlers.Add("PRIVMSG", b.handlePrivMsg) | ||||||
| 	i.Handlers.Add("CTCP_ACTION", b.handlePrivMsg) | 	i.Handlers.Add("CTCP_ACTION", b.handlePrivMsg) | ||||||
| 	i.Handlers.Add(girc.RPL_TOPICWHOTIME, b.handleTopicWhoTime) | 	i.Handlers.Add(girc.RPL_TOPICWHOTIME, b.handleTopicWhoTime) | ||||||
| @@ -296,8 +306,6 @@ func (b *Birc) handleNewConnection(client *girc.Client, event girc.Event) { | |||||||
| 	i.Handlers.Add("PART", b.handleJoinPart) | 	i.Handlers.Add("PART", b.handleJoinPart) | ||||||
| 	i.Handlers.Add("QUIT", b.handleJoinPart) | 	i.Handlers.Add("QUIT", b.handleJoinPart) | ||||||
| 	i.Handlers.Add("KICK", b.handleJoinPart) | 	i.Handlers.Add("KICK", b.handleJoinPart) | ||||||
| 	// we are now fully connected |  | ||||||
| 	b.connected <- struct{}{} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { | func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { | ||||||
| @@ -309,13 +317,13 @@ func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { | |||||||
| 	if event.Command == "KICK" && event.Params[1] == b.Nick { | 	if event.Command == "KICK" && event.Params[1] == b.Nick { | ||||||
| 		b.Log.Infof("Got kicked from %s by %s", channel, event.Source.Name) | 		b.Log.Infof("Got kicked from %s by %s", channel, event.Source.Name) | ||||||
| 		time.Sleep(time.Duration(b.GetInt("RejoinDelay")) * time.Second) | 		time.Sleep(time.Duration(b.GetInt("RejoinDelay")) * time.Second) | ||||||
| 		b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: channel, Account: b.Account, Event: config.EVENT_REJOIN_CHANNELS} | 		b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: channel, Account: b.Account, Event: config.EventRejoinChannels} | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	if event.Command == "QUIT" { | 	if event.Command == "QUIT" { | ||||||
| 		if event.Source.Name == b.Nick && strings.Contains(event.Trailing, "Ping timeout") { | 		if event.Source.Name == b.Nick && strings.Contains(event.Trailing, "Ping timeout") { | ||||||
| 			b.Log.Infof("%s reconnecting ..", b.Account) | 			b.Log.Infof("%s reconnecting ..", b.Account) | ||||||
| 			b.Remote <- config.Message{Username: "system", Text: "reconnect", Channel: channel, Account: b.Account, Event: config.EVENT_FAILURE} | 			b.Remote <- config.Message{Username: "system", Text: "reconnect", Channel: channel, Account: b.Account, Event: config.EventFailure} | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -324,7 +332,7 @@ func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { | |||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		b.Log.Debugf("<= Sending JOIN_LEAVE event from %s to gateway", b.Account) | 		b.Log.Debugf("<= Sending JOIN_LEAVE event from %s to gateway", b.Account) | ||||||
| 		msg := config.Message{Username: "system", Text: event.Source.Name + " " + strings.ToLower(event.Command) + "s", Channel: channel, Account: b.Account, Event: config.EVENT_JOIN_LEAVE} | 		msg := config.Message{Username: "system", Text: event.Source.Name + " " + strings.ToLower(event.Command) + "s", Channel: channel, Account: b.Account, Event: config.EventJoinLeave} | ||||||
| 		b.Log.Debugf("<= Message is %#v", msg) | 		b.Log.Debugf("<= Message is %#v", msg) | ||||||
| 		b.Remote <- msg | 		b.Remote <- msg | ||||||
| 		return | 		return | ||||||
| @@ -334,8 +342,7 @@ func (b *Birc) handleJoinPart(client *girc.Client, event girc.Event) { | |||||||
|  |  | ||||||
| func (b *Birc) handleNotice(client *girc.Client, event girc.Event) { | func (b *Birc) handleNotice(client *girc.Client, event girc.Event) { | ||||||
| 	if strings.Contains(event.String(), "This nickname is registered") && event.Source.Name == b.GetString("NickServNick") { | 	if strings.Contains(event.String(), "This nickname is registered") && event.Source.Name == b.GetString("NickServNick") { | ||||||
| 		b.Log.Debugf("Sending identify to nickserv %s", b.GetString("NickServNick")) | 		b.handleNickServ() | ||||||
| 		b.i.Cmd.Message(b.GetString("NickServNick"), "IDENTIFY "+b.GetString("NickServPassword")) |  | ||||||
| 	} else { | 	} else { | ||||||
| 		b.handlePrivMsg(client, event) | 		b.handlePrivMsg(client, event) | ||||||
| 	} | 	} | ||||||
| @@ -357,10 +364,9 @@ func (b *Birc) handleOther(client *girc.Client, event girc.Event) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) handleOtherAuth(client *girc.Client, event girc.Event) { | func (b *Birc) handleOtherAuth(client *girc.Client, event girc.Event) { | ||||||
| 	if strings.EqualFold(b.GetString("NickServNick"), "Q@CServe.quakenet.org") { | 	b.handleNickServ() | ||||||
| 		b.Log.Debugf("Authenticating %s against %s", b.GetString("NickServUsername"), b.GetString("NickServNick")) | 	// we are now fully connected | ||||||
| 		b.i.Cmd.Message(b.GetString("NickServNick"), "AUTH "+b.GetString("NickServUsername")+" "+b.GetString("NickServPassword")) | 	b.connected <- nil | ||||||
| 	} |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) skipPrivMsg(event girc.Event) bool { | func (b *Birc) skipPrivMsg(event girc.Event) bool { | ||||||
| @@ -391,7 +397,7 @@ func (b *Birc) handlePrivMsg(client *girc.Client, event girc.Event) { | |||||||
|  |  | ||||||
| 	// set action event | 	// set action event | ||||||
| 	if event.IsAction() { | 	if event.IsAction() { | ||||||
| 		rmsg.Event = config.EVENT_USER_ACTION | 		rmsg.Event = config.EventUserAction | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// strip action, we made an event if it was an action | 	// strip action, we made an event if it was an action | ||||||
| @@ -461,6 +467,20 @@ func (b *Birc) storeNames(client *girc.Client, event girc.Event) { | |||||||
| 		strings.Split(strings.TrimSpace(event.Trailing), " ")...) | 		strings.Split(strings.TrimSpace(event.Trailing), " ")...) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Birc) formatnicks(nicks []string, continued bool) string { | func (b *Birc) formatnicks(nicks []string) string { | ||||||
| 	return plainformatter(nicks, b.nicksPerRow()) | 	return strings.Join(nicks, ", ") + " currently on IRC" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Birc) handleNickServ() { | ||||||
|  | 	if !b.GetBool("UseSASL") && b.GetString("NickServNick") != "" && b.GetString("NickServPassword") != "" { | ||||||
|  | 		b.Log.Debugf("Sending identify to nickserv %s", b.GetString("NickServNick")) | ||||||
|  | 		b.i.Cmd.Message(b.GetString("NickServNick"), "IDENTIFY "+b.GetString("NickServPassword")) | ||||||
|  | 	} | ||||||
|  | 	if strings.EqualFold(b.GetString("NickServNick"), "Q@CServe.quakenet.org") { | ||||||
|  | 		b.Log.Debugf("Authenticating %s against %s", b.GetString("NickServUsername"), b.GetString("NickServNick")) | ||||||
|  | 		b.i.Cmd.Message(b.GetString("NickServNick"), "AUTH "+b.GetString("NickServUsername")+" "+b.GetString("NickServPassword")) | ||||||
|  | 	} | ||||||
|  | 	// give nickserv some slack | ||||||
|  | 	time.Sleep(time.Second * 5) | ||||||
|  | 	b.authDone = true | ||||||
| } | } | ||||||
|   | |||||||
| @@ -72,9 +72,12 @@ func (b *Bmatrix) Send(msg config.Message) (string, error) { | |||||||
| 	b.Log.Debugf("Channel %s maps to channel id %s", msg.Channel, channel) | 	b.Log.Debugf("Channel %s maps to channel id %s", msg.Channel, channel) | ||||||
|  |  | ||||||
| 	// Make a action /me of the message | 	// Make a action /me of the message | ||||||
| 	if msg.Event == config.EVENT_USER_ACTION { | 	if msg.Event == config.EventUserAction { | ||||||
| 		resp, err := b.mc.SendMessageEvent(channel, "m.room.message", | 		m := matrix.TextMessage{ | ||||||
| 			matrix.TextMessage{"m.emote", msg.Username + msg.Text}) | 			MsgType: "m.emote", | ||||||
|  | 			Body:    msg.Username + msg.Text, | ||||||
|  | 		} | ||||||
|  | 		resp, err := b.mc.SendMessageEvent(channel, "m.room.message", m) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return "", err | 			return "", err | ||||||
| 		} | 		} | ||||||
| @@ -82,7 +85,7 @@ func (b *Bmatrix) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
| @@ -126,7 +129,7 @@ func (b *Bmatrix) getRoomID(channel string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmatrix) handlematrix() error { | func (b *Bmatrix) handlematrix() { | ||||||
| 	syncer := b.mc.Syncer.(*matrix.DefaultSyncer) | 	syncer := b.mc.Syncer.(*matrix.DefaultSyncer) | ||||||
| 	syncer.OnEventType("m.room.redaction", b.handleEvent) | 	syncer.OnEventType("m.room.redaction", b.handleEvent) | ||||||
| 	syncer.OnEventType("m.room.message", b.handleEvent) | 	syncer.OnEventType("m.room.message", b.handleEvent) | ||||||
| @@ -137,7 +140,6 @@ func (b *Bmatrix) handlematrix() error { | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
| 	return nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmatrix) handleEvent(ev *matrix.Event) { | func (b *Bmatrix) handleEvent(ev *matrix.Event) { | ||||||
| @@ -158,7 +160,8 @@ func (b *Bmatrix) handleEvent(ev *matrix.Event) { | |||||||
|  |  | ||||||
| 		// Text must be a string | 		// Text must be a string | ||||||
| 		if rmsg.Text, ok = ev.Content["body"].(string); !ok { | 		if rmsg.Text, ok = ev.Content["body"].(string); !ok { | ||||||
| 			b.Log.Errorf("Content[body] wasn't a %T ?", rmsg.Text) | 			b.Log.Errorf("Content[body] is not a string: %T\n%#v", | ||||||
|  | 				ev.Content["body"], ev.Content) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @@ -170,16 +173,16 @@ func (b *Bmatrix) handleEvent(ev *matrix.Event) { | |||||||
|  |  | ||||||
| 		// Delete event | 		// Delete event | ||||||
| 		if ev.Type == "m.room.redaction" { | 		if ev.Type == "m.room.redaction" { | ||||||
| 			rmsg.Event = config.EVENT_MSG_DELETE | 			rmsg.Event = config.EventMsgDelete | ||||||
| 			rmsg.ID = ev.Redacts | 			rmsg.ID = ev.Redacts | ||||||
| 			rmsg.Text = config.EVENT_MSG_DELETE | 			rmsg.Text = config.EventMsgDelete | ||||||
| 			b.Remote <- rmsg | 			b.Remote <- rmsg | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Do we have a /me action | 		// Do we have a /me action | ||||||
| 		if ev.Content["msgtype"].(string) == "m.emote" { | 		if ev.Content["msgtype"].(string) == "m.emote" { | ||||||
| 			rmsg.Event = config.EVENT_USER_ACTION | 			rmsg.Event = config.EventUserAction | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Do we have attachments | 		// Do we have attachments | ||||||
| @@ -231,11 +234,11 @@ func (b *Bmatrix) handleDownloadFile(rmsg *config.Message, content map[string]in | |||||||
| 		if msgtype == "m.image" { | 		if msgtype == "m.image" { | ||||||
| 			mext, _ := mime.ExtensionsByType(mtype) | 			mext, _ := mime.ExtensionsByType(mtype) | ||||||
| 			if len(mext) > 0 { | 			if len(mext) > 0 { | ||||||
| 				name = name + mext[0] | 				name += mext[0] | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			// just a default .png extension if we don't have mime info | 			// just a default .png extension if we don't have mime info | ||||||
| 			name = name + ".png" | 			name += ".png" | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import ( | |||||||
| 	"github.com/42wim/matterbridge/bridge/helper" | 	"github.com/42wim/matterbridge/bridge/helper" | ||||||
| 	"github.com/42wim/matterbridge/matterclient" | 	"github.com/42wim/matterbridge/matterclient" | ||||||
| 	"github.com/42wim/matterbridge/matterhook" | 	"github.com/42wim/matterbridge/matterhook" | ||||||
|  | 	"github.com/mattermost/platform/model" | ||||||
| 	"github.com/rs/xid" | 	"github.com/rs/xid" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -22,6 +23,8 @@ type Bmattermost struct { | |||||||
| 	avatarMap map[string]string | 	avatarMap map[string]string | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const mattermostPlugin = "mattermost.plugin" | ||||||
|  |  | ||||||
| func New(cfg *bridge.Config) bridge.Bridger { | func New(cfg *bridge.Config) bridge.Bridger { | ||||||
| 	b := &Bmattermost{Config: cfg, avatarMap: make(map[string]string)} | 	b := &Bmattermost{Config: cfg, avatarMap: make(map[string]string)} | ||||||
| 	b.uuid = xid.New().String() | 	b.uuid = xid.New().String() | ||||||
| @@ -33,25 +36,29 @@ func (b *Bmattermost) Command(cmd string) string { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmattermost) Connect() error { | func (b *Bmattermost) Connect() error { | ||||||
|  | 	if b.Account == mattermostPlugin { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
| 	if b.GetString("WebhookBindAddress") != "" { | 	if b.GetString("WebhookBindAddress") != "" { | ||||||
| 		if b.GetString("WebhookURL") != "" { | 		switch { | ||||||
|  | 		case b.GetString("WebhookURL") != "": | ||||||
| 			b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)") | 			b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)") | ||||||
| 			b.mh = matterhook.New(b.GetString("WebhookURL"), | 			b.mh = matterhook.New(b.GetString("WebhookURL"), | ||||||
| 				matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | 				matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | ||||||
| 					BindAddress: b.GetString("WebhookBindAddress")}) | 					BindAddress: b.GetString("WebhookBindAddress")}) | ||||||
| 		} else if b.GetString("Token") != "" { | 		case b.GetString("Token") != "": | ||||||
| 			b.Log.Info("Connecting using token (sending)") | 			b.Log.Info("Connecting using token (sending)") | ||||||
| 			err := b.apiLogin() | 			err := b.apiLogin() | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} else if b.GetString("Login") != "" { | 		case b.GetString("Login") != "": | ||||||
| 			b.Log.Info("Connecting using login/password (sending)") | 			b.Log.Info("Connecting using login/password (sending)") | ||||||
| 			err := b.apiLogin() | 			err := b.apiLogin() | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} else { | 		default: | ||||||
| 			b.Log.Info("Connecting using webhookbindaddress (receiving)") | 			b.Log.Info("Connecting using webhookbindaddress (receiving)") | ||||||
| 			b.mh = matterhook.New(b.GetString("WebhookURL"), | 			b.mh = matterhook.New(b.GetString("WebhookURL"), | ||||||
| 				matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | 				matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | ||||||
| @@ -60,7 +67,8 @@ func (b *Bmattermost) Connect() error { | |||||||
| 		go b.handleMatter() | 		go b.handleMatter() | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	if b.GetString("WebhookURL") != "" { | 	switch { | ||||||
|  | 	case b.GetString("WebhookURL") != "": | ||||||
| 		b.Log.Info("Connecting using webhookurl (sending)") | 		b.Log.Info("Connecting using webhookurl (sending)") | ||||||
| 		b.mh = matterhook.New(b.GetString("WebhookURL"), | 		b.mh = matterhook.New(b.GetString("WebhookURL"), | ||||||
| 			matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | 			matterhook.Config{InsecureSkipVerify: b.GetBool("SkipTLSVerify"), | ||||||
| @@ -81,14 +89,14 @@ func (b *Bmattermost) Connect() error { | |||||||
| 			go b.handleMatter() | 			go b.handleMatter() | ||||||
| 		} | 		} | ||||||
| 		return nil | 		return nil | ||||||
| 	} else if b.GetString("Token") != "" { | 	case b.GetString("Token") != "": | ||||||
| 		b.Log.Info("Connecting using token (sending and receiving)") | 		b.Log.Info("Connecting using token (sending and receiving)") | ||||||
| 		err := b.apiLogin() | 		err := b.apiLogin() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		go b.handleMatter() | 		go b.handleMatter() | ||||||
| 	} else if b.GetString("Login") != "" { | 	case b.GetString("Login") != "": | ||||||
| 		b.Log.Info("Connecting using login/password (sending and receiving)") | 		b.Log.Info("Connecting using login/password (sending and receiving)") | ||||||
| 		err := b.apiLogin() | 		err := b.apiLogin() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @@ -107,9 +115,12 @@ func (b *Bmattermost) Disconnect() error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmattermost) JoinChannel(channel config.ChannelInfo) error { | func (b *Bmattermost) JoinChannel(channel config.ChannelInfo) error { | ||||||
|  | 	if b.Account == mattermostPlugin { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
| 	// we can only join channels using the API | 	// we can only join channels using the API | ||||||
| 	if b.GetString("WebhookURL") == "" && b.GetString("WebhookBindAddress") == "" { | 	if b.GetString("WebhookURL") == "" && b.GetString("WebhookBindAddress") == "" { | ||||||
| 		id := b.mc.GetChannelId(channel.Name, "") | 		id := b.mc.GetChannelId(channel.Name, b.TeamID) | ||||||
| 		if id == "" { | 		if id == "" { | ||||||
| 			return fmt.Errorf("Could not find channel ID for channel %s", channel.Name) | 			return fmt.Errorf("Could not find channel ID for channel %s", channel.Name) | ||||||
| 		} | 		} | ||||||
| @@ -119,15 +130,18 @@ func (b *Bmattermost) JoinChannel(channel config.ChannelInfo) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmattermost) Send(msg config.Message) (string, error) { | func (b *Bmattermost) Send(msg config.Message) (string, error) { | ||||||
|  | 	if b.Account == mattermostPlugin { | ||||||
|  | 		return "", nil | ||||||
|  | 	} | ||||||
| 	b.Log.Debugf("=> Receiving %#v", msg) | 	b.Log.Debugf("=> Receiving %#v", msg) | ||||||
|  |  | ||||||
| 	// Make a action /me of the message | 	// Make a action /me of the message | ||||||
| 	if msg.Event == config.EVENT_USER_ACTION { | 	if msg.Event == config.EventUserAction { | ||||||
| 		msg.Text = "*" + msg.Text + "*" | 		msg.Text = "*" + msg.Text + "*" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// map the file SHA to our user (caches the avatar) | 	// map the file SHA to our user (caches the avatar) | ||||||
| 	if msg.Event == config.EVENT_AVATAR_DOWNLOAD { | 	if msg.Event == config.EventAvatarDownload { | ||||||
| 		return b.cacheAvatar(&msg) | 		return b.cacheAvatar(&msg) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -137,7 +151,7 @@ func (b *Bmattermost) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
| @@ -147,7 +161,7 @@ func (b *Bmattermost) Send(msg config.Message) (string, error) { | |||||||
| 	// Upload a file if it exists | 	// Upload a file if it exists | ||||||
| 	if msg.Extra != nil { | 	if msg.Extra != nil { | ||||||
| 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | ||||||
| 			b.mc.PostMessage(b.mc.GetChannelId(rmsg.Channel, ""), rmsg.Username+rmsg.Text) | 			b.mc.PostMessage(b.mc.GetChannelId(rmsg.Channel, b.TeamID), rmsg.Username+rmsg.Text) | ||||||
| 		} | 		} | ||||||
| 		if len(msg.Extra["file"]) > 0 { | 		if len(msg.Extra["file"]) > 0 { | ||||||
| 			return b.handleUploadFile(&msg) | 			return b.handleUploadFile(&msg) | ||||||
| @@ -165,7 +179,7 @@ func (b *Bmattermost) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Post normal message | 	// Post normal message | ||||||
| 	return b.mc.PostMessage(b.mc.GetChannelId(msg.Channel, ""), msg.Text) | 	return b.mc.PostMessage(b.mc.GetChannelId(msg.Channel, b.TeamID), msg.Text) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bmattermost) handleMatter() { | func (b *Bmattermost) handleMatter() { | ||||||
| @@ -187,7 +201,7 @@ func (b *Bmattermost) handleMatter() { | |||||||
| 		message.Account = b.Account | 		message.Account = b.Account | ||||||
| 		message.Text, ok = b.replaceAction(message.Text) | 		message.Text, ok = b.replaceAction(message.Text) | ||||||
| 		if ok { | 		if ok { | ||||||
| 			message.Event = config.EVENT_USER_ACTION | 			message.Event = config.EventUserAction | ||||||
| 		} | 		} | ||||||
| 		b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account) | 		b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account) | ||||||
| 		b.Log.Debugf("<= Message is %#v", message) | 		b.Log.Debugf("<= Message is %#v", message) | ||||||
| @@ -237,12 +251,12 @@ func (b *Bmattermost) handleMatterClient(messages chan *config.Message) { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// create a text for bridges that don't support native editing | 		// create a text for bridges that don't support native editing | ||||||
| 		if message.Raw.Event == "post_edited" && !b.GetBool("EditDisable") { | 		if message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED && !b.GetBool("EditDisable") { | ||||||
| 			rmsg.Text = message.Text + b.GetString("EditSuffix") | 			rmsg.Text = message.Text + b.GetString("EditSuffix") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if message.Raw.Event == "post_deleted" { | 		if message.Raw.Event == model.WEBSOCKET_EVENT_POST_DELETED { | ||||||
| 			rmsg.Event = config.EVENT_MSG_DELETE | 			rmsg.Event = config.EventMsgDelete | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if len(message.Post.FileIds) > 0 { | 		if len(message.Post.FileIds) > 0 { | ||||||
| @@ -273,7 +287,7 @@ func (b *Bmattermost) handleMatterHook(messages chan *config.Message) { | |||||||
| func (b *Bmattermost) apiLogin() error { | func (b *Bmattermost) apiLogin() error { | ||||||
| 	password := b.GetString("Password") | 	password := b.GetString("Password") | ||||||
| 	if b.GetString("Token") != "" { | 	if b.GetString("Token") != "" { | ||||||
| 		password = "MMAUTHTOKEN=" + b.GetString("Token") | 		password = "token=" + b.GetString("Token") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	b.mc = matterclient.New(b.GetString("Login"), password, b.GetString("Team"), b.GetString("Server")) | 	b.mc = matterclient.New(b.GetString("Login"), password, b.GetString("Team"), b.GetString("Server")) | ||||||
| @@ -317,7 +331,7 @@ func (b *Bmattermost) cacheAvatar(msg *config.Message) (string, error) { | |||||||
| // sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful. | // sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful. | ||||||
| // logs an error message if it fails | // logs an error message if it fails | ||||||
| func (b *Bmattermost) handleDownloadAvatar(userid string, channel string) { | func (b *Bmattermost) handleDownloadAvatar(userid string, channel string) { | ||||||
| 	rmsg := config.Message{Username: "system", Text: "avatar", Channel: channel, Account: b.Account, UserID: userid, Event: config.EVENT_AVATAR_DOWNLOAD, Extra: make(map[string][]interface{})} | 	rmsg := config.Message{Username: "system", Text: "avatar", Channel: channel, Account: b.Account, UserID: userid, Event: config.EventAvatarDownload, Extra: make(map[string][]interface{})} | ||||||
| 	if _, ok := b.avatarMap[userid]; !ok { | 	if _, ok := b.avatarMap[userid]; !ok { | ||||||
| 		data, resp := b.mc.Client.GetProfileImage(userid, "") | 		data, resp := b.mc.Client.GetProfileImage(userid, "") | ||||||
| 		if resp.Error != nil { | 		if resp.Error != nil { | ||||||
| @@ -357,7 +371,7 @@ func (b *Bmattermost) handleDownloadFile(rmsg *config.Message, id string) error | |||||||
| func (b *Bmattermost) handleUploadFile(msg *config.Message) (string, error) { | func (b *Bmattermost) handleUploadFile(msg *config.Message) (string, error) { | ||||||
| 	var err error | 	var err error | ||||||
| 	var res, id string | 	var res, id string | ||||||
| 	channelID := b.mc.GetChannelId(msg.Channel, "") | 	channelID := b.mc.GetChannelId(msg.Channel, b.TeamID) | ||||||
| 	for _, f := range msg.Extra["file"] { | 	for _, f := range msg.Extra["file"] { | ||||||
| 		fi := f.(config.FileInfo) | 		fi := f.(config.FileInfo) | ||||||
| 		id, err = b.mc.UploadFile(*fi.Data, channelID, fi.Name) | 		id, err = b.mc.UploadFile(*fi.Data, channelID, fi.Name) | ||||||
| @@ -386,6 +400,7 @@ func (b *Bmattermost) sendWebhook(msg config.Message) (string, error) { | |||||||
| 	if msg.Extra != nil { | 	if msg.Extra != nil { | ||||||
| 		// this sends a message only if we received a config.EVENT_FILE_FAILURE_SIZE | 		// this sends a message only if we received a config.EVENT_FILE_FAILURE_SIZE | ||||||
| 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | ||||||
|  | 			rmsg := rmsg // scopelint | ||||||
| 			iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl")) | 			iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl")) | ||||||
| 			matterMessage := matterhook.OMessage{IconURL: iconURL, Channel: rmsg.Channel, UserName: rmsg.Username, Text: rmsg.Text, Props: make(map[string]interface{})} | 			matterMessage := matterhook.OMessage{IconURL: iconURL, Channel: rmsg.Channel, UserName: rmsg.Username, Text: rmsg.Text, Props: make(map[string]interface{})} | ||||||
| 			matterMessage.Props["matterbridge_"+b.uuid] = true | 			matterMessage.Props["matterbridge_"+b.uuid] = true | ||||||
| @@ -427,12 +442,12 @@ func (b *Bmattermost) skipMessage(message *matterclient.Message) bool { | |||||||
| 			return true | 			return true | ||||||
| 		} | 		} | ||||||
| 		b.Log.Debugf("Sending JOIN_LEAVE event from %s to gateway", b.Account) | 		b.Log.Debugf("Sending JOIN_LEAVE event from %s to gateway", b.Account) | ||||||
| 		b.Remote <- config.Message{Username: "system", Text: message.Text, Channel: message.Channel, Account: b.Account, Event: config.EVENT_JOIN_LEAVE} | 		b.Remote <- config.Message{Username: "system", Text: message.Text, Channel: message.Channel, Account: b.Account, Event: config.EventJoinLeave} | ||||||
| 		return true | 		return true | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Handle edited messages | 	// Handle edited messages | ||||||
| 	if (message.Raw.Event == "post_edited") && b.GetBool("EditDisable") { | 	if (message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED) && b.GetBool("EditDisable") { | ||||||
| 		return true | 		return true | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -460,7 +475,7 @@ func (b *Bmattermost) skipMessage(message *matterclient.Message) bool { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// only handle posted, edited or deleted events | 	// only handle posted, edited or deleted events | ||||||
| 	if !(message.Raw.Event == "posted" || message.Raw.Event == "post_edited" || message.Raw.Event == "post_deleted") { | 	if !(message.Raw.Event == "posted" || message.Raw.Event == model.WEBSOCKET_EVENT_POST_EDITED || message.Raw.Event == model.WEBSOCKET_EVENT_POST_DELETED) { | ||||||
| 		return true | 		return true | ||||||
| 	} | 	} | ||||||
| 	return false | 	return false | ||||||
|   | |||||||
| @@ -8,13 +8,9 @@ import ( | |||||||
| 	"github.com/42wim/matterbridge/matterhook" | 	"github.com/42wim/matterbridge/matterhook" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type MMhook struct { | type Brocketchat struct { | ||||||
| 	mh *matterhook.Client | 	mh *matterhook.Client | ||||||
| 	rh *rockethook.Client | 	rh *rockethook.Client | ||||||
| } |  | ||||||
|  |  | ||||||
| type Brocketchat struct { |  | ||||||
| 	MMhook |  | ||||||
| 	*bridge.Config | 	*bridge.Config | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -47,12 +43,13 @@ func (b *Brocketchat) JoinChannel(channel config.ChannelInfo) error { | |||||||
|  |  | ||||||
| func (b *Brocketchat) Send(msg config.Message) (string, error) { | func (b *Brocketchat) Send(msg config.Message) (string, error) { | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
| 	b.Log.Debugf("=> Receiving %#v", msg) | 	b.Log.Debugf("=> Receiving %#v", msg) | ||||||
| 	if msg.Extra != nil { | 	if msg.Extra != nil { | ||||||
| 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | 		for _, rmsg := range helper.HandleExtra(&msg, b.General) { | ||||||
|  | 			rmsg := rmsg // scopelint | ||||||
| 			iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl")) | 			iconURL := config.GetIconURL(&rmsg, b.GetString("iconurl")) | ||||||
| 			matterMessage := matterhook.OMessage{IconURL: iconURL, Channel: rmsg.Channel, UserName: rmsg.Username, Text: rmsg.Text} | 			matterMessage := matterhook.OMessage{IconURL: iconURL, Channel: rmsg.Channel, UserName: rmsg.Username, Text: rmsg.Text} | ||||||
| 			b.mh.Send(matterMessage) | 			b.mh.Send(matterMessage) | ||||||
|   | |||||||
							
								
								
									
										311
									
								
								bridge/slack/handlers.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,311 @@ | |||||||
|  | package bslack | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"html" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
|  | 	"github.com/42wim/matterbridge/bridge/helper" | ||||||
|  | 	"github.com/nlopes/slack" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleSlack() { | ||||||
|  | 	messages := make(chan *config.Message) | ||||||
|  | 	if b.GetString(incomingWebhookConfig) != "" { | ||||||
|  | 		b.Log.Debugf("Choosing webhooks based receiving") | ||||||
|  | 		go b.handleMatterHook(messages) | ||||||
|  | 	} else { | ||||||
|  | 		b.Log.Debugf("Choosing token based receiving") | ||||||
|  | 		go b.handleSlackClient(messages) | ||||||
|  | 	} | ||||||
|  | 	time.Sleep(time.Second) | ||||||
|  | 	b.Log.Debug("Start listening for Slack messages") | ||||||
|  | 	for message := range messages { | ||||||
|  | 		if message.Event != config.EventUserTyping { | ||||||
|  | 			b.Log.Debugf("<= Sending message from %s on %s to gateway", message.Username, b.Account) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// cleanup the message | ||||||
|  | 		message.Text = b.replaceMention(message.Text) | ||||||
|  | 		message.Text = b.replaceVariable(message.Text) | ||||||
|  | 		message.Text = b.replaceChannel(message.Text) | ||||||
|  | 		message.Text = b.replaceURL(message.Text) | ||||||
|  | 		message.Text = html.UnescapeString(message.Text) | ||||||
|  |  | ||||||
|  | 		// Add the avatar | ||||||
|  | 		message.Avatar = b.getAvatar(message.UserID) | ||||||
|  |  | ||||||
|  | 		b.Log.Debugf("<= Message is %#v", message) | ||||||
|  | 		b.Remote <- *message | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleSlackClient(messages chan *config.Message) { | ||||||
|  | 	for msg := range b.rtm.IncomingEvents { | ||||||
|  | 		if msg.Type != sUserTyping && msg.Type != sLatencyReport { | ||||||
|  | 			b.Log.Debugf("== Receiving event %#v", msg.Data) | ||||||
|  | 		} | ||||||
|  | 		switch ev := msg.Data.(type) { | ||||||
|  | 		case *slack.UserTypingEvent: | ||||||
|  | 			if !b.GetBool("ShowUserTyping") { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			rmsg, err := b.handleTypingEvent(ev) | ||||||
|  | 			if err != nil { | ||||||
|  | 				b.Log.Errorf("%#v", err) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			messages <- rmsg | ||||||
|  | 		case *slack.MessageEvent: | ||||||
|  | 			if b.skipMessageEvent(ev) { | ||||||
|  | 				b.Log.Debugf("Skipped message: %#v", ev) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			rmsg, err := b.handleMessageEvent(ev) | ||||||
|  | 			if err != nil { | ||||||
|  | 				b.Log.Errorf("%#v", err) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			messages <- rmsg | ||||||
|  | 		case *slack.OutgoingErrorEvent: | ||||||
|  | 			b.Log.Debugf("%#v", ev.Error()) | ||||||
|  | 		case *slack.ChannelJoinedEvent: | ||||||
|  | 			// When we join a channel we update the full list of users as | ||||||
|  | 			// well as the information for the channel that we joined as this | ||||||
|  | 			// should now tell that we are a member of it. | ||||||
|  | 			b.populateUsers() | ||||||
|  |  | ||||||
|  | 			b.channelsMutex.Lock() | ||||||
|  | 			b.channelsByID[ev.Channel.ID] = &ev.Channel | ||||||
|  | 			b.channelsByName[ev.Channel.Name] = &ev.Channel | ||||||
|  | 			b.channelsMutex.Unlock() | ||||||
|  | 		case *slack.ConnectedEvent: | ||||||
|  | 			b.si = ev.Info | ||||||
|  | 			b.populateChannels() | ||||||
|  | 			b.populateUsers() | ||||||
|  | 		case *slack.InvalidAuthEvent: | ||||||
|  | 			b.Log.Fatalf("Invalid Token %#v", ev) | ||||||
|  | 		case *slack.ConnectionErrorEvent: | ||||||
|  | 			b.Log.Errorf("Connection failed %#v %#v", ev.Error(), ev.ErrorObj) | ||||||
|  | 		default: | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleMatterHook(messages chan *config.Message) { | ||||||
|  | 	for { | ||||||
|  | 		message := b.mh.Receive() | ||||||
|  | 		b.Log.Debugf("receiving from matterhook (slack) %#v", message) | ||||||
|  | 		if message.UserName == "slackbot" { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		messages <- &config.Message{ | ||||||
|  | 			Username: message.UserName, | ||||||
|  | 			Text:     message.Text, | ||||||
|  | 			Channel:  message.ChannelName, | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // skipMessageEvent skips event that need to be skipped :-) | ||||||
|  | func (b *Bslack) skipMessageEvent(ev *slack.MessageEvent) bool { | ||||||
|  | 	switch ev.SubType { | ||||||
|  | 	case sChannelLeave, sChannelJoin: | ||||||
|  | 		return b.GetBool(noSendJoinConfig) | ||||||
|  | 	case sPinnedItem, sUnpinnedItem: | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Skip any messages that we made ourselves or from 'slackbot' (see #527). | ||||||
|  | 	if ev.Username == sSlackBotUser || | ||||||
|  | 		(b.rtm != nil && ev.Username == b.si.User.Name) || | ||||||
|  | 		(len(ev.Attachments) > 0 && ev.Attachments[0].CallbackID == "matterbridge_"+b.uuid) { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// It seems ev.SubMessage.Edited == nil when slack unfurls. | ||||||
|  | 	// Do not forward these messages. See Github issue #266. | ||||||
|  | 	if ev.SubMessage != nil && | ||||||
|  | 		ev.SubMessage.ThreadTimestamp != ev.SubMessage.Timestamp && | ||||||
|  | 		ev.SubMessage.Edited == nil { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(ev.Files) > 0 { | ||||||
|  | 		return b.filesCached(ev.Files) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) filesCached(files []slack.File) bool { | ||||||
|  | 	for i := range files { | ||||||
|  | 		if !b.fileCached(&files[i]) { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // handleMessageEvent handles the message events. Together with any called sub-methods, | ||||||
|  | // this method implements the following event processing pipeline: | ||||||
|  | // | ||||||
|  | // 1. Check if the message should be ignored. | ||||||
|  | //    NOTE: This is not actually part of the method below but is done just before it | ||||||
|  | //          is called via the 'skipMessageEvent()' method. | ||||||
|  | // 2. Populate the Matterbridge message that will be sent to the router based on the | ||||||
|  | //    received event and logic that is common to all events that are not skipped. | ||||||
|  | // 3. Detect and handle any message that is "status" related (think join channel, etc.). | ||||||
|  | //    This might result in an early exit from the pipeline and passing of the | ||||||
|  | //    pre-populated message to the Matterbridge router. | ||||||
|  | // 4. Handle the specific case of messages that edit existing messages depending on | ||||||
|  | //    configuration. | ||||||
|  | // 5. Handle any attachments of the received event. | ||||||
|  | // 6. Check that the Matterbridge message that we end up with after at the end of the | ||||||
|  | //    pipeline is valid before sending it to the Matterbridge router. | ||||||
|  | func (b *Bslack) handleMessageEvent(ev *slack.MessageEvent) (*config.Message, error) { | ||||||
|  | 	rmsg, err := b.populateReceivedMessage(ev) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Handle some message types early. | ||||||
|  | 	if b.handleStatusEvent(ev, rmsg) { | ||||||
|  | 		return rmsg, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	b.handleAttachments(ev, rmsg) | ||||||
|  |  | ||||||
|  | 	// Verify that we have the right information and the message | ||||||
|  | 	// is well-formed before sending it out to the router. | ||||||
|  | 	if len(ev.Files) == 0 && (rmsg.Text == "" || rmsg.Username == "") { | ||||||
|  | 		if ev.BotID != "" { | ||||||
|  | 			// This is probably a webhook we couldn't resolve. | ||||||
|  | 			return nil, fmt.Errorf("message handling resulted in an empty bot message (probably an incoming webhook we couldn't resolve): %#v", ev) | ||||||
|  | 		} | ||||||
|  | 		return nil, fmt.Errorf("message handling resulted in an empty message: %#v", ev) | ||||||
|  | 	} | ||||||
|  | 	return rmsg, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleStatusEvent(ev *slack.MessageEvent, rmsg *config.Message) bool { | ||||||
|  | 	switch ev.SubType { | ||||||
|  | 	case sChannelJoined, sMemberJoined: | ||||||
|  | 		b.populateUsers() | ||||||
|  | 		// There's no further processing needed on channel events | ||||||
|  | 		// so we return 'true'. | ||||||
|  | 		return true | ||||||
|  | 	case sChannelJoin, sChannelLeave: | ||||||
|  | 		rmsg.Username = sSystemUser | ||||||
|  | 		rmsg.Event = config.EventJoinLeave | ||||||
|  | 	case sChannelTopic, sChannelPurpose: | ||||||
|  | 		rmsg.Event = config.EventTopicChange | ||||||
|  | 	case sMessageChanged: | ||||||
|  | 		rmsg.Text = ev.SubMessage.Text | ||||||
|  | 		// handle deleted thread starting messages | ||||||
|  | 		if ev.SubMessage.Text == "This message was deleted." { | ||||||
|  | 			rmsg.Event = config.EventMsgDelete | ||||||
|  | 			return true | ||||||
|  | 		} | ||||||
|  | 	case sMessageDeleted: | ||||||
|  | 		rmsg.Text = config.EventMsgDelete | ||||||
|  | 		rmsg.Event = config.EventMsgDelete | ||||||
|  | 		rmsg.ID = ev.DeletedTimestamp | ||||||
|  | 		// If a message is being deleted we do not need to process | ||||||
|  | 		// the event any further so we return 'true'. | ||||||
|  | 		return true | ||||||
|  | 	case sMeMessage: | ||||||
|  | 		rmsg.Event = config.EventUserAction | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleAttachments(ev *slack.MessageEvent, rmsg *config.Message) { | ||||||
|  | 	// File comments are set by the system (because there is no username given). | ||||||
|  | 	if ev.SubType == sFileComment { | ||||||
|  | 		rmsg.Username = sSystemUser | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// See if we have some text in the attachments. | ||||||
|  | 	if rmsg.Text == "" { | ||||||
|  | 		for _, attach := range ev.Attachments { | ||||||
|  | 			if attach.Text != "" { | ||||||
|  | 				if attach.Title != "" { | ||||||
|  | 					rmsg.Text = attach.Title + "\n" | ||||||
|  | 				} | ||||||
|  | 				rmsg.Text += attach.Text | ||||||
|  | 			} else { | ||||||
|  | 				rmsg.Text = attach.Fallback | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Save the attachments, so that we can send them to other slack (compatible) bridges. | ||||||
|  | 	if len(ev.Attachments) > 0 { | ||||||
|  | 		rmsg.Extra[sSlackAttachment] = append(rmsg.Extra[sSlackAttachment], ev.Attachments) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If we have files attached, download them (in memory) and put a pointer to it in msg.Extra. | ||||||
|  | 	for i := range ev.Files { | ||||||
|  | 		if err := b.handleDownloadFile(rmsg, &ev.Files[i]); err != nil { | ||||||
|  | 			b.Log.Errorf("Could not download incoming file: %#v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleTypingEvent(ev *slack.UserTypingEvent) (*config.Message, error) { | ||||||
|  | 	channelInfo, err := b.getChannelByID(ev.Channel) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &config.Message{ | ||||||
|  | 		Channel: channelInfo.Name, | ||||||
|  | 		Account: b.Account, | ||||||
|  | 		Event:   config.EventUserTyping, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // handleDownloadFile handles file download | ||||||
|  | func (b *Bslack) handleDownloadFile(rmsg *config.Message, file *slack.File) error { | ||||||
|  | 	if b.fileCached(file) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	// Check that the file is neither too large nor blacklisted. | ||||||
|  | 	if err := helper.HandleDownloadSize(b.Log, rmsg, file.Name, int64(file.Size), b.General); err != nil { | ||||||
|  | 		b.Log.WithError(err).Infof("Skipping download of incoming file.") | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Actually download the file. | ||||||
|  | 	data, err := helper.DownloadFileAuth(file.URLPrivateDownload, "Bearer "+b.GetString(tokenConfig)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("download %s failed %#v", file.URLPrivateDownload, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If a comment is attached to the file(s) it is in the 'Text' field of the Slack messge event | ||||||
|  | 	// and should be added as comment to only one of the files. We reset the 'Text' field to ensure | ||||||
|  | 	// that the comment is not duplicated. | ||||||
|  | 	comment := rmsg.Text | ||||||
|  | 	rmsg.Text = "" | ||||||
|  | 	helper.HandleDownloadData(b.Log, rmsg, file.Name, comment, file.URLPrivateDownload, data, b.General) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // fileCached implements Matterbridge's caching logic for files | ||||||
|  | // shared via Slack. | ||||||
|  | // | ||||||
|  | // We consider that a file was cached if its ID was added in the last minute or | ||||||
|  | // it's name was registered in the last 10 seconds. This ensures that an | ||||||
|  | // identically named file but with different content will be uploaded correctly | ||||||
|  | // (the assumption is that such name collisions will not occur within the given | ||||||
|  | // timeframes). | ||||||
|  | func (b *Bslack) fileCached(file *slack.File) bool { | ||||||
|  | 	if ts, ok := b.cache.Get("file" + file.ID); ok && time.Since(ts.(time.Time)) < time.Minute { | ||||||
|  | 		return true | ||||||
|  | 	} else if ts, ok = b.cache.Get("filename" + file.Name); ok && time.Since(ts.(time.Time)) < 10*time.Second { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
							
								
								
									
										323
									
								
								bridge/slack/helpers.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,323 @@ | |||||||
|  | package bslack | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"regexp" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
|  | 	"github.com/nlopes/slack" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (b *Bslack) getUser(id string) *slack.User { | ||||||
|  | 	b.usersMutex.RLock() | ||||||
|  | 	defer b.usersMutex.RUnlock() | ||||||
|  |  | ||||||
|  | 	return b.users[id] | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) getUsername(id string) string { | ||||||
|  | 	if user := b.getUser(id); user != nil { | ||||||
|  | 		if user.Profile.DisplayName != "" { | ||||||
|  | 			return user.Profile.DisplayName | ||||||
|  | 		} | ||||||
|  | 		return user.Name | ||||||
|  | 	} | ||||||
|  | 	b.Log.Warnf("Could not find user with ID '%s'", id) | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) getAvatar(id string) string { | ||||||
|  | 	if user := b.getUser(id); user != nil { | ||||||
|  | 		return user.Profile.Image48 | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) getChannel(channel string) (*slack.Channel, error) { | ||||||
|  | 	if strings.HasPrefix(channel, "ID:") { | ||||||
|  | 		return b.getChannelByID(strings.TrimPrefix(channel, "ID:")) | ||||||
|  | 	} | ||||||
|  | 	return b.getChannelByName(channel) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) getChannelByName(name string) (*slack.Channel, error) { | ||||||
|  | 	b.channelsMutex.RLock() | ||||||
|  | 	defer b.channelsMutex.RUnlock() | ||||||
|  |  | ||||||
|  | 	if channel, ok := b.channelsByName[name]; ok { | ||||||
|  | 		return channel, nil | ||||||
|  | 	} | ||||||
|  | 	return nil, fmt.Errorf("%s: channel %s not found", b.Account, name) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) getChannelByID(ID string) (*slack.Channel, error) { | ||||||
|  | 	b.channelsMutex.RLock() | ||||||
|  | 	defer b.channelsMutex.RUnlock() | ||||||
|  |  | ||||||
|  | 	if channel, ok := b.channelsByID[ID]; ok { | ||||||
|  | 		return channel, nil | ||||||
|  | 	} | ||||||
|  | 	return nil, fmt.Errorf("%s: channel %s not found", b.Account, ID) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const minimumRefreshInterval = 10 * time.Second | ||||||
|  |  | ||||||
|  | func (b *Bslack) populateUsers() { | ||||||
|  | 	b.refreshMutex.Lock() | ||||||
|  | 	if time.Now().Before(b.earliestUserRefresh) || b.refreshInProgress { | ||||||
|  | 		b.Log.Debugf("Not refreshing user list as it was done less than %v ago.", | ||||||
|  | 			minimumRefreshInterval) | ||||||
|  | 		b.refreshMutex.Unlock() | ||||||
|  |  | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	b.refreshInProgress = true | ||||||
|  | 	b.refreshMutex.Unlock() | ||||||
|  |  | ||||||
|  | 	newUsers := map[string]*slack.User{} | ||||||
|  | 	pagination := b.sc.GetUsersPaginated(slack.GetUsersOptionLimit(200)) | ||||||
|  | 	for { | ||||||
|  | 		var err error | ||||||
|  | 		pagination, err = pagination.Next(context.Background()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			if pagination.Done(err) { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if err = b.handleRateLimit(err); err != nil { | ||||||
|  | 				b.Log.Errorf("Could not retrieve users: %#v", err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for i := range pagination.Users { | ||||||
|  | 			newUsers[pagination.Users[i].ID] = &pagination.Users[i] | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	b.usersMutex.Lock() | ||||||
|  | 	defer b.usersMutex.Unlock() | ||||||
|  | 	b.users = newUsers | ||||||
|  |  | ||||||
|  | 	b.refreshMutex.Lock() | ||||||
|  | 	defer b.refreshMutex.Unlock() | ||||||
|  | 	b.earliestUserRefresh = time.Now().Add(minimumRefreshInterval) | ||||||
|  | 	b.refreshInProgress = false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) populateChannels() { | ||||||
|  | 	b.refreshMutex.Lock() | ||||||
|  | 	if time.Now().Before(b.earliestChannelRefresh) || b.refreshInProgress { | ||||||
|  | 		b.Log.Debugf("Not refreshing channel list as it was done less than %v seconds ago.", | ||||||
|  | 			minimumRefreshInterval) | ||||||
|  | 		b.refreshMutex.Unlock() | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	b.refreshInProgress = true | ||||||
|  | 	b.refreshMutex.Unlock() | ||||||
|  |  | ||||||
|  | 	newChannelsByID := map[string]*slack.Channel{} | ||||||
|  | 	newChannelsByName := map[string]*slack.Channel{} | ||||||
|  |  | ||||||
|  | 	// We only retrieve public and private channels, not IMs | ||||||
|  | 	// and MPIMs as those do not have a channel name. | ||||||
|  | 	queryParams := &slack.GetConversationsParameters{ | ||||||
|  | 		ExcludeArchived: "true", | ||||||
|  | 		Types:           []string{"public_channel,private_channel"}, | ||||||
|  | 	} | ||||||
|  | 	for { | ||||||
|  | 		channels, nextCursor, err := b.sc.GetConversations(queryParams) | ||||||
|  | 		if err != nil { | ||||||
|  | 			if err = b.handleRateLimit(err); err != nil { | ||||||
|  | 				b.Log.Errorf("Could not retrieve channels: %#v", err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for i := range channels { | ||||||
|  | 			newChannelsByID[channels[i].ID] = &channels[i] | ||||||
|  | 			newChannelsByName[channels[i].Name] = &channels[i] | ||||||
|  | 		} | ||||||
|  | 		if nextCursor == "" { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		queryParams.Cursor = nextCursor | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	b.channelsMutex.Lock() | ||||||
|  | 	defer b.channelsMutex.Unlock() | ||||||
|  | 	b.channelsByID = newChannelsByID | ||||||
|  | 	b.channelsByName = newChannelsByName | ||||||
|  |  | ||||||
|  | 	b.refreshMutex.Lock() | ||||||
|  | 	defer b.refreshMutex.Unlock() | ||||||
|  | 	b.earliestChannelRefresh = time.Now().Add(minimumRefreshInterval) | ||||||
|  | 	b.refreshInProgress = false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // populateReceivedMessage shapes the initial Matterbridge message that we will forward to the | ||||||
|  | // router before we apply message-dependent modifications. | ||||||
|  | func (b *Bslack) populateReceivedMessage(ev *slack.MessageEvent) (*config.Message, error) { | ||||||
|  | 	// Use our own func because rtm.GetChannelInfo doesn't work for private channels. | ||||||
|  | 	channel, err := b.getChannelByID(ev.Channel) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	rmsg := &config.Message{ | ||||||
|  | 		Text:     ev.Text, | ||||||
|  | 		Channel:  channel.Name, | ||||||
|  | 		Account:  b.Account, | ||||||
|  | 		ID:       ev.Timestamp, | ||||||
|  | 		Extra:    make(map[string][]interface{}), | ||||||
|  | 		ParentID: ev.ThreadTimestamp, | ||||||
|  | 		Protocol: b.Protocol, | ||||||
|  | 	} | ||||||
|  | 	if b.useChannelID { | ||||||
|  | 		rmsg.Channel = "ID:" + channel.ID | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Handle 'edit' messages. | ||||||
|  | 	if ev.SubMessage != nil && !b.GetBool(editDisableConfig) { | ||||||
|  | 		rmsg.ID = ev.SubMessage.Timestamp | ||||||
|  | 		if ev.SubMessage.ThreadTimestamp != ev.SubMessage.Timestamp { | ||||||
|  | 			b.Log.Debugf("SubMessage %#v", ev.SubMessage) | ||||||
|  | 			rmsg.Text = ev.SubMessage.Text + b.GetString(editSuffixConfig) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if err = b.populateMessageWithUserInfo(ev, rmsg); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return rmsg, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) populateMessageWithUserInfo(ev *slack.MessageEvent, rmsg *config.Message) error { | ||||||
|  | 	if ev.SubType == sMessageDeleted || ev.SubType == sFileComment { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// First, deal with bot-originating messages but only do so when not using webhooks: we | ||||||
|  | 	// would not be able to distinguish which bot would be sending them. | ||||||
|  | 	if err := b.populateMessageWithBotInfo(ev, rmsg); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Second, deal with "real" users if we have the necessary information. | ||||||
|  | 	var userID string | ||||||
|  | 	switch { | ||||||
|  | 	case ev.User != "": | ||||||
|  | 		userID = ev.User | ||||||
|  | 	case ev.SubMessage != nil && ev.SubMessage.User != "": | ||||||
|  | 		userID = ev.SubMessage.User | ||||||
|  | 	default: | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	user := b.getUser(userID) | ||||||
|  | 	if user == nil { | ||||||
|  | 		return fmt.Errorf("could not find information for user with id %s", ev.User) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	rmsg.UserID = user.ID | ||||||
|  | 	rmsg.Username = user.Name | ||||||
|  | 	if user.Profile.DisplayName != "" { | ||||||
|  | 		rmsg.Username = user.Profile.DisplayName | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) populateMessageWithBotInfo(ev *slack.MessageEvent, rmsg *config.Message) error { | ||||||
|  | 	if ev.BotID == "" || b.GetString(outgoingWebhookConfig) != "" { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var err error | ||||||
|  | 	var bot *slack.Bot | ||||||
|  | 	for { | ||||||
|  | 		bot, err = b.rtm.GetBotInfo(ev.BotID) | ||||||
|  | 		if err == nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if err = b.handleRateLimit(err); err != nil { | ||||||
|  | 			b.Log.Errorf("Could not retrieve bot information: %#v", err) | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if bot.Name != "" && bot.Name != "Slack API Tester" { | ||||||
|  | 		rmsg.Username = bot.Name | ||||||
|  | 		if ev.Username != "" { | ||||||
|  | 			rmsg.Username = ev.Username | ||||||
|  | 		} | ||||||
|  | 		rmsg.UserID = bot.ID | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var ( | ||||||
|  | 	mentionRE  = regexp.MustCompile(`<@([a-zA-Z0-9]+)>`) | ||||||
|  | 	channelRE  = regexp.MustCompile(`<#[a-zA-Z0-9]+\|(.+?)>`) | ||||||
|  | 	variableRE = regexp.MustCompile(`<!((?:subteam\^)?[a-zA-Z0-9]+)(?:\|@?(.+?))?>`) | ||||||
|  | 	urlRE      = regexp.MustCompile(`<(.*?)(\|.*?)?>`) | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // @see https://api.slack.com/docs/message-formatting#linking_to_channels_and_users | ||||||
|  | func (b *Bslack) replaceMention(text string) string { | ||||||
|  | 	replaceFunc := func(match string) string { | ||||||
|  | 		userID := strings.Trim(match, "@<>") | ||||||
|  | 		if username := b.getUsername(userID); userID != "" { | ||||||
|  | 			return "@" + username | ||||||
|  | 		} | ||||||
|  | 		return match | ||||||
|  | 	} | ||||||
|  | 	return mentionRE.ReplaceAllStringFunc(text, replaceFunc) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // @see https://api.slack.com/docs/message-formatting#linking_to_channels_and_users | ||||||
|  | func (b *Bslack) replaceChannel(text string) string { | ||||||
|  | 	for _, r := range channelRE.FindAllStringSubmatch(text, -1) { | ||||||
|  | 		text = strings.Replace(text, r[0], "#"+r[1], 1) | ||||||
|  | 	} | ||||||
|  | 	return text | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // @see https://api.slack.com/docs/message-formatting#variables | ||||||
|  | func (b *Bslack) replaceVariable(text string) string { | ||||||
|  | 	for _, r := range variableRE.FindAllStringSubmatch(text, -1) { | ||||||
|  | 		if r[2] != "" { | ||||||
|  | 			text = strings.Replace(text, r[0], "@"+r[2], 1) | ||||||
|  | 		} else { | ||||||
|  | 			text = strings.Replace(text, r[0], "@"+r[1], 1) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return text | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // @see https://api.slack.com/docs/message-formatting#linking_to_urls | ||||||
|  | func (b *Bslack) replaceURL(text string) string { | ||||||
|  | 	for _, r := range urlRE.FindAllStringSubmatch(text, -1) { | ||||||
|  | 		if len(strings.TrimSpace(r[2])) == 1 { // A display text separator was found, but the text was blank | ||||||
|  | 			text = strings.Replace(text, r[0], "", 1) | ||||||
|  | 		} else { | ||||||
|  | 			text = strings.Replace(text, r[0], r[1], 1) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return text | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bslack) handleRateLimit(err error) error { | ||||||
|  | 	rateLimit, ok := err.(*slack.RateLimitedError) | ||||||
|  | 	if !ok { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	b.Log.Infof("Rate-limited by Slack. Sleeping for %v", rateLimit.RetryAfter) | ||||||
|  | 	time.Sleep(rateLimit.RetryAfter) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										74
									
								
								bridge/slack/legacy.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,74 @@ | |||||||
|  | package bslack | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  |  | ||||||
|  | 	"github.com/42wim/matterbridge/bridge" | ||||||
|  | 	"github.com/42wim/matterbridge/matterhook" | ||||||
|  | 	"github.com/nlopes/slack" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type BLegacy struct { | ||||||
|  | 	*Bslack | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewLegacy(cfg *bridge.Config) bridge.Bridger { | ||||||
|  | 	return &BLegacy{Bslack: newBridge(cfg)} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *BLegacy) Connect() error { | ||||||
|  | 	b.RLock() | ||||||
|  | 	defer b.RUnlock() | ||||||
|  | 	if b.GetString(incomingWebhookConfig) != "" { | ||||||
|  | 		switch { | ||||||
|  | 		case b.GetString(outgoingWebhookConfig) != "": | ||||||
|  | 			b.Log.Info("Connecting using webhookurl (sending) and webhookbindaddress (receiving)") | ||||||
|  | 			b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{ | ||||||
|  | 				InsecureSkipVerify: b.GetBool(skipTLSConfig), | ||||||
|  | 				BindAddress:        b.GetString(incomingWebhookConfig), | ||||||
|  | 			}) | ||||||
|  | 		case b.GetString(tokenConfig) != "": | ||||||
|  | 			b.Log.Info("Connecting using token (sending)") | ||||||
|  | 			b.sc = slack.New(b.GetString(tokenConfig)) | ||||||
|  | 			b.rtm = b.sc.NewRTM() | ||||||
|  | 			go b.rtm.ManageConnection() | ||||||
|  | 			b.Log.Info("Connecting using webhookbindaddress (receiving)") | ||||||
|  | 			b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{ | ||||||
|  | 				InsecureSkipVerify: b.GetBool(skipTLSConfig), | ||||||
|  | 				BindAddress:        b.GetString(incomingWebhookConfig), | ||||||
|  | 			}) | ||||||
|  | 		default: | ||||||
|  | 			b.Log.Info("Connecting using webhookbindaddress (receiving)") | ||||||
|  | 			b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{ | ||||||
|  | 				InsecureSkipVerify: b.GetBool(skipTLSConfig), | ||||||
|  | 				BindAddress:        b.GetString(incomingWebhookConfig), | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 		go b.handleSlack() | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if b.GetString(outgoingWebhookConfig) != "" { | ||||||
|  | 		b.Log.Info("Connecting using webhookurl (sending)") | ||||||
|  | 		b.mh = matterhook.New(b.GetString(outgoingWebhookConfig), matterhook.Config{ | ||||||
|  | 			InsecureSkipVerify: b.GetBool(skipTLSConfig), | ||||||
|  | 			DisableServer:      true, | ||||||
|  | 		}) | ||||||
|  | 		if b.GetString(tokenConfig) != "" { | ||||||
|  | 			b.Log.Info("Connecting using token (receiving)") | ||||||
|  | 			b.sc = slack.New(b.GetString(tokenConfig)) | ||||||
|  | 			b.rtm = b.sc.NewRTM() | ||||||
|  | 			go b.rtm.ManageConnection() | ||||||
|  | 			go b.handleSlack() | ||||||
|  | 		} | ||||||
|  | 	} else if b.GetString(tokenConfig) != "" { | ||||||
|  | 		b.Log.Info("Connecting using token (sending and receiving)") | ||||||
|  | 		b.sc = slack.New(b.GetString(tokenConfig)) | ||||||
|  | 		b.rtm = b.sc.NewRTM() | ||||||
|  | 		go b.rtm.ManageConnection() | ||||||
|  | 		go b.handleSlack() | ||||||
|  | 	} | ||||||
|  | 	if b.GetString(incomingWebhookConfig) == "" && b.GetString(outgoingWebhookConfig) == "" && b.GetString(tokenConfig) == "" { | ||||||
|  | 		return errors.New("no connection method found. See that you have WebhookBindAddress, WebhookURL or Token configured") | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @@ -31,7 +31,7 @@ func (b *Bsshchat) Connect() error { | |||||||
| 			b.w = w | 			b.w = w | ||||||
| 			b.r.Scan() | 			b.r.Scan() | ||||||
| 			w.Write([]byte("/theme mono\r\n")) | 			w.Write([]byte("/theme mono\r\n")) | ||||||
| 			b.handleSshChat() | 			b.handleSSHChat() | ||||||
| 			return nil | 			return nil | ||||||
| 		}) | 		}) | ||||||
| 	}() | 	}() | ||||||
| @@ -53,7 +53,7 @@ func (b *Bsshchat) JoinChannel(channel config.ChannelInfo) error { | |||||||
|  |  | ||||||
| func (b *Bsshchat) Send(msg config.Message) (string, error) { | func (b *Bsshchat) Send(msg config.Message) (string, error) { | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
| 	b.Log.Debugf("=> Receiving %#v", msg) | 	b.Log.Debugf("=> Receiving %#v", msg) | ||||||
| @@ -113,7 +113,7 @@ func stripPrompt(s string) string { | |||||||
| 	return s[pos+3:] | 	return s[pos+3:] | ||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bsshchat) handleSshChat() error { | func (b *Bsshchat) handleSSHChat() error { | ||||||
| 	/* | 	/* | ||||||
| 		done := b.sshchatKeepAlive() | 		done := b.sshchatKeepAlive() | ||||||
| 		defer close(done) | 		defer close(done) | ||||||
|   | |||||||
| @@ -2,6 +2,9 @@ package bsteam | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"strconv" | ||||||
|  | 	"sync" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/42wim/matterbridge/bridge" | 	"github.com/42wim/matterbridge/bridge" | ||||||
| 	"github.com/42wim/matterbridge/bridge/config" | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
| @@ -9,10 +12,6 @@ import ( | |||||||
| 	"github.com/Philipp15b/go-steam" | 	"github.com/Philipp15b/go-steam" | ||||||
| 	"github.com/Philipp15b/go-steam/protocol/steamlang" | 	"github.com/Philipp15b/go-steam/protocol/steamlang" | ||||||
| 	"github.com/Philipp15b/go-steam/steamid" | 	"github.com/Philipp15b/go-steam/steamid" | ||||||
| 	//"io/ioutil" |  | ||||||
| 	"strconv" |  | ||||||
| 	"sync" |  | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Bsteam struct { | type Bsteam struct { | ||||||
| @@ -61,7 +60,7 @@ func (b *Bsteam) JoinChannel(channel config.ChannelInfo) error { | |||||||
|  |  | ||||||
| func (b *Bsteam) Send(msg config.Message) (string, error) { | func (b *Bsteam) Send(msg config.Message) (string, error) { | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
| 	id, err := steamid.NewId(msg.Channel) | 	id, err := steamid.NewId(msg.Channel) | ||||||
| @@ -174,8 +173,6 @@ func (b *Bsteam) handleEvents() { | |||||||
| 			b.c.Connect() | 			b.c.Connect() | ||||||
| 		case steam.FatalErrorEvent: | 		case steam.FatalErrorEvent: | ||||||
| 			b.Log.Error(e) | 			b.Log.Error(e) | ||||||
| 		case error: |  | ||||||
| 			b.Log.Error(e) |  | ||||||
| 		default: | 		default: | ||||||
| 			b.Log.Debugf("unknown event %#v", e) | 			b.Log.Debugf("unknown event %#v", e) | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package btelegram | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"html" | 	"html" | ||||||
|  | 	"io" | ||||||
|  |  | ||||||
| 	"github.com/russross/blackfriday" | 	"github.com/russross/blackfriday" | ||||||
| ) | ) | ||||||
| @@ -32,7 +33,7 @@ func (options *customHTML) Header(out *bytes.Buffer, text func() bool, level int | |||||||
| 	options.Paragraph(out, text) | 	options.Paragraph(out, text) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (options *customHTML) HRule(out *bytes.Buffer) { | func (options *customHTML) HRule(out io.ByteWriter) { | ||||||
| 	out.WriteByte('\n') | 	out.WriteByte('\n') | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -53,13 +54,16 @@ func (options *customHTML) ListItem(out *bytes.Buffer, text []byte, flags int) { | |||||||
| } | } | ||||||
|  |  | ||||||
| func makeHTML(input string) string { | func makeHTML(input string) string { | ||||||
| 	return string(blackfriday.Markdown([]byte(input), | 	extensions := blackfriday.NoIntraEmphasis | | ||||||
| 		&customHTML{blackfriday.HtmlRenderer(blackfriday.HTML_USE_XHTML|blackfriday.HTML_SKIP_IMAGES, "", "")}, | 		blackfriday.FencedCode | | ||||||
| 		blackfriday.EXTENSION_NO_INTRA_EMPHASIS| | 		blackfriday.Autolink | | ||||||
| 			blackfriday.EXTENSION_FENCED_CODE| | 		blackfriday.SpaceHeadings | | ||||||
| 			blackfriday.EXTENSION_AUTOLINK| | 		blackfriday.HeadingIDs | | ||||||
| 			blackfriday.EXTENSION_SPACE_HEADERS| | 		blackfriday.BackslashLineBreak | | ||||||
| 			blackfriday.EXTENSION_HEADER_IDS| | 		blackfriday.DefinitionLists | ||||||
| 			blackfriday.EXTENSION_BACKSLASH_LINE_BREAK| |  | ||||||
| 			blackfriday.EXTENSION_DEFINITION_LISTS)) | 	renderer := &customHTML{blackfriday.NewHTMLRenderer(blackfriday.HTMLRendererParameters{ | ||||||
|  | 		Flags: blackfriday.UseXHTML | blackfriday.SkipImages, | ||||||
|  | 	})} | ||||||
|  | 	return string(blackfriday.Run([]byte(input), blackfriday.WithExtensions(extensions), blackfriday.WithRenderer(renderer))) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,6 +12,12 @@ import ( | |||||||
| 	"github.com/go-telegram-bot-api/telegram-bot-api" | 	"github.com/go-telegram-bot-api/telegram-bot-api" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	unknownUser = "unknown" | ||||||
|  | 	HTMLFormat  = "HTML" | ||||||
|  | 	HTMLNick    = "htmlnick" | ||||||
|  | ) | ||||||
|  |  | ||||||
| type Btelegram struct { | type Btelegram struct { | ||||||
| 	c *tgbotapi.BotAPI | 	c *tgbotapi.BotAPI | ||||||
| 	*bridge.Config | 	*bridge.Config | ||||||
| @@ -60,16 +66,16 @@ func (b *Btelegram) Send(msg config.Message) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// map the file SHA to our user (caches the avatar) | 	// map the file SHA to our user (caches the avatar) | ||||||
| 	if msg.Event == config.EVENT_AVATAR_DOWNLOAD { | 	if msg.Event == config.EventAvatarDownload { | ||||||
| 		return b.cacheAvatar(&msg) | 		return b.cacheAvatar(&msg) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if b.GetString("MessageFormat") == "HTML" { | 	if b.GetString("MessageFormat") == HTMLFormat { | ||||||
| 		msg.Text = makeHTML(msg.Text) | 		msg.Text = makeHTML(msg.Text) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
| @@ -98,12 +104,12 @@ func (b *Btelegram) Send(msg config.Message) (string, error) { | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return "", err | 			return "", err | ||||||
| 		} | 		} | ||||||
| 		if strings.ToLower(b.GetString("MessageFormat")) == "htmlnick" { | 		if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick { | ||||||
| 			b.Log.Debug("Using mode HTML - nick only") | 			b.Log.Debug("Using mode HTML - nick only") | ||||||
| 			msg.Text = html.EscapeString(msg.Text) | 			msg.Text = html.EscapeString(msg.Text) | ||||||
| 		} | 		} | ||||||
| 		m := tgbotapi.NewEditMessageText(chatid, msgid, msg.Username+msg.Text) | 		m := tgbotapi.NewEditMessageText(chatid, msgid, msg.Username+msg.Text) | ||||||
| 		if b.GetString("MessageFormat") == "HTML" { | 		if b.GetString("MessageFormat") == HTMLFormat { | ||||||
| 			b.Log.Debug("Using mode HTML") | 			b.Log.Debug("Using mode HTML") | ||||||
| 			m.ParseMode = tgbotapi.ModeHTML | 			m.ParseMode = tgbotapi.ModeHTML | ||||||
| 		} | 		} | ||||||
| @@ -111,7 +117,7 @@ func (b *Btelegram) Send(msg config.Message) (string, error) { | |||||||
| 			b.Log.Debug("Using mode markdown") | 			b.Log.Debug("Using mode markdown") | ||||||
| 			m.ParseMode = tgbotapi.ModeMarkdown | 			m.ParseMode = tgbotapi.ModeMarkdown | ||||||
| 		} | 		} | ||||||
| 		if strings.ToLower(b.GetString("MessageFormat")) == "htmlnick" { | 		if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick { | ||||||
| 			b.Log.Debug("Using mode HTML - nick only") | 			b.Log.Debug("Using mode HTML - nick only") | ||||||
| 			m.ParseMode = tgbotapi.ModeHTML | 			m.ParseMode = tgbotapi.ModeHTML | ||||||
| 		} | 		} | ||||||
| @@ -187,7 +193,7 @@ func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) { | |||||||
|  |  | ||||||
| 		// if we really didn't find a username, set it to unknown | 		// if we really didn't find a username, set it to unknown | ||||||
| 		if rmsg.Username == "" { | 		if rmsg.Username == "" { | ||||||
| 			rmsg.Username = "unknown" | 			rmsg.Username = unknownUser | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// handle any downloads | 		// handle any downloads | ||||||
| @@ -209,7 +215,7 @@ func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			if usernameForward == "" { | 			if usernameForward == "" { | ||||||
| 				usernameForward = "unknown" | 				usernameForward = unknownUser | ||||||
| 			} | 			} | ||||||
| 			rmsg.Text = "Forwarded from " + usernameForward + ": " + rmsg.Text | 			rmsg.Text = "Forwarded from " + usernameForward + ": " + rmsg.Text | ||||||
| 		} | 		} | ||||||
| @@ -229,7 +235,7 @@ func (b *Btelegram) handleRecv(updates <-chan tgbotapi.Update) { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			if usernameReply == "" { | 			if usernameReply == "" { | ||||||
| 				usernameReply = "unknown" | 				usernameReply = unknownUser | ||||||
| 			} | 			} | ||||||
| 			if !b.GetBool("QuoteDisable") { | 			if !b.GetBool("QuoteDisable") { | ||||||
| 				rmsg.Text = b.handleQuote(rmsg.Text, usernameReply, message.ReplyToMessage.Text) | 				rmsg.Text = b.handleQuote(rmsg.Text, usernameReply, message.ReplyToMessage.Text) | ||||||
| @@ -262,7 +268,7 @@ func (b *Btelegram) getFileDirectURL(id string) string { | |||||||
| // sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful. | // sends a EVENT_AVATAR_DOWNLOAD message to the gateway if successful. | ||||||
| // logs an error message if it fails | // logs an error message if it fails | ||||||
| func (b *Btelegram) handleDownloadAvatar(userid int, channel string) { | func (b *Btelegram) handleDownloadAvatar(userid int, channel string) { | ||||||
| 	rmsg := config.Message{Username: "system", Text: "avatar", Channel: channel, Account: b.Account, UserID: strconv.Itoa(userid), Event: config.EVENT_AVATAR_DOWNLOAD, Extra: make(map[string][]interface{})} | 	rmsg := config.Message{Username: "system", Text: "avatar", Channel: channel, Account: b.Account, UserID: strconv.Itoa(userid), Event: config.EventAvatarDownload, Extra: make(map[string][]interface{})} | ||||||
| 	if _, ok := b.avatarMap[strconv.Itoa(userid)]; !ok { | 	if _, ok := b.avatarMap[strconv.Itoa(userid)]; !ok { | ||||||
| 		photos, err := b.c.GetUserProfilePhotos(tgbotapi.UserProfilePhotosConfig{UserID: userid, Limit: 1}) | 		photos, err := b.c.GetUserProfilePhotos(tgbotapi.UserProfilePhotosConfig{UserID: userid, Limit: 1}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @@ -303,7 +309,7 @@ func (b *Btelegram) handleDownload(message *tgbotapi.Message, rmsg *config.Messa | |||||||
| 		urlPart := strings.Split(url, "/") | 		urlPart := strings.Split(url, "/") | ||||||
| 		name = urlPart[len(urlPart)-1] | 		name = urlPart[len(urlPart)-1] | ||||||
| 		if !strings.HasSuffix(name, ".webp") { | 		if !strings.HasSuffix(name, ".webp") { | ||||||
| 			name = name + ".webp" | 			name += ".webp" | ||||||
| 		} | 		} | ||||||
| 		text = " " + url | 		text = " " + url | ||||||
| 	} | 	} | ||||||
| @@ -338,7 +344,7 @@ func (b *Btelegram) handleDownload(message *tgbotapi.Message, rmsg *config.Messa | |||||||
| 		name = urlPart[len(urlPart)-1] | 		name = urlPart[len(urlPart)-1] | ||||||
| 		text = " " + url | 		text = " " + url | ||||||
| 		if !strings.HasSuffix(name, ".ogg") { | 		if !strings.HasSuffix(name, ".ogg") { | ||||||
| 			name = name + ".ogg" | 			name += ".ogg" | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if message.Audio != nil { | 	if message.Audio != nil { | ||||||
| @@ -356,7 +362,7 @@ func (b *Btelegram) handleDownload(message *tgbotapi.Message, rmsg *config.Messa | |||||||
| 	// use the URL instead of native upload | 	// use the URL instead of native upload | ||||||
| 	if b.GetBool("UseInsecureURL") { | 	if b.GetBool("UseInsecureURL") { | ||||||
| 		b.Log.Debugf("Setting message text to :%s", text) | 		b.Log.Debugf("Setting message text to :%s", text) | ||||||
| 		rmsg.Text = rmsg.Text + text | 		rmsg.Text += text | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	// if we have a file attached, download it (in memory) and put a pointer to it in msg.Extra | 	// if we have a file attached, download it (in memory) and put a pointer to it in msg.Extra | ||||||
| @@ -377,7 +383,10 @@ func (b *Btelegram) handleUploadFile(msg *config.Message, chatid int64) (string, | |||||||
| 	var c tgbotapi.Chattable | 	var c tgbotapi.Chattable | ||||||
| 	for _, f := range msg.Extra["file"] { | 	for _, f := range msg.Extra["file"] { | ||||||
| 		fi := f.(config.FileInfo) | 		fi := f.(config.FileInfo) | ||||||
| 		file := tgbotapi.FileBytes{fi.Name, *fi.Data} | 		file := tgbotapi.FileBytes{ | ||||||
|  | 			Name:  fi.Name, | ||||||
|  | 			Bytes: *fi.Data, | ||||||
|  | 		} | ||||||
| 		re := regexp.MustCompile(".(jpg|png)$") | 		re := regexp.MustCompile(".(jpg|png)$") | ||||||
| 		if re.MatchString(fi.Name) { | 		if re.MatchString(fi.Name) { | ||||||
| 			c = tgbotapi.NewPhotoUpload(chatid, file) | 			c = tgbotapi.NewPhotoUpload(chatid, file) | ||||||
| @@ -398,7 +407,7 @@ func (b *Btelegram) handleUploadFile(msg *config.Message, chatid int64) (string, | |||||||
| func (b *Btelegram) sendMessage(chatid int64, username, text string) (string, error) { | func (b *Btelegram) sendMessage(chatid int64, username, text string) (string, error) { | ||||||
| 	m := tgbotapi.NewMessage(chatid, "") | 	m := tgbotapi.NewMessage(chatid, "") | ||||||
| 	m.Text = username + text | 	m.Text = username + text | ||||||
| 	if b.GetString("MessageFormat") == "HTML" { | 	if b.GetString("MessageFormat") == HTMLFormat { | ||||||
| 		b.Log.Debug("Using mode HTML") | 		b.Log.Debug("Using mode HTML") | ||||||
| 		m.ParseMode = tgbotapi.ModeHTML | 		m.ParseMode = tgbotapi.ModeHTML | ||||||
| 	} | 	} | ||||||
| @@ -406,7 +415,7 @@ func (b *Btelegram) sendMessage(chatid int64, username, text string) (string, er | |||||||
| 		b.Log.Debug("Using mode markdown") | 		b.Log.Debug("Using mode markdown") | ||||||
| 		m.ParseMode = tgbotapi.ModeMarkdown | 		m.ParseMode = tgbotapi.ModeMarkdown | ||||||
| 	} | 	} | ||||||
| 	if strings.ToLower(b.GetString("MessageFormat")) == "htmlnick" { | 	if strings.ToLower(b.GetString("MessageFormat")) == HTMLNick { | ||||||
| 		b.Log.Debug("Using mode HTML - nick only") | 		b.Log.Debug("Using mode HTML - nick only") | ||||||
| 		m.Text = username + html.EscapeString(text) | 		m.Text = username + html.EscapeString(text) | ||||||
| 		m.ParseMode = tgbotapi.ModeHTML | 		m.ParseMode = tgbotapi.ModeHTML | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ func (b *Bxmpp) Connect() error { | |||||||
| 			time.Sleep(d) | 			time.Sleep(d) | ||||||
| 			b.xc, err = b.createXMPP() | 			b.xc, err = b.createXMPP() | ||||||
| 			if err == nil { | 			if err == nil { | ||||||
| 				b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EVENT_REJOIN_CHANNELS} | 				b.Remote <- config.Message{Username: "system", Text: "rejoin", Channel: "", Account: b.Account, Event: config.EventRejoinChannels} | ||||||
| 				b.handleXMPP() | 				b.handleXMPP() | ||||||
| 				bf.Reset() | 				bf.Reset() | ||||||
| 			} | 			} | ||||||
| @@ -75,10 +75,8 @@ func (b *Bxmpp) JoinChannel(channel config.ChannelInfo) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (b *Bxmpp) Send(msg config.Message) (string, error) { | func (b *Bxmpp) Send(msg config.Message) (string, error) { | ||||||
| 	var msgid = "" |  | ||||||
| 	var msgreplaceid = "" |  | ||||||
| 	// ignore delete messages | 	// ignore delete messages | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		return "", nil | 		return "", nil | ||||||
| 	} | 	} | ||||||
| 	b.Log.Debugf("=> Receiving %#v", msg) | 	b.Log.Debugf("=> Receiving %#v", msg) | ||||||
| @@ -93,7 +91,8 @@ func (b *Bxmpp) Send(msg config.Message) (string, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	msgid = xid.New().String() | 	var msgreplaceid string | ||||||
|  | 	msgid := xid.New().String() | ||||||
| 	if msg.ID != "" { | 	if msg.ID != "" { | ||||||
| 		msgid = msg.ID | 		msgid = msg.ID | ||||||
| 		msgreplaceid = msg.ID | 		msgreplaceid = msg.ID | ||||||
| @@ -178,7 +177,7 @@ func (b *Bxmpp) handleXMPP() error { | |||||||
| 				// check if we have an action event | 				// check if we have an action event | ||||||
| 				rmsg.Text, ok = b.replaceAction(rmsg.Text) | 				rmsg.Text, ok = b.replaceAction(rmsg.Text) | ||||||
| 				if ok { | 				if ok { | ||||||
| 					rmsg.Event = config.EVENT_USER_ACTION | 					rmsg.Event = config.EventUserAction | ||||||
| 				} | 				} | ||||||
| 				b.Log.Debugf("<= Sending message from %s on %s to gateway", rmsg.Username, b.Account) | 				b.Log.Debugf("<= Sending message from %s on %s to gateway", rmsg.Username, b.Account) | ||||||
| 				b.Log.Debugf("<= Message is %#v", rmsg) | 				b.Log.Debugf("<= Message is %#v", rmsg) | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ func (b *Bzulip) Send(msg config.Message) (string, error) { | |||||||
| 	b.Log.Debugf("=> Receiving %#v", msg) | 	b.Log.Debugf("=> Receiving %#v", msg) | ||||||
|  |  | ||||||
| 	// Delete message | 	// Delete message | ||||||
| 	if msg.Event == config.EVENT_MSG_DELETE { | 	if msg.Event == config.EventMsgDelete { | ||||||
| 		if msg.ID == "" { | 		if msg.ID == "" { | ||||||
| 			return "", nil | 			return "", nil | ||||||
| 		} | 		} | ||||||
|   | |||||||
							
								
								
									
										47
									
								
								changelog.md
									
									
									
									
									
								
							
							
						
						| @@ -1,3 +1,50 @@ | |||||||
|  | # v1.12.1 | ||||||
|  |  | ||||||
|  | ## Bugfix | ||||||
|  | * discord: fix regression on server ID connection #619 #617 | ||||||
|  | * discord: Limit discord username via webhook to 32 chars | ||||||
|  | * slack: Make sure threaded files stay in thread (slack). Fixes #590 | ||||||
|  | * slack: Do not post empty messages (slack). Fixes #574 | ||||||
|  | * slack: Handle deleted/edited thread starting messages (slack). Fixes #600 (#605) | ||||||
|  | * irc: Rework connection logic (irc) | ||||||
|  | * irc: Fix Nickserv logic (irc) #602 | ||||||
|  |  | ||||||
|  | # v1.12.0 | ||||||
|  |  | ||||||
|  | ## Breaking changes | ||||||
|  | The slack bridge has been split in a `slack-legacy` and `slack` bridge. | ||||||
|  | If you're still using `legacy tokens` and want to keep using them you'll have to rename `slack` to `slack-legacy` in your configuration. See [wiki](https://github.com/42wim/matterbridge/wiki/Section-Slack-(basic)#legacy-configuration) for more information. | ||||||
|  |  | ||||||
|  | To migrate to the new bot-token based setup you can follow the instructions [here](https://github.com/42wim/matterbridge/wiki/Slack-bot-setup). | ||||||
|  |  | ||||||
|  | Slack legacy tokens may be deprecated by Slack at short notice, so it is STRONGLY recommended to use a proper bot-token instead. | ||||||
|  |  | ||||||
|  | ## New features | ||||||
|  | * general: New {GATEWAY} variable for `RemoteNickFormat` #501. See `RemoteNickFormat` in matterbridge.toml.sample. | ||||||
|  | * general: New {CHANNEL} variable for `RemoteNickFormat` #515. See `RemoteNickFormat` in matterbridge.toml.sample. | ||||||
|  | * general: Remove hyphens when auto-loading envvars from viper config #545 | ||||||
|  | * discord: You can mention discord-users from other bridges. | ||||||
|  | * slack: Preserve threading between Slack instances #529. See `PreserveThreading` in matterbridge.toml.sample. | ||||||
|  | * slack: Add ability to show when user is typing across Slack bridges #559 | ||||||
|  | * slack: Add rate-limiting | ||||||
|  | * mattermost: Add support for mattermost [matterbridge plugin](https://github.com/matterbridge/mattermost-plugin) | ||||||
|  | * api: Respond with message on connect. #550 | ||||||
|  | * api: Add a health endpoint to API #554 | ||||||
|  |  | ||||||
|  | ## Bugfix | ||||||
|  | * slack: Refactoring and making it better. | ||||||
|  | * slack: Restore file comments coming from Slack. #583 | ||||||
|  | * irc: Fix IRC line splitting. #587 | ||||||
|  | * mattermost: Fix cookie and personal token behaviour. #530 | ||||||
|  | * mattermost: Check for expiring sessions and reconnect. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Contributors | ||||||
|  | This release couldn't exist without the following contributors: | ||||||
|  | @jheiselman, @NikkyAI, @dajohi, @NetwideRogue, @patcon and @Helcaraxan | ||||||
|  |  | ||||||
|  | Special thanks to @Helcaraxan and @patcon for their work on improving/refactoring slack. | ||||||
|  |  | ||||||
| # v1.11.3 | # v1.11.3 | ||||||
|  |  | ||||||
| ## Bugfix | ## Bugfix | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| #!/bin/bash | #!/bin/bash | ||||||
| go version |grep go1.10 || exit | go version | grep go1.11 || exit | ||||||
| VERSION=$(git describe --tags) | VERSION=$(git describe --tags) | ||||||
| mkdir ci/binaries | mkdir ci/binaries | ||||||
| GOOS=windows GOARCH=amd64 go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-windows-amd64.exe | GOOS=windows GOARCH=amd64 go build -ldflags "-s -w -X main.githash=$(git log --pretty=format:'%h' -n 1)" -o ci/binaries/matterbridge-$VERSION-windows-amd64.exe | ||||||
|   | |||||||
| @@ -2,10 +2,15 @@ package gateway | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"crypto/sha1" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"regexp" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/42wim/matterbridge/bridge" | 	"github.com/42wim/matterbridge/bridge" | ||||||
| 	"github.com/42wim/matterbridge/bridge/api" | 	"github.com/42wim/matterbridge/bridge/api" | ||||||
| @@ -23,19 +28,13 @@ import ( | |||||||
| 	bxmpp "github.com/42wim/matterbridge/bridge/xmpp" | 	bxmpp "github.com/42wim/matterbridge/bridge/xmpp" | ||||||
| 	bzulip "github.com/42wim/matterbridge/bridge/zulip" | 	bzulip "github.com/42wim/matterbridge/bridge/zulip" | ||||||
| 	"github.com/hashicorp/golang-lru" | 	"github.com/hashicorp/golang-lru" | ||||||
| 	log "github.com/sirupsen/logrus" |  | ||||||
| 	//	"github.com/davecgh/go-spew/spew" |  | ||||||
| 	"crypto/sha1" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"regexp" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"github.com/peterhellberg/emojilib" | 	"github.com/peterhellberg/emojilib" | ||||||
|  | 	log "github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Gateway struct { | type Gateway struct { | ||||||
| 	*config.Config | 	config.Config | ||||||
|  |  | ||||||
| 	Router         *Router | 	Router         *Router | ||||||
| 	MyConfig       *config.Gateway | 	MyConfig       *config.Gateway | ||||||
| 	Bridges        map[string]*bridge.Bridge | 	Bridges        map[string]*bridge.Bridge | ||||||
| @@ -55,26 +54,28 @@ type BrMsgID struct { | |||||||
| var flog *log.Entry | var flog *log.Entry | ||||||
|  |  | ||||||
| var bridgeMap = map[string]bridge.Factory{ | var bridgeMap = map[string]bridge.Factory{ | ||||||
| 	"api":        api.New, | 	"api":          api.New, | ||||||
| 	"discord":    bdiscord.New, | 	"discord":      bdiscord.New, | ||||||
| 	"gitter":     bgitter.New, | 	"gitter":       bgitter.New, | ||||||
| 	"irc":        birc.New, | 	"irc":          birc.New, | ||||||
| 	"mattermost": bmattermost.New, | 	"mattermost":   bmattermost.New, | ||||||
| 	"matrix":     bmatrix.New, | 	"matrix":       bmatrix.New, | ||||||
| 	"rocketchat": brocketchat.New, | 	"rocketchat":   brocketchat.New, | ||||||
| 	"slack":      bslack.New, | 	"slack-legacy": bslack.NewLegacy, | ||||||
| 	"sshchat":    bsshchat.New, | 	"slack":        bslack.New, | ||||||
| 	"steam":      bsteam.New, | 	"sshchat":      bsshchat.New, | ||||||
| 	"telegram":   btelegram.New, | 	"steam":        bsteam.New, | ||||||
| 	"xmpp":       bxmpp.New, | 	"telegram":     btelegram.New, | ||||||
| 	"zulip":      bzulip.New, | 	"xmpp":         bxmpp.New, | ||||||
|  | 	"zulip":        bzulip.New, | ||||||
| } | } | ||||||
|  |  | ||||||
| func init() { | const ( | ||||||
| 	flog = log.WithFields(log.Fields{"prefix": "gateway"}) | 	apiProtocol = "api" | ||||||
| } | ) | ||||||
|  |  | ||||||
| func New(cfg config.Gateway, r *Router) *Gateway { | func New(cfg config.Gateway, r *Router) *Gateway { | ||||||
|  | 	flog = log.WithFields(log.Fields{"prefix": "gateway"}) | ||||||
| 	gw := &Gateway{Channels: make(map[string]*config.ChannelInfo), Message: r.Message, | 	gw := &Gateway{Channels: make(map[string]*config.ChannelInfo), Message: r.Message, | ||||||
| 		Router: r, Bridges: make(map[string]*bridge.Bridge), Config: r.Config} | 		Router: r, Bridges: make(map[string]*bridge.Bridge), Config: r.Config} | ||||||
| 	cache, _ := lru.New(5000) | 	cache, _ := lru.New(5000) | ||||||
| @@ -83,12 +84,32 @@ func New(cfg config.Gateway, r *Router) *Gateway { | |||||||
| 	return gw | 	return gw | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Find the canonical ID that the message is keyed under in cache | ||||||
|  | func (gw *Gateway) FindCanonicalMsgID(protocol string, mID string) string { | ||||||
|  | 	ID := protocol + " " + mID | ||||||
|  | 	if gw.Messages.Contains(ID) { | ||||||
|  | 		return mID | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If not keyed, iterate through cache for downstream, and infer upstream. | ||||||
|  | 	for _, mid := range gw.Messages.Keys() { | ||||||
|  | 		v, _ := gw.Messages.Peek(mid) | ||||||
|  | 		ids := v.([]*BrMsgID) | ||||||
|  | 		for _, downstreamMsgObj := range ids { | ||||||
|  | 			if ID == downstreamMsgObj.ID { | ||||||
|  | 				return strings.Replace(mid.(string), protocol+" ", "", 1) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
| func (gw *Gateway) AddBridge(cfg *config.Bridge) error { | func (gw *Gateway) AddBridge(cfg *config.Bridge) error { | ||||||
| 	br := gw.Router.getBridge(cfg.Account) | 	br := gw.Router.getBridge(cfg.Account) | ||||||
| 	if br == nil { | 	if br == nil { | ||||||
| 		br = bridge.New(cfg) | 		br = bridge.New(cfg) | ||||||
| 		br.Config = gw.Router.Config | 		br.Config = gw.Router.Config | ||||||
| 		br.General = &gw.General | 		br.General = &gw.BridgeValues().General | ||||||
| 		// set logging | 		// set logging | ||||||
| 		br.Log = log.WithFields(log.Fields{"prefix": "bridge"}) | 		br.Log = log.WithFields(log.Fields{"prefix": "bridge"}) | ||||||
| 		brconfig := &bridge.Config{Remote: gw.Message, Log: log.WithFields(log.Fields{"prefix": br.Protocol}), Bridge: br} | 		brconfig := &bridge.Config{Remote: gw.Message, Log: log.WithFields(log.Fields{"prefix": br.Protocol}), Bridge: br} | ||||||
| @@ -105,6 +126,7 @@ func (gw *Gateway) AddConfig(cfg *config.Gateway) error { | |||||||
| 	gw.MyConfig = cfg | 	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...)...) { | ||||||
|  | 		br := br //scopelint | ||||||
| 		err := gw.AddBridge(&br) | 		err := gw.AddBridge(&br) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| @@ -138,8 +160,8 @@ RECONNECT: | |||||||
|  |  | ||||||
| func (gw *Gateway) mapChannelConfig(cfg []config.Bridge, direction string) { | func (gw *Gateway) mapChannelConfig(cfg []config.Bridge, direction string) { | ||||||
| 	for _, br := range cfg { | 	for _, br := range cfg { | ||||||
| 		if isApi(br.Account) { | 		if isAPI(br.Account) { | ||||||
| 			br.Channel = "api" | 			br.Channel = apiProtocol | ||||||
| 		} | 		} | ||||||
| 		// make sure to lowercase irc channels in config #348 | 		// make sure to lowercase irc channels in config #348 | ||||||
| 		if strings.HasPrefix(br.Account, "irc.") { | 		if strings.HasPrefix(br.Account, "irc.") { | ||||||
| @@ -172,7 +194,7 @@ func (gw *Gateway) getDestChannel(msg *config.Message, dest bridge.Bridge) []con | |||||||
| 	var channels []config.ChannelInfo | 	var channels []config.ChannelInfo | ||||||
|  |  | ||||||
| 	// for messages received from the api check that the gateway is the specified one | 	// for messages received from the api check that the gateway is the specified one | ||||||
| 	if msg.Protocol == "api" && gw.Name != msg.Gateway { | 	if msg.Protocol == apiProtocol && gw.Name != msg.Gateway { | ||||||
| 		return channels | 		return channels | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -199,19 +221,33 @@ func (gw *Gateway) getDestChannel(msg *config.Message, dest bridge.Bridge) []con | |||||||
| 			} | 			} | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		if strings.Contains(channel.Direction, "out") && channel.Account == dest.Account && gw.validGatewayDest(msg, channel) { | 		if strings.Contains(channel.Direction, "out") && channel.Account == dest.Account && gw.validGatewayDest(msg) { | ||||||
| 			channels = append(channels, *channel) | 			channels = append(channels, *channel) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return channels | 	return channels | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (gw *Gateway) getDestMsgID(msgID string, dest *bridge.Bridge, channel config.ChannelInfo) string { | ||||||
|  | 	if res, ok := gw.Messages.Get(msgID); ok { | ||||||
|  | 		IDs := res.([]*BrMsgID) | ||||||
|  | 		for _, id := range IDs { | ||||||
|  | 			// check protocol, bridge name and channelname | ||||||
|  | 			// for people that reuse the same bridge multiple times. see #342 | ||||||
|  | 			if dest.Protocol == id.br.Protocol && dest.Name == id.br.Name && channel.ID == id.ChannelID { | ||||||
|  | 				return strings.Replace(id.ID, dest.Protocol+" ", "", 1) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
| func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) []*BrMsgID { | func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) []*BrMsgID { | ||||||
| 	var brMsgIDs []*BrMsgID | 	var brMsgIDs []*BrMsgID | ||||||
|  |  | ||||||
| 	// if we have an attached file, or other info | 	// if we have an attached file, or other info | ||||||
| 	if msg.Extra != nil { | 	if msg.Extra != nil { | ||||||
| 		if len(msg.Extra[config.EVENT_FILE_FAILURE_SIZE]) != 0 { | 		if len(msg.Extra[config.EventFileFailureSize]) != 0 { | ||||||
| 			if msg.Text == "" { | 			if msg.Text == "" { | ||||||
| 				return brMsgIDs | 				return brMsgIDs | ||||||
| 			} | 			} | ||||||
| @@ -219,7 +255,7 @@ func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) []*BrM | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Avatar downloads are only relevant for telegram and mattermost for now | 	// Avatar downloads are only relevant for telegram and mattermost for now | ||||||
| 	if msg.Event == config.EVENT_AVATAR_DOWNLOAD { | 	if msg.Event == config.EventAvatarDownload { | ||||||
| 		if dest.Protocol != "mattermost" && | 		if dest.Protocol != "mattermost" && | ||||||
| 			dest.Protocol != "telegram" { | 			dest.Protocol != "telegram" { | ||||||
| 			return brMsgIDs | 			return brMsgIDs | ||||||
| @@ -227,27 +263,33 @@ func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) []*BrM | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// only relay join/part when configured | 	// only relay join/part when configured | ||||||
| 	if msg.Event == config.EVENT_JOIN_LEAVE && !gw.Bridges[dest.Account].GetBool("ShowJoinPart") { | 	if msg.Event == config.EventJoinLeave && !gw.Bridges[dest.Account].GetBool("ShowJoinPart") { | ||||||
| 		return brMsgIDs | 		return brMsgIDs | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// only relay topic change when configured | 	// only relay topic change when configured | ||||||
| 	if msg.Event == config.EVENT_TOPIC_CHANGE && !gw.Bridges[dest.Account].GetBool("ShowTopicChange") { | 	if msg.Event == config.EventTopicChange && !gw.Bridges[dest.Account].GetBool("ShowTopicChange") { | ||||||
| 		return brMsgIDs | 		return brMsgIDs | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// broadcast to every out channel (irc QUIT) | 	// broadcast to every out channel (irc QUIT) | ||||||
| 	if msg.Channel == "" && msg.Event != config.EVENT_JOIN_LEAVE { | 	if msg.Channel == "" && msg.Event != config.EventJoinLeave { | ||||||
| 		flog.Debug("empty channel") | 		flog.Debug("empty channel") | ||||||
| 		return brMsgIDs | 		return brMsgIDs | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// Get the ID of the parent message in thread | ||||||
|  | 	var canonicalParentMsgID string | ||||||
|  | 	if msg.ParentID != "" && (gw.BridgeValues().General.PreserveThreading || dest.GetBool("PreserveThreading")) { | ||||||
|  | 		canonicalParentMsgID = gw.FindCanonicalMsgID(msg.Protocol, msg.ParentID) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	originchannel := msg.Channel | 	originchannel := msg.Channel | ||||||
| 	origmsg := msg | 	origmsg := msg | ||||||
| 	channels := gw.getDestChannel(&msg, *dest) | 	channels := gw.getDestChannel(&msg, *dest) | ||||||
| 	for _, channel := range channels { | 	for _, channel := range channels { | ||||||
| 		// Only send the avatar download event to ourselves. | 		// Only send the avatar download event to ourselves. | ||||||
| 		if msg.Event == config.EVENT_AVATAR_DOWNLOAD { | 		if msg.Event == config.EventAvatarDownload { | ||||||
| 			if channel.ID != getChannelID(origmsg) { | 			if channel.ID != getChannelID(origmsg) { | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| @@ -257,33 +299,43 @@ func (gw *Gateway) handleMessage(msg config.Message, dest *bridge.Bridge) []*BrM | |||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		flog.Debugf("=> Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, originchannel, dest.Account, channel.Name) |  | ||||||
|  | 		// Too noisy to log like other events | ||||||
|  | 		if msg.Event != config.EventUserTyping { | ||||||
|  | 			flog.Debugf("=> Sending %#v from %s (%s) to %s (%s)", msg, msg.Account, originchannel, dest.Account, channel.Name) | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		msg.Channel = channel.Name | 		msg.Channel = channel.Name | ||||||
| 		msg.Avatar = gw.modifyAvatar(origmsg, dest) | 		msg.Avatar = gw.modifyAvatar(origmsg, dest) | ||||||
| 		msg.Username = gw.modifyUsername(origmsg, dest) | 		msg.Username = gw.modifyUsername(origmsg, dest) | ||||||
| 		msg.ID = "" |  | ||||||
| 		if res, ok := gw.Messages.Get(origmsg.ID); ok { | 		msg.ID = gw.getDestMsgID(origmsg.Protocol+" "+origmsg.ID, dest, channel) | ||||||
| 			IDs := res.([]*BrMsgID) |  | ||||||
| 			for _, id := range IDs { |  | ||||||
| 				// check protocol, bridge name and channelname |  | ||||||
| 				// for people that reuse the same bridge multiple times. see #342 |  | ||||||
| 				if dest.Protocol == id.br.Protocol && dest.Name == id.br.Name && channel.ID == id.ChannelID { |  | ||||||
| 					msg.ID = id.ID |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		// for api we need originchannel as channel | 		// for api we need originchannel as channel | ||||||
| 		if dest.Protocol == "api" { | 		if dest.Protocol == apiProtocol { | ||||||
| 			msg.Channel = originchannel | 			msg.Channel = originchannel | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		msg.ParentID = gw.getDestMsgID(origmsg.Protocol+" "+canonicalParentMsgID, dest, channel) | ||||||
|  | 		if msg.ParentID == "" { | ||||||
|  | 			msg.ParentID = canonicalParentMsgID | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// if we are using mattermost plugin account, send messages to MattermostPlugin channel | ||||||
|  | 		// that can be picked up by the mattermost matterbridge plugin | ||||||
|  | 		if dest.Account == "mattermost.plugin" { | ||||||
|  | 			gw.Router.MattermostPlugin <- msg | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		mID, err := dest.Send(msg) | 		mID, err := dest.Send(msg) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			flog.Error(err) | 			flog.Error(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// append the message ID (mID) from this bridge (dest) to our brMsgIDs slice | 		// append the message ID (mID) from this bridge (dest) to our brMsgIDs slice | ||||||
| 		if mID != "" { | 		if mID != "" { | ||||||
| 			flog.Debugf("mID %s: %s", dest.Account, mID) | 			flog.Debugf("mID %s: %s", dest.Account, mID) | ||||||
| 			brMsgIDs = append(brMsgIDs, &BrMsgID{dest, mID, channel.ID}) | 			brMsgIDs = append(brMsgIDs, &BrMsgID{dest, dest.Protocol + " " + mID, channel.ID}) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return brMsgIDs | 	return brMsgIDs | ||||||
| @@ -297,11 +349,14 @@ func (gw *Gateway) ignoreMessage(msg *config.Message) bool { | |||||||
|  |  | ||||||
| 	// check if we need to ignore a empty message | 	// check if we need to ignore a empty message | ||||||
| 	if msg.Text == "" { | 	if msg.Text == "" { | ||||||
|  | 		if msg.Event == config.EventUserTyping { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
| 		// we have an attachment or actual bytes, do not ignore | 		// we have an attachment or actual bytes, do not ignore | ||||||
| 		if msg.Extra != nil && | 		if msg.Extra != nil && | ||||||
| 			(msg.Extra["attachments"] != nil || | 			(msg.Extra["attachments"] != nil || | ||||||
| 				len(msg.Extra["file"]) > 0 || | 				len(msg.Extra["file"]) > 0 || | ||||||
| 				len(msg.Extra[config.EVENT_FILE_FAILURE_SIZE]) > 0) { | 				len(msg.Extra[config.EventFileFailureSize]) > 0) { | ||||||
| 			return false | 			return false | ||||||
| 		} | 		} | ||||||
| 		flog.Debugf("ignoring empty message %#v from %s", msg, msg.Account) | 		flog.Debugf("ignoring empty message %#v from %s", msg, msg.Account) | ||||||
| @@ -337,13 +392,13 @@ func (gw *Gateway) ignoreMessage(msg *config.Message) bool { | |||||||
| func (gw *Gateway) modifyUsername(msg config.Message, dest *bridge.Bridge) string { | func (gw *Gateway) modifyUsername(msg config.Message, dest *bridge.Bridge) string { | ||||||
| 	br := gw.Bridges[msg.Account] | 	br := gw.Bridges[msg.Account] | ||||||
| 	msg.Protocol = br.Protocol | 	msg.Protocol = br.Protocol | ||||||
| 	if gw.Config.General.StripNick || dest.GetBool("StripNick") { | 	if gw.BridgeValues().General.StripNick || dest.GetBool("StripNick") { | ||||||
| 		re := regexp.MustCompile("[^a-zA-Z0-9]+") | 		re := regexp.MustCompile("[^a-zA-Z0-9]+") | ||||||
| 		msg.Username = re.ReplaceAllString(msg.Username, "") | 		msg.Username = re.ReplaceAllString(msg.Username, "") | ||||||
| 	} | 	} | ||||||
| 	nick := dest.GetString("RemoteNickFormat") | 	nick := dest.GetString("RemoteNickFormat") | ||||||
| 	if nick == "" { | 	if nick == "" { | ||||||
| 		nick = gw.Config.General.RemoteNickFormat | 		nick = gw.BridgeValues().General.RemoteNickFormat | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// loop to replace nicks | 	// loop to replace nicks | ||||||
| @@ -374,13 +429,15 @@ func (gw *Gateway) modifyUsername(msg config.Message, dest *bridge.Bridge) strin | |||||||
|  |  | ||||||
| 	nick = strings.Replace(nick, "{BRIDGE}", br.Name, -1) | 	nick = strings.Replace(nick, "{BRIDGE}", br.Name, -1) | ||||||
| 	nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1) | 	nick = strings.Replace(nick, "{PROTOCOL}", br.Protocol, -1) | ||||||
|  | 	nick = strings.Replace(nick, "{GATEWAY}", gw.Name, -1) | ||||||
| 	nick = strings.Replace(nick, "{LABEL}", br.GetString("Label"), -1) | 	nick = strings.Replace(nick, "{LABEL}", br.GetString("Label"), -1) | ||||||
| 	nick = strings.Replace(nick, "{NICK}", msg.Username, -1) | 	nick = strings.Replace(nick, "{NICK}", msg.Username, -1) | ||||||
|  | 	nick = strings.Replace(nick, "{CHANNEL}", msg.Channel, -1) | ||||||
| 	return nick | 	return nick | ||||||
| } | } | ||||||
|  |  | ||||||
| func (gw *Gateway) modifyAvatar(msg config.Message, dest *bridge.Bridge) string { | func (gw *Gateway) modifyAvatar(msg config.Message, dest *bridge.Bridge) string { | ||||||
| 	iconurl := gw.Config.General.IconURL | 	iconurl := gw.BridgeValues().General.IconURL | ||||||
| 	if iconurl == "" { | 	if iconurl == "" { | ||||||
| 		iconurl = dest.GetString("IconURL") | 		iconurl = dest.GetString("IconURL") | ||||||
| 	} | 	} | ||||||
| @@ -410,7 +467,7 @@ func (gw *Gateway) modifyMessage(msg *config.Message) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// messages from api have Gateway specified, don't overwrite | 	// messages from api have Gateway specified, don't overwrite | ||||||
| 	if msg.Protocol != "api" { | 	if msg.Protocol != apiProtocol { | ||||||
| 		msg.Gateway = gw.Name | 		msg.Gateway = gw.Name | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @@ -421,7 +478,9 @@ func (gw *Gateway) handleFiles(msg *config.Message) { | |||||||
| 	reg := regexp.MustCompile("[^a-zA-Z0-9]+") | 	reg := regexp.MustCompile("[^a-zA-Z0-9]+") | ||||||
|  |  | ||||||
| 	// If we don't have a attachfield or we don't have a mediaserver configured return | 	// If we don't have a attachfield or we don't have a mediaserver configured return | ||||||
| 	if msg.Extra == nil || (gw.Config.General.MediaServerUpload == "" && gw.Config.General.MediaDownloadPath == "") { | 	if msg.Extra == nil || | ||||||
|  | 		(gw.BridgeValues().General.MediaServerUpload == "" && | ||||||
|  | 			gw.BridgeValues().General.MediaDownloadPath == "") { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -439,14 +498,14 @@ func (gw *Gateway) handleFiles(msg *config.Message) { | |||||||
| 		ext := filepath.Ext(fi.Name) | 		ext := filepath.Ext(fi.Name) | ||||||
| 		fi.Name = fi.Name[0 : len(fi.Name)-len(ext)] | 		fi.Name = fi.Name[0 : len(fi.Name)-len(ext)] | ||||||
| 		fi.Name = reg.ReplaceAllString(fi.Name, "_") | 		fi.Name = reg.ReplaceAllString(fi.Name, "_") | ||||||
| 		fi.Name = fi.Name + ext | 		fi.Name += ext | ||||||
|  |  | ||||||
| 		sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] | 		sha1sum := fmt.Sprintf("%x", sha1.Sum(*fi.Data))[:8] | ||||||
|  |  | ||||||
| 		if gw.Config.General.MediaServerUpload != "" { | 		if gw.BridgeValues().General.MediaServerUpload != "" { | ||||||
| 			// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth. | 			// Use MediaServerUpload. Upload using a PUT HTTP request and basicauth. | ||||||
|  |  | ||||||
| 			url := gw.Config.General.MediaServerUpload + "/" + sha1sum + "/" + fi.Name | 			url := gw.BridgeValues().General.MediaServerUpload + "/" + sha1sum + "/" + fi.Name | ||||||
|  |  | ||||||
| 			req, err := http.NewRequest("PUT", url, bytes.NewReader(*fi.Data)) | 			req, err := http.NewRequest("PUT", url, bytes.NewReader(*fi.Data)) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| @@ -465,7 +524,7 @@ func (gw *Gateway) handleFiles(msg *config.Message) { | |||||||
| 		} else { | 		} else { | ||||||
| 			// Use MediaServerPath. Place the file on the current filesystem. | 			// Use MediaServerPath. Place the file on the current filesystem. | ||||||
|  |  | ||||||
| 			dir := gw.Config.General.MediaDownloadPath + "/" + sha1sum | 			dir := gw.BridgeValues().General.MediaDownloadPath + "/" + sha1sum | ||||||
| 			err := os.Mkdir(dir, os.ModePerm) | 			err := os.Mkdir(dir, os.ModePerm) | ||||||
| 			if err != nil && !os.IsExist(err) { | 			if err != nil && !os.IsExist(err) { | ||||||
| 				flog.Errorf("mediaserver path failed, could not mkdir: %s %#v", err, err) | 				flog.Errorf("mediaserver path failed, could not mkdir: %s %#v", err, err) | ||||||
| @@ -483,7 +542,7 @@ func (gw *Gateway) handleFiles(msg *config.Message) { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Download URL. | 		// Download URL. | ||||||
| 		durl := gw.Config.General.MediaServerDownload + "/" + sha1sum + "/" + fi.Name | 		durl := gw.BridgeValues().General.MediaServerDownload + "/" + sha1sum + "/" + fi.Name | ||||||
|  |  | ||||||
| 		flog.Debugf("mediaserver download URL = %s", durl) | 		flog.Debugf("mediaserver download URL = %s", durl) | ||||||
|  |  | ||||||
| @@ -495,7 +554,7 @@ func (gw *Gateway) handleFiles(msg *config.Message) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (gw *Gateway) validGatewayDest(msg *config.Message, channel *config.ChannelInfo) bool { | func (gw *Gateway) validGatewayDest(msg *config.Message) bool { | ||||||
| 	return msg.Gateway == gw.Name | 	return msg.Gateway == gw.Name | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -503,6 +562,6 @@ func getChannelID(msg config.Message) string { | |||||||
| 	return msg.Channel + msg.Account | 	return msg.Channel + msg.Account | ||||||
| } | } | ||||||
|  |  | ||||||
| func isApi(account string) bool { | func isAPI(account string) bool { | ||||||
| 	return strings.HasPrefix(account, "api.") | 	return strings.HasPrefix(account, "api.") | ||||||
| } | } | ||||||
|   | |||||||
| @@ -152,6 +152,12 @@ enable=true | |||||||
|     channel="--333333333333" |     channel="--333333333333" | ||||||
| `) | `) | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	ircTestAccount   = "irc.zzz" | ||||||
|  | 	tgTestAccount    = "telegram.zzz" | ||||||
|  | 	slackTestAccount = "slack.zzz" | ||||||
|  | ) | ||||||
|  |  | ||||||
| func maketestRouter(input []byte) *Router { | func maketestRouter(input []byte) *Router { | ||||||
| 	cfg := config.NewConfigFromString(input) | 	cfg := config.NewConfigFromString(input) | ||||||
| 	r, err := NewRouter(cfg) | 	r, err := NewRouter(cfg) | ||||||
| @@ -172,18 +178,27 @@ func TestNewRouter(t *testing.T) { | |||||||
| 	assert.Equal(t, 3, len(r.Gateways["bridge2"].Bridges)) | 	assert.Equal(t, 3, len(r.Gateways["bridge2"].Bridges)) | ||||||
| 	assert.Equal(t, 4, len(r.Gateways["bridge1"].Channels)) | 	assert.Equal(t, 4, len(r.Gateways["bridge1"].Channels)) | ||||||
| 	assert.Equal(t, 3, len(r.Gateways["bridge2"].Channels)) | 	assert.Equal(t, 3, len(r.Gateways["bridge2"].Channels)) | ||||||
| 	assert.Equal(t, &config.ChannelInfo{Name: "42wim/testroom", Direction: "out", | 	assert.Equal(t, &config.ChannelInfo{ | ||||||
| 		ID: "42wim/testroomgitter.42wim", Account: "gitter.42wim", | 		Name:        "42wim/testroom", | ||||||
| 		SameChannel: map[string]bool{"bridge2": false}}, | 		Direction:   "out", | ||||||
| 		r.Gateways["bridge2"].Channels["42wim/testroomgitter.42wim"]) | 		ID:          "42wim/testroomgitter.42wim", | ||||||
| 	assert.Equal(t, &config.ChannelInfo{Name: "42wim/testroom", Direction: "in", | 		Account:     "gitter.42wim", | ||||||
| 		ID: "42wim/testroomgitter.42wim", Account: "gitter.42wim", | 		SameChannel: map[string]bool{"bridge2": false}, | ||||||
| 		SameChannel: map[string]bool{"bridge1": false}}, | 	}, r.Gateways["bridge2"].Channels["42wim/testroomgitter.42wim"]) | ||||||
| 		r.Gateways["bridge1"].Channels["42wim/testroomgitter.42wim"]) | 	assert.Equal(t, &config.ChannelInfo{ | ||||||
| 	assert.Equal(t, &config.ChannelInfo{Name: "general", Direction: "inout", | 		Name:        "42wim/testroom", | ||||||
| 		ID: "generaldiscord.test", Account: "discord.test", | 		Direction:   "in", | ||||||
| 		SameChannel: map[string]bool{"bridge1": false}}, | 		ID:          "42wim/testroomgitter.42wim", | ||||||
| 		r.Gateways["bridge1"].Channels["generaldiscord.test"]) | 		Account:     "gitter.42wim", | ||||||
|  | 		SameChannel: map[string]bool{"bridge1": false}, | ||||||
|  | 	}, r.Gateways["bridge1"].Channels["42wim/testroomgitter.42wim"]) | ||||||
|  | 	assert.Equal(t, &config.ChannelInfo{ | ||||||
|  | 		Name:        "general", | ||||||
|  | 		Direction:   "inout", | ||||||
|  | 		ID:          "generaldiscord.test", | ||||||
|  | 		Account:     "discord.test", | ||||||
|  | 		SameChannel: map[string]bool{"bridge1": false}, | ||||||
|  | 	}, r.Gateways["bridge1"].Channels["generaldiscord.test"]) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestGetDestChannel(t *testing.T) { | func TestGetDestChannel(t *testing.T) { | ||||||
| @@ -192,11 +207,23 @@ func TestGetDestChannel(t *testing.T) { | |||||||
| 	for _, br := range r.Gateways["bridge1"].Bridges { | 	for _, br := range r.Gateways["bridge1"].Bridges { | ||||||
| 		switch br.Account { | 		switch br.Account { | ||||||
| 		case "discord.test": | 		case "discord.test": | ||||||
| 			assert.Equal(t, []config.ChannelInfo{{Name: "general", Account: "discord.test", Direction: "inout", ID: "generaldiscord.test", SameChannel: map[string]bool{"bridge1": false}, Options: config.ChannelOptions{Key: ""}}}, | 			assert.Equal(t, []config.ChannelInfo{{ | ||||||
| 				r.Gateways["bridge1"].getDestChannel(msg, *br)) | 				Name:        "general", | ||||||
|  | 				Account:     "discord.test", | ||||||
|  | 				Direction:   "inout", | ||||||
|  | 				ID:          "generaldiscord.test", | ||||||
|  | 				SameChannel: map[string]bool{"bridge1": false}, | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 			}}, r.Gateways["bridge1"].getDestChannel(msg, *br)) | ||||||
| 		case "slack.test": | 		case "slack.test": | ||||||
| 			assert.Equal(t, []config.ChannelInfo{{Name: "testing", Account: "slack.test", Direction: "out", ID: "testingslack.test", SameChannel: map[string]bool{"bridge1": false}, Options: config.ChannelOptions{Key: ""}}}, | 			assert.Equal(t, []config.ChannelInfo{{ | ||||||
| 				r.Gateways["bridge1"].getDestChannel(msg, *br)) | 				Name:        "testing", | ||||||
|  | 				Account:     "slack.test", | ||||||
|  | 				Direction:   "out", | ||||||
|  | 				ID:          "testingslack.test", | ||||||
|  | 				SameChannel: map[string]bool{"bridge1": false}, | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 			}}, r.Gateways["bridge1"].getDestChannel(msg, *br)) | ||||||
| 		case "gitter.42wim": | 		case "gitter.42wim": | ||||||
| 			assert.Equal(t, []config.ChannelInfo(nil), r.Gateways["bridge1"].getDestChannel(msg, *br)) | 			assert.Equal(t, []config.ChannelInfo(nil), r.Gateways["bridge1"].getDestChannel(msg, *br)) | ||||||
| 		case "irc.freenode": | 		case "irc.freenode": | ||||||
| @@ -226,35 +253,87 @@ func TestGetDestChannelAdvanced(t *testing.T) { | |||||||
| 				} | 				} | ||||||
| 				switch gw.Name { | 				switch gw.Name { | ||||||
| 				case "bridge": | 				case "bridge": | ||||||
| 					if (msg.Channel == "#main" || msg.Channel == "-1111111111111" || msg.Channel == "irc") && (msg.Account == "irc.zzz" || msg.Account == "telegram.zzz" || msg.Account == "slack.zzz") { | 					if (msg.Channel == "#main" || msg.Channel == "-1111111111111" || msg.Channel == "irc") && | ||||||
|  | 						(msg.Account == ircTestAccount || msg.Account == tgTestAccount || msg.Account == slackTestAccount) { | ||||||
| 						hits[gw.Name]++ | 						hits[gw.Name]++ | ||||||
| 						switch br.Account { | 						switch br.Account { | ||||||
| 						case "irc.zzz": | 						case ircTestAccount: | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "#main", Account: "irc.zzz", Direction: "inout", ID: "#mainirc.zzz", SameChannel: map[string]bool{"bridge": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
| 						case "telegram.zzz": | 								Name:        "#main", | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "-1111111111111", Account: "telegram.zzz", Direction: "inout", ID: "-1111111111111telegram.zzz", SameChannel: map[string]bool{"bridge": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 								Account:     ircTestAccount, | ||||||
| 						case "slack.zzz": | 								Direction:   "inout", | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "irc", Account: "slack.zzz", Direction: "inout", ID: "ircslack.zzz", SameChannel: map[string]bool{"bridge": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 								ID:          "#mainirc.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
|  | 						case tgTestAccount: | ||||||
|  | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 								Name:        "-1111111111111", | ||||||
|  | 								Account:     tgTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "-1111111111111telegram.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
|  | 						case slackTestAccount: | ||||||
|  | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 								Name:        "irc", | ||||||
|  | 								Account:     slackTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "ircslack.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 				case "bridge2": | 				case "bridge2": | ||||||
| 					if (msg.Channel == "#main-help" || msg.Channel == "--444444444444") && (msg.Account == "irc.zzz" || msg.Account == "telegram.zzz") { | 					if (msg.Channel == "#main-help" || msg.Channel == "--444444444444") && | ||||||
|  | 						(msg.Account == ircTestAccount || msg.Account == tgTestAccount) { | ||||||
| 						hits[gw.Name]++ | 						hits[gw.Name]++ | ||||||
| 						switch br.Account { | 						switch br.Account { | ||||||
| 						case "irc.zzz": | 						case ircTestAccount: | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "#main-help", Account: "irc.zzz", Direction: "inout", ID: "#main-helpirc.zzz", SameChannel: map[string]bool{"bridge2": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
| 						case "telegram.zzz": | 								Name:        "#main-help", | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "--444444444444", Account: "telegram.zzz", Direction: "inout", ID: "--444444444444telegram.zzz", SameChannel: map[string]bool{"bridge2": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 								Account:     ircTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "#main-helpirc.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge2": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
|  | 						case tgTestAccount: | ||||||
|  | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 								Name:        "--444444444444", | ||||||
|  | 								Account:     tgTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "--444444444444telegram.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge2": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 				case "bridge3": | 				case "bridge3": | ||||||
| 					if (msg.Channel == "#main-telegram" || msg.Channel == "--333333333333") && (msg.Account == "irc.zzz" || msg.Account == "telegram.zzz") { | 					if (msg.Channel == "#main-telegram" || msg.Channel == "--333333333333") && | ||||||
|  | 						(msg.Account == ircTestAccount || msg.Account == tgTestAccount) { | ||||||
| 						hits[gw.Name]++ | 						hits[gw.Name]++ | ||||||
| 						switch br.Account { | 						switch br.Account { | ||||||
| 						case "irc.zzz": | 						case ircTestAccount: | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "#main-telegram", Account: "irc.zzz", Direction: "inout", ID: "#main-telegramirc.zzz", SameChannel: map[string]bool{"bridge3": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
| 						case "telegram.zzz": | 								Name:        "#main-telegram", | ||||||
| 							assert.Equal(t, []config.ChannelInfo{{Name: "--333333333333", Account: "telegram.zzz", Direction: "inout", ID: "--333333333333telegram.zzz", SameChannel: map[string]bool{"bridge3": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 								Account:     ircTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "#main-telegramirc.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge3": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
|  | 						case tgTestAccount: | ||||||
|  | 							assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 								Name:        "--333333333333", | ||||||
|  | 								Account:     tgTestAccount, | ||||||
|  | 								Direction:   "inout", | ||||||
|  | 								ID:          "--333333333333telegram.zzz", | ||||||
|  | 								SameChannel: map[string]bool{"bridge3": false}, | ||||||
|  | 								Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 							}}, channels) | ||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 				case "announcements": | 				case "announcements": | ||||||
| @@ -264,12 +343,42 @@ func TestGetDestChannelAdvanced(t *testing.T) { | |||||||
| 					} | 					} | ||||||
| 					hits[gw.Name]++ | 					hits[gw.Name]++ | ||||||
| 					switch br.Account { | 					switch br.Account { | ||||||
| 					case "irc.zzz": | 					case ircTestAccount: | ||||||
| 						assert.Equal(t, []config.ChannelInfo{{Name: "#main", Account: "irc.zzz", Direction: "out", ID: "#mainirc.zzz", SameChannel: map[string]bool{"announcements": false}, Options: config.ChannelOptions{Key: ""}}, {Name: "#main-help", Account: "irc.zzz", Direction: "out", ID: "#main-helpirc.zzz", SameChannel: map[string]bool{"announcements": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 						assert.Len(t, channels, 2) | ||||||
| 					case "slack.zzz": | 						assert.Contains(t, channels, config.ChannelInfo{ | ||||||
| 						assert.Equal(t, []config.ChannelInfo{{Name: "general", Account: "slack.zzz", Direction: "out", ID: "generalslack.zzz", SameChannel: map[string]bool{"announcements": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 							Name:        "#main", | ||||||
| 					case "telegram.zzz": | 							Account:     ircTestAccount, | ||||||
| 						assert.Equal(t, []config.ChannelInfo{{Name: "--333333333333", Account: "telegram.zzz", Direction: "out", ID: "--333333333333telegram.zzz", SameChannel: map[string]bool{"announcements": false}, Options: config.ChannelOptions{Key: ""}}}, channels) | 							Direction:   "out", | ||||||
|  | 							ID:          "#mainirc.zzz", | ||||||
|  | 							SameChannel: map[string]bool{"announcements": false}, | ||||||
|  | 							Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 						}) | ||||||
|  | 						assert.Contains(t, channels, config.ChannelInfo{ | ||||||
|  | 							Name:        "#main-help", | ||||||
|  | 							Account:     ircTestAccount, | ||||||
|  | 							Direction:   "out", | ||||||
|  | 							ID:          "#main-helpirc.zzz", | ||||||
|  | 							SameChannel: map[string]bool{"announcements": false}, | ||||||
|  | 							Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 						}) | ||||||
|  | 					case slackTestAccount: | ||||||
|  | 						assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 							Name:        "general", | ||||||
|  | 							Account:     slackTestAccount, | ||||||
|  | 							Direction:   "out", | ||||||
|  | 							ID:          "generalslack.zzz", | ||||||
|  | 							SameChannel: map[string]bool{"announcements": false}, | ||||||
|  | 							Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 						}}, channels) | ||||||
|  | 					case tgTestAccount: | ||||||
|  | 						assert.Equal(t, []config.ChannelInfo{{ | ||||||
|  | 							Name:        "--333333333333", | ||||||
|  | 							Account:     tgTestAccount, | ||||||
|  | 							Direction:   "out", | ||||||
|  | 							ID:          "--333333333333telegram.zzz", | ||||||
|  | 							SameChannel: map[string]bool{"announcements": false}, | ||||||
|  | 							Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 						}}, channels) | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|   | |||||||
| @@ -2,26 +2,32 @@ package gateway | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/42wim/matterbridge/bridge" | 	"github.com/42wim/matterbridge/bridge" | ||||||
| 	"github.com/42wim/matterbridge/bridge/config" | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
| 	samechannelgateway "github.com/42wim/matterbridge/gateway/samechannel" | 	samechannelgateway "github.com/42wim/matterbridge/gateway/samechannel" | ||||||
| 	//	"github.com/davecgh/go-spew/spew" |  | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Router struct { | type Router struct { | ||||||
| 	Gateways map[string]*Gateway | 	config.Config | ||||||
| 	Message  chan config.Message |  | ||||||
| 	*config.Config | 	Gateways         map[string]*Gateway | ||||||
|  | 	Message          chan config.Message | ||||||
|  | 	MattermostPlugin chan config.Message | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewRouter(cfg *config.Config) (*Router, error) { | func NewRouter(cfg config.Config) (*Router, error) { | ||||||
| 	r := &Router{Message: make(chan config.Message), Gateways: make(map[string]*Gateway), Config: cfg} | 	r := &Router{ | ||||||
|  | 		Config:           cfg, | ||||||
|  | 		Message:          make(chan config.Message), | ||||||
|  | 		MattermostPlugin: make(chan config.Message), | ||||||
|  | 		Gateways:         make(map[string]*Gateway), | ||||||
|  | 	} | ||||||
| 	sgw := samechannelgateway.New(cfg) | 	sgw := samechannelgateway.New(cfg) | ||||||
| 	gwconfigs := sgw.GetConfig() | 	gwconfigs := sgw.GetConfig() | ||||||
|  |  | ||||||
| 	for _, entry := range append(gwconfigs, cfg.Gateway...) { | 	for _, entry := range append(gwconfigs, cfg.BridgeValues().Gateway...) { | ||||||
| 		if !entry.Enable { | 		if !entry.Enable { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| @@ -70,7 +76,8 @@ func (r *Router) getBridge(account string) *bridge.Bridge { | |||||||
|  |  | ||||||
| func (r *Router) handleReceive() { | func (r *Router) handleReceive() { | ||||||
| 	for msg := range r.Message { | 	for msg := range r.Message { | ||||||
| 		if msg.Event == config.EVENT_FAILURE { | 		msg := msg // scopelint | ||||||
|  | 		if msg.Event == config.EventFailure { | ||||||
| 		Loop: | 		Loop: | ||||||
| 			for _, gw := range r.Gateways { | 			for _, gw := range r.Gateways { | ||||||
| 				for _, br := range gw.Bridges { | 				for _, br := range gw.Bridges { | ||||||
| @@ -81,7 +88,7 @@ func (r *Router) handleReceive() { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		if msg.Event == config.EVENT_REJOIN_CHANNELS { | 		if msg.Event == config.EventRejoinChannels { | ||||||
| 			for _, gw := range r.Gateways { | 			for _, gw := range r.Gateways { | ||||||
| 				for _, br := range gw.Bridges { | 				for _, br := range gw.Bridges { | ||||||
| 					if msg.Account == br.Account { | 					if msg.Account == br.Account { | ||||||
| @@ -102,8 +109,8 @@ func (r *Router) handleReceive() { | |||||||
| 					msgIDs = append(msgIDs, gw.handleMessage(msg, br)...) | 					msgIDs = append(msgIDs, gw.handleMessage(msg, br)...) | ||||||
| 				} | 				} | ||||||
| 				// only add the message ID if it doesn't already exists | 				// only add the message ID if it doesn't already exists | ||||||
| 				if _, ok := gw.Messages.Get(msg.ID); !ok && msg.ID != "" { | 				if _, ok := gw.Messages.Get(msg.Protocol + " " + msg.ID); !ok && msg.ID != "" { | ||||||
| 					gw.Messages.Add(msg.ID, msgIDs) | 					gw.Messages.Add(msg.Protocol+" "+msg.ID, msgIDs) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -5,17 +5,17 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| type SameChannelGateway struct { | type SameChannelGateway struct { | ||||||
| 	*config.Config | 	config.Config | ||||||
| } | } | ||||||
|  |  | ||||||
| func New(cfg *config.Config) *SameChannelGateway { | func New(cfg config.Config) *SameChannelGateway { | ||||||
| 	return &SameChannelGateway{Config: cfg} | 	return &SameChannelGateway{Config: cfg} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (sgw *SameChannelGateway) GetConfig() []config.Gateway { | func (sgw *SameChannelGateway) GetConfig() []config.Gateway { | ||||||
| 	var gwconfigs []config.Gateway | 	var gwconfigs []config.Gateway | ||||||
| 	cfg := sgw.Config | 	cfg := sgw.Config | ||||||
| 	for _, gw := range cfg.SameChannelGateway { | 	for _, gw := range cfg.BridgeValues().SameChannelGateway { | ||||||
| 		gwconfig := config.Gateway{Name: gw.Name, Enable: gw.Enable} | 		gwconfig := config.Gateway{Name: gw.Name, Enable: gw.Enable} | ||||||
| 		for _, account := range gw.Accounts { | 		for _, account := range gw.Accounts { | ||||||
| 			for _, channel := range gw.Channels { | 			for _, channel := range gw.Channels { | ||||||
|   | |||||||
| @@ -1,16 +1,13 @@ | |||||||
| package samechannelgateway | package samechannelgateway | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" |  | ||||||
|  |  | ||||||
| 	"github.com/42wim/matterbridge/bridge/config" | 	"github.com/42wim/matterbridge/bridge/config" | ||||||
| 	"github.com/BurntSushi/toml" |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
|  |  | ||||||
| 	"testing" | 	"testing" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var testconfig = ` | const testConfig = ` | ||||||
| [mattermost.test] | [mattermost.test] | ||||||
| [slack.test] | [slack.test] | ||||||
|  |  | ||||||
| @@ -21,12 +18,56 @@ var testconfig = ` | |||||||
|       channels = [ "testing","testing2","testing10"] |       channels = [ "testing","testing2","testing10"] | ||||||
| ` | ` | ||||||
|  |  | ||||||
| func TestGetConfig(t *testing.T) { | var ( | ||||||
| 	var cfg *config.Config | 	expectedConfig = config.Gateway{ | ||||||
| 	if _, err := toml.Decode(testconfig, &cfg); err != nil { | 		Name:   "blah", | ||||||
| 		fmt.Println(err) | 		Enable: true, | ||||||
|  | 		In:     []config.Bridge(nil), | ||||||
|  | 		Out:    []config.Bridge(nil), | ||||||
|  | 		InOut: []config.Bridge{ | ||||||
|  | 			{ | ||||||
|  | 				Account:     "mattermost.test", | ||||||
|  | 				Channel:     "testing", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Account:     "mattermost.test", | ||||||
|  | 				Channel:     "testing2", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Account:     "mattermost.test", | ||||||
|  | 				Channel:     "testing10", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Account:     "slack.test", | ||||||
|  | 				Channel:     "testing", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Account:     "slack.test", | ||||||
|  | 				Channel:     "testing2", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Account:     "slack.test", | ||||||
|  | 				Channel:     "testing10", | ||||||
|  | 				Options:     config.ChannelOptions{Key: ""}, | ||||||
|  | 				SameChannel: true, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
| 	} | 	} | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestGetConfig(t *testing.T) { | ||||||
|  | 	cfg := config.NewConfigFromString([]byte(testConfig)) | ||||||
| 	sgw := New(cfg) | 	sgw := New(cfg) | ||||||
| 	configs := sgw.GetConfig() | 	configs := sgw.GetConfig() | ||||||
| 	assert.Equal(t, []config.Gateway{{Name: "blah", Enable: true, In: []config.Bridge(nil), Out: []config.Bridge(nil), InOut: []config.Bridge{{Account: "mattermost.test", Channel: "testing", Options: config.ChannelOptions{Key: ""}, SameChannel: true}, {Account: "mattermost.test", Channel: "testing2", Options: config.ChannelOptions{Key: ""}, SameChannel: true}, {Account: "mattermost.test", Channel: "testing10", Options: config.ChannelOptions{Key: ""}, SameChannel: true}, {Account: "slack.test", Channel: "testing", Options: config.ChannelOptions{Key: ""}, SameChannel: true}, {Account: "slack.test", Channel: "testing2", Options: config.ChannelOptions{Key: ""}, SameChannel: true}, {Account: "slack.test", Channel: "testing10", Options: config.ChannelOptions{Key: ""}, SameChannel: true}}}}, configs) | 	assert.Equal(t, []config.Gateway{expectedConfig}, configs) | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						| @@ -2,12 +2,10 @@ module github.com/42wim/matterbridge | |||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/42wim/go-gitter v0.0.0-20170828205020-017310c2d557 | 	github.com/42wim/go-gitter v0.0.0-20170828205020-017310c2d557 | ||||||
| 	github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14 | 	github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14 // indirect | ||||||
| 	github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3 | 	github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3 | ||||||
| 	github.com/Sirupsen/logrus v1.0.6 // indirect |  | ||||||
| 	github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b // indirect | 	github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b // indirect | ||||||
| 	github.com/bwmarrin/discordgo v0.0.0-20180201002541-8d5ab59c63e5 // indirect | 	github.com/bwmarrin/discordgo v0.19.0 | ||||||
| 	github.com/davecgh/go-spew v1.1.0 // indirect |  | ||||||
| 	github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d | 	github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d | ||||||
| 	github.com/dgrijalva/jwt-go v0.0.0-20170508165458-6c8dedd55f8a // indirect | 	github.com/dgrijalva/jwt-go v0.0.0-20170508165458-6c8dedd55f8a // indirect | ||||||
| 	github.com/fsnotify/fsnotify v1.4.7 | 	github.com/fsnotify/fsnotify v1.4.7 | ||||||
| @@ -16,7 +14,7 @@ require ( | |||||||
| 	github.com/google/gops v0.0.0-20170319002943-62f833fc9f6c | 	github.com/google/gops v0.0.0-20170319002943-62f833fc9f6c | ||||||
| 	github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f // indirect | 	github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f // indirect | ||||||
| 	github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c | 	github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c | ||||||
| 	github.com/gorilla/websocket v0.0.0-20170319172727-a91eba7f9777 | 	github.com/gorilla/websocket v1.4.0 | ||||||
| 	github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad | 	github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad | ||||||
| 	github.com/hashicorp/hcl v0.0.0-20171017181929-23c074d0eceb // indirect | 	github.com/hashicorp/hcl v0.0.0-20171017181929-23c074d0eceb // indirect | ||||||
| 	github.com/hpcloud/tail v1.0.0 // indirect | 	github.com/hpcloud/tail v1.0.0 // indirect | ||||||
| @@ -26,9 +24,10 @@ require ( | |||||||
| 	github.com/kr/pretty v0.1.0 // indirect | 	github.com/kr/pretty v0.1.0 // indirect | ||||||
| 	github.com/labstack/echo v0.0.0-20180219162101-7eec915044a1 | 	github.com/labstack/echo v0.0.0-20180219162101-7eec915044a1 | ||||||
| 	github.com/labstack/gommon v0.2.1 // indirect | 	github.com/labstack/gommon v0.2.1 // indirect | ||||||
| 	github.com/lrstanley/girc v0.0.0-20180427160007-102f17f86306 | 	github.com/lrstanley/girc v0.0.0-20180913221000-0fb5b684054e | ||||||
|  | 	github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5 // indirect | ||||||
|  | 	github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6 // indirect | ||||||
| 	github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885 // indirect | 	github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885 // indirect | ||||||
| 	github.com/matterbridge/discordgo v0.0.0-20180806170629-ef40ff5ba64f |  | ||||||
| 	github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91 | 	github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91 | ||||||
| 	github.com/matterbridge/gomatrix v0.0.0-20171224233421-78ac6a1a0f5f | 	github.com/matterbridge/gomatrix v0.0.0-20171224233421-78ac6a1a0f5f | ||||||
| 	github.com/matterbridge/gozulipbot v0.0.0-20180507190239-b6bb12d33544 | 	github.com/matterbridge/gozulipbot v0.0.0-20180507190239-b6bb12d33544 | ||||||
| @@ -41,7 +40,7 @@ require ( | |||||||
| 	github.com/mreiferson/go-httpclient v0.0.0-20160630210159-31f0106b4474 // indirect | 	github.com/mreiferson/go-httpclient v0.0.0-20160630210159-31f0106b4474 // indirect | ||||||
| 	github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff // indirect | 	github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff // indirect | ||||||
| 	github.com/nicksnyder/go-i18n v1.4.0 // indirect | 	github.com/nicksnyder/go-i18n v1.4.0 // indirect | ||||||
| 	github.com/nlopes/slack v0.3.1-0.20180805133408-21749ab136a8 | 	github.com/nlopes/slack v0.4.0 | ||||||
| 	github.com/onsi/ginkgo v1.6.0 // indirect | 	github.com/onsi/ginkgo v1.6.0 // indirect | ||||||
| 	github.com/onsi/gomega v1.4.1 // indirect | 	github.com/onsi/gomega v1.4.1 // indirect | ||||||
| 	github.com/paulrosania/go-charset v0.0.0-20151028000031-621bb39fcc83 | 	github.com/paulrosania/go-charset v0.0.0-20151028000031-621bb39fcc83 | ||||||
| @@ -49,13 +48,13 @@ require ( | |||||||
| 	github.com/pelletier/go-toml v0.0.0-20180228233631-05bcc0fb0d3e // indirect | 	github.com/pelletier/go-toml v0.0.0-20180228233631-05bcc0fb0d3e // indirect | ||||||
| 	github.com/peterhellberg/emojilib v0.0.0-20170616163716-41920917e271 | 	github.com/peterhellberg/emojilib v0.0.0-20170616163716-41920917e271 | ||||||
| 	github.com/pkg/errors v0.8.0 // indirect | 	github.com/pkg/errors v0.8.0 // indirect | ||||||
| 	github.com/pmezard/go-difflib v1.0.0 // indirect |  | ||||||
| 	github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a | 	github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a | ||||||
| 	github.com/russross/blackfriday v1.5.1 | 	github.com/russross/blackfriday v2.0.0+incompatible | ||||||
| 	github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca | 	github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca | ||||||
| 	github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1 // indirect | 	github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1 // indirect | ||||||
| 	github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991 | 	github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991 | ||||||
| 	github.com/sirupsen/logrus v0.0.0-20180213143110-8c0189d9f6bb | 	github.com/shurcooL/sanitized_anchor_name v0.0.0-20170918181015-86672fcb3f95 // indirect | ||||||
|  | 	github.com/sirupsen/logrus v1.2.0 | ||||||
| 	github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 // indirect | 	github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 // indirect | ||||||
| 	github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a // indirect | 	github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a // indirect | ||||||
| 	github.com/spf13/afero v0.0.0-20180211162714-bbf41cb36dff // indirect | 	github.com/spf13/afero v0.0.0-20180211162714-bbf41cb36dff // indirect | ||||||
| @@ -63,21 +62,19 @@ require ( | |||||||
| 	github.com/spf13/jwalterweatherman v0.0.0-20180109140146-7c0cea34c8ec // indirect | 	github.com/spf13/jwalterweatherman v0.0.0-20180109140146-7c0cea34c8ec // indirect | ||||||
| 	github.com/spf13/pflag v0.0.0-20180220143236-ee5fd03fd6ac // indirect | 	github.com/spf13/pflag v0.0.0-20180220143236-ee5fd03fd6ac // indirect | ||||||
| 	github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7 | 	github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7 | ||||||
| 	github.com/stretchr/testify v0.0.0-20170714215325-05e8a0eda380 | 	github.com/stretchr/testify v1.2.2 | ||||||
| 	github.com/technoweenie/multipartstreamer v1.0.1 // indirect | 	github.com/technoweenie/multipartstreamer v1.0.1 // indirect | ||||||
| 	github.com/valyala/bytebufferpool v0.0.0-20160817181652-e746df99fe4a // indirect | 	github.com/valyala/bytebufferpool v0.0.0-20160817181652-e746df99fe4a // indirect | ||||||
| 	github.com/valyala/fasttemplate v0.0.0-20170224212429-dcecefd839c4 // indirect | 	github.com/valyala/fasttemplate v0.0.0-20170224212429-dcecefd839c4 // indirect | ||||||
| 	github.com/x-cray/logrus-prefixed-formatter v0.5.2 // indirect | 	github.com/x-cray/logrus-prefixed-formatter v0.5.2 // indirect | ||||||
| 	github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6 | 	github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6 | ||||||
| 	golang.org/x/crypto v0.0.0-20180228161326-91a49db82a88 // indirect | 	golang.org/x/crypto v0.0.0-20181112202954-3d3f9f413869 // indirect | ||||||
| 	golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37 // indirect | 	golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37 // indirect | ||||||
| 	golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f // indirect | 	golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f // indirect | ||||||
| 	golang.org/x/sys v0.0.0-20171130163741-8b4580aae2a0 // indirect | 	golang.org/x/sys v0.0.0-20181116161606-93218def8b18 // indirect | ||||||
| 	golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978 // indirect | 	golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978 // indirect | ||||||
| 	gopkg.in/airbrake/gobrake.v2 v2.0.9 // indirect |  | ||||||
| 	gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect | 	gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect | ||||||
| 	gopkg.in/fsnotify.v1 v1.4.7 // indirect | 	gopkg.in/fsnotify.v1 v1.4.7 // indirect | ||||||
| 	gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.1 // indirect |  | ||||||
| 	gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect | 	gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect | ||||||
| 	gopkg.in/yaml.v2 v2.0.0-20160301204022-a83829b6f129 // indirect | 	gopkg.in/yaml.v2 v2.0.0-20160301204022-a83829b6f129 // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										64
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						| @@ -4,14 +4,12 @@ github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14 h1:v/zr4ns/4sSahF9 | |||||||
| github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | github.com/BurntSushi/toml v0.0.0-20170318202913-d94612f9fc14/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||||
| github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3 h1:V4+1E1SRYUySqwOoI3ZphFADtabbF568zTHa5ix/zU0= | github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3 h1:V4+1E1SRYUySqwOoI3ZphFADtabbF568zTHa5ix/zU0= | ||||||
| github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3/go.mod h1:HuVM+sZFzumUdKPWiz+IlCMb4RdsKdT3T+nQBKL+sYg= | github.com/Philipp15b/go-steam v0.0.0-20161020161927-e0f3bb9566e3/go.mod h1:HuVM+sZFzumUdKPWiz+IlCMb4RdsKdT3T+nQBKL+sYg= | ||||||
| github.com/Sirupsen/logrus v1.0.6 h1:HCAGQRk48dRVPA5Y+Yh0qdCSTzPOyU1tBJ7Q9YzotII= |  | ||||||
| github.com/Sirupsen/logrus v1.0.6/go.mod h1:rmk17hk6i8ZSAJkSDa7nOxamrG+SP4P0mm+DAvExv4U= |  | ||||||
| github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b h1:1OpGXps6UOY5HtQaQcLowsV1qMWCNBzhFvK7q4fgXtc= | github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b h1:1OpGXps6UOY5HtQaQcLowsV1qMWCNBzhFvK7q4fgXtc= | ||||||
| github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b/go.mod h1:iCVmQ9g4TfaRX5m5jq5sXY7RXYWPv9/PynM/GocbG3w= | github.com/alecthomas/log4go v0.0.0-20160307011253-e5dc62318d9b/go.mod h1:iCVmQ9g4TfaRX5m5jq5sXY7RXYWPv9/PynM/GocbG3w= | ||||||
| github.com/bwmarrin/discordgo v0.0.0-20180201002541-8d5ab59c63e5 h1:M7u44DKGpA5goDIBf0zRMYhT1Sp2Rd7hiTzXfeuw1UY= | github.com/bwmarrin/discordgo v0.19.0 h1:kMED/DB0NR1QhRcalb85w0Cu3Ep2OrGAqZH1R5awQiY= | ||||||
| github.com/bwmarrin/discordgo v0.0.0-20180201002541-8d5ab59c63e5/go.mod h1:5NIvFv5Z7HddYuXbuQegZ684DleQaCFqChP2iuBivJ8= | github.com/bwmarrin/discordgo v0.19.0/go.mod h1:O9S4p+ofTFwB02em7jkpkV8M3R0/PUVOwN61zSZ0r4Q= | ||||||
| github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8= | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d h1:rONNnZDE5CYuaSFQk+gP4GEQTXEUcyQ5p6p/dgxIHas= | github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d h1:rONNnZDE5CYuaSFQk+gP4GEQTXEUcyQ5p6p/dgxIHas= | ||||||
| github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d/go.mod h1:UGa5M2Sz/Uh13AMse4+RELKCDw7kqgqlTjeGae+7vUY= | github.com/dfordsoft/golib v0.0.0-20180313113957-2ea3495aee1d/go.mod h1:UGa5M2Sz/Uh13AMse4+RELKCDw7kqgqlTjeGae+7vUY= | ||||||
| github.com/dgrijalva/jwt-go v0.0.0-20170508165458-6c8dedd55f8a h1:MuHMeSsXbNEeUyxjB7T9P8s1+5k8OLTC/M27qsVwixM= | github.com/dgrijalva/jwt-go v0.0.0-20170508165458-6c8dedd55f8a h1:MuHMeSsXbNEeUyxjB7T9P8s1+5k8OLTC/M27qsVwixM= | ||||||
| @@ -28,8 +26,8 @@ github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f h1:FDM3EtwZLyhW4 | |||||||
| github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= | github.com/gopherjs/gopherjs v0.0.0-20180628210949-0892b62f0d9f/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= | ||||||
| github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c h1:mORYpib1aLu3M2Oi50Z1pNTXuDJEHcoLb6oo6VdOutk= | github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c h1:mORYpib1aLu3M2Oi50Z1pNTXuDJEHcoLb6oo6VdOutk= | ||||||
| github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c/go.mod h1:kgLaKoK1FELgZqMAVxx/5cbj0kT+57qxUrAlIO2eleU= | github.com/gorilla/schema v0.0.0-20170317173100-f3c80893412c/go.mod h1:kgLaKoK1FELgZqMAVxx/5cbj0kT+57qxUrAlIO2eleU= | ||||||
| github.com/gorilla/websocket v0.0.0-20170319172727-a91eba7f9777 h1:JIM+OacoOJRU30xpjMf8sulYqjr0ViA3WDrTX6j/yDI= | github.com/gorilla/websocket v1.4.0 h1:WDFjx/TMzVgy9VdMMQi2K2Emtwi2QcUQsztZ/zLaH/Q= | ||||||
| github.com/gorilla/websocket v0.0.0-20170319172727-a91eba7f9777/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= | github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= | ||||||
| github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad h1:eMxs9EL0PvIGS9TTtxg4R+JxuPGav82J8rA+GFnY7po= | github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad h1:eMxs9EL0PvIGS9TTtxg4R+JxuPGav82J8rA+GFnY7po= | ||||||
| github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | github.com/hashicorp/golang-lru v0.0.0-20160813221303-0a025b7e63ad/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||||
| github.com/hashicorp/hcl v0.0.0-20171017181929-23c074d0eceb h1:1OvvPvZkn/yCQ3xBcM8y4020wdkMXPHLB4+NfoGWh4U= | github.com/hashicorp/hcl v0.0.0-20171017181929-23c074d0eceb h1:1OvvPvZkn/yCQ3xBcM8y4020wdkMXPHLB4+NfoGWh4U= | ||||||
| @@ -42,6 +40,8 @@ github.com/jtolds/gls v4.2.1+incompatible h1:fSuqC+Gmlu6l/ZYAoZzx2pyucC8Xza35fpR | |||||||
| github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | ||||||
| github.com/kardianos/osext v0.0.0-20170207191655-9b883c5eb462 h1:oSOOTPHkCzMeu1vJ0nHxg5+XZBdMMjNa+6NPnm8arok= | github.com/kardianos/osext v0.0.0-20170207191655-9b883c5eb462 h1:oSOOTPHkCzMeu1vJ0nHxg5+XZBdMMjNa+6NPnm8arok= | ||||||
| github.com/kardianos/osext v0.0.0-20170207191655-9b883c5eb462/go.mod h1:1NbS8ALrpOvjt0rHPNLyCIeMtbizbir8U//inJ+zuB8= | github.com/kardianos/osext v0.0.0-20170207191655-9b883c5eb462/go.mod h1:1NbS8ALrpOvjt0rHPNLyCIeMtbizbir8U//inJ+zuB8= | ||||||
|  | github.com/konsorten/go-windows-terminal-sequences v1.0.1 h1:mweAR1A6xJ3oS2pRaGiHgQ4OO8tzTaLawm8vnODuwDk= | ||||||
|  | github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||||
| github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||||
| @@ -51,12 +51,14 @@ github.com/labstack/echo v0.0.0-20180219162101-7eec915044a1 h1:cOIt0LZKdfeirAfTP | |||||||
| github.com/labstack/echo v0.0.0-20180219162101-7eec915044a1/go.mod h1:0INS7j/VjnFxD4E2wkz67b8cVwCLbBmJyDaka6Cmk1s= | github.com/labstack/echo v0.0.0-20180219162101-7eec915044a1/go.mod h1:0INS7j/VjnFxD4E2wkz67b8cVwCLbBmJyDaka6Cmk1s= | ||||||
| github.com/labstack/gommon v0.2.1 h1:C+I4NYknueQncqKYZQ34kHsLZJVeB5KwPUhnO0nmbpU= | github.com/labstack/gommon v0.2.1 h1:C+I4NYknueQncqKYZQ34kHsLZJVeB5KwPUhnO0nmbpU= | ||||||
| github.com/labstack/gommon v0.2.1/go.mod h1:/tj9csK2iPSBvn+3NLM9e52usepMtrd5ilFYA+wQNJ4= | github.com/labstack/gommon v0.2.1/go.mod h1:/tj9csK2iPSBvn+3NLM9e52usepMtrd5ilFYA+wQNJ4= | ||||||
| github.com/lrstanley/girc v0.0.0-20180427160007-102f17f86306 h1:IqN61cmi7LM/IaYaP9a/KXFtHRS2a3+WHu8GhAXJT7c= | github.com/lrstanley/girc v0.0.0-20180913221000-0fb5b684054e h1:RpktB2igr6nS1EN7bCvjldAEfngrM5GyAbmOa4/cafU= | ||||||
| github.com/lrstanley/girc v0.0.0-20180427160007-102f17f86306/go.mod h1:7cRs1SIBfKQ7e3Tam6GKTILSNHzR862JD0JpINaZoJk= | github.com/lrstanley/girc v0.0.0-20180913221000-0fb5b684054e/go.mod h1:7cRs1SIBfKQ7e3Tam6GKTILSNHzR862JD0JpINaZoJk= | ||||||
|  | github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5 h1:AsEBgzv3DhuYHI/GiQh2HxvTP71HCCE9E/tzGUzGdtU= | ||||||
|  | github.com/lusis/go-slackbot v0.0.0-20180109053408-401027ccfef5/go.mod h1:c2mYKRyMb1BPkO5St0c/ps62L4S0W2NAkaTXj9qEI+0= | ||||||
|  | github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6 h1:iOAVXzZyXtW408TMYejlUPo6BIn92HmOacWtIfNyYns= | ||||||
|  | github.com/lusis/slack-test v0.0.0-20180109053238-3c758769bfa6/go.mod h1:sFlOUpQL1YcjhFVXhg1CG8ZASEs/Mf1oVb6H75JL/zg= | ||||||
| github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885 h1:HWxJJvF+QceKcql4r9PC93NtMEgEBfBxlQrZPvbcQvs= | github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885 h1:HWxJJvF+QceKcql4r9PC93NtMEgEBfBxlQrZPvbcQvs= | ||||||
| github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | github.com/magiconair/properties v0.0.0-20180217134545-2c9e95027885/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | ||||||
| github.com/matterbridge/discordgo v0.0.0-20180806170629-ef40ff5ba64f h1:9IIOO9Aznn8zJx3nokZ4U6nfuzWw5xAlygPvuRZMisQ= |  | ||||||
| github.com/matterbridge/discordgo v0.0.0-20180806170629-ef40ff5ba64f/go.mod h1:5QtN542bJn9FunZqYlIbleNtToxfLCVV9pW7m7Q42Fc= |  | ||||||
| github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91 h1:KzDEcy8eDbTx881giW8a6llsAck3e2bJvMyKvh1IK+k= | github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91 h1:KzDEcy8eDbTx881giW8a6llsAck3e2bJvMyKvh1IK+k= | ||||||
| github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91/go.mod h1:ECDRehsR9TYTKCAsRS8/wLeOk6UUqDydw47ln7wG41Q= | github.com/matterbridge/go-xmpp v0.0.0-20180529212104-cd19799fba91/go.mod h1:ECDRehsR9TYTKCAsRS8/wLeOk6UUqDydw47ln7wG41Q= | ||||||
| github.com/matterbridge/gomatrix v0.0.0-20171224233421-78ac6a1a0f5f h1:2eKh6Qi/sJ8bXvYMoyVfQxHgR8UcCDWjOmhV1oCstMU= | github.com/matterbridge/gomatrix v0.0.0-20171224233421-78ac6a1a0f5f h1:2eKh6Qi/sJ8bXvYMoyVfQxHgR8UcCDWjOmhV1oCstMU= | ||||||
| @@ -81,10 +83,8 @@ github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff h1:HLGD5/9UxxfEuO9Dt | |||||||
| github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff/go.mod h1:B8jLfIIPn2sKyWr0D7cL2v7tnrDD5z291s2Zypdu89E= | github.com/mrexodia/wray v0.0.0-20160318003008-78a2c1f284ff/go.mod h1:B8jLfIIPn2sKyWr0D7cL2v7tnrDD5z291s2Zypdu89E= | ||||||
| github.com/nicksnyder/go-i18n v1.4.0 h1:AgLl+Yq7kg5OYlzCgu9cKTZOyI4tD/NgukKqLqC8E+I= | github.com/nicksnyder/go-i18n v1.4.0 h1:AgLl+Yq7kg5OYlzCgu9cKTZOyI4tD/NgukKqLqC8E+I= | ||||||
| github.com/nicksnyder/go-i18n v1.4.0/go.mod h1:HrK7VCrbOvQoUAQ7Vpy7i87N7JZZZ7R2xBGjv0j365Q= | github.com/nicksnyder/go-i18n v1.4.0/go.mod h1:HrK7VCrbOvQoUAQ7Vpy7i87N7JZZZ7R2xBGjv0j365Q= | ||||||
| github.com/nlopes/slack v0.0.0-20180101221843-107290b5bbaf h1:M+xGhDxie/MqC+tzs+3ZHBSY4Wsv+fEkrpIMCKy8PTg= | github.com/nlopes/slack v0.4.0 h1:OVnHm7lv5gGT5gkcHsZAyw++oHVFihbjWbL3UceUpiA= | ||||||
| github.com/nlopes/slack v0.0.0-20180101221843-107290b5bbaf/go.mod h1:jVI4BBK3lSktibKahxBF74txcK2vyvkza1z/+rRnVAM= | github.com/nlopes/slack v0.4.0/go.mod h1:jVI4BBK3lSktibKahxBF74txcK2vyvkza1z/+rRnVAM= | ||||||
| github.com/nlopes/slack v0.3.1-0.20180805133408-21749ab136a8 h1:PSy8NkmkyldLmPPnNNw7mwfQFOHDqOI6bINpJ+/KV7Y= |  | ||||||
| github.com/nlopes/slack v0.3.1-0.20180805133408-21749ab136a8/go.mod h1:jVI4BBK3lSktibKahxBF74txcK2vyvkza1z/+rRnVAM= |  | ||||||
| github.com/onsi/ginkgo v1.6.0 h1:Ix8l273rp3QzYgXSR+c8d1fTG7UPgYkOSELPhiY/YGw= | github.com/onsi/ginkgo v1.6.0 h1:Ix8l273rp3QzYgXSR+c8d1fTG7UPgYkOSELPhiY/YGw= | ||||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||||
| github.com/onsi/gomega v1.4.1 h1:PZSj/UFNaVp3KxrzHOcS7oyuWA7LoOY/77yCTEFu21U= | github.com/onsi/gomega v1.4.1 h1:PZSj/UFNaVp3KxrzHOcS7oyuWA7LoOY/77yCTEFu21U= | ||||||
| @@ -103,16 +103,18 @@ github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZb | |||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
| github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a h1:UWKek6MK3K6/TpbsFcv+8rrO6rSc6KKSp2FbMOHWsq4= | github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a h1:UWKek6MK3K6/TpbsFcv+8rrO6rSc6KKSp2FbMOHWsq4= | ||||||
| github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ= | github.com/rs/xid v0.0.0-20180525034800-088c5cf1423a/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ= | ||||||
| github.com/russross/blackfriday v1.5.1 h1:B8ZN6pD4PVofmlDCDUdELeYrbsVIDM/bpjW3v3zgcRc= | github.com/russross/blackfriday v2.0.0+incompatible h1:cBXrhZNUf9C+La9/YpS+UHpUT8YD6Td9ZMSU9APFcsk= | ||||||
| github.com/russross/blackfriday v1.5.1/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= | github.com/russross/blackfriday v2.0.0+incompatible/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= | ||||||
| github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca h1:NugYot0LIVPxTvN8n+Kvkn6TrbMyxQiuvKdEwFdR9vI= | github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca h1:NugYot0LIVPxTvN8n+Kvkn6TrbMyxQiuvKdEwFdR9vI= | ||||||
| github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca/go.mod h1:uugorj2VCxiV1x+LzaIdVa9b4S4qGAcH6cbhh4qVxOU= | github.com/saintfish/chardet v0.0.0-20120816061221-3af4cd4741ca/go.mod h1:uugorj2VCxiV1x+LzaIdVa9b4S4qGAcH6cbhh4qVxOU= | ||||||
| github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1 h1:Lx3BlDGFElJt4u/zKc9A3BuGYbQAGlEFyPuUA3jeMD0= | github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1 h1:Lx3BlDGFElJt4u/zKc9A3BuGYbQAGlEFyPuUA3jeMD0= | ||||||
| github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1/go.mod h1:vt2jWY/3Qw1bIzle5thrJWucsLuuX9iUNnp20CqCciI= | github.com/shazow/rateio v0.0.0-20150116013248-e8e00881e5c1/go.mod h1:vt2jWY/3Qw1bIzle5thrJWucsLuuX9iUNnp20CqCciI= | ||||||
| github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991 h1:PQiUTDzUC5EUh0vNurK7KQS22zlKqLLOFn+K9nJXDQQ= | github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991 h1:PQiUTDzUC5EUh0vNurK7KQS22zlKqLLOFn+K9nJXDQQ= | ||||||
| github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991/go.mod h1:KwtnpMClmrXsHCKTbRui5xBUNt17n1GGrGhdiw2KcoY= | github.com/shazow/ssh-chat v0.0.0-20171012174035-2078e1381991/go.mod h1:KwtnpMClmrXsHCKTbRui5xBUNt17n1GGrGhdiw2KcoY= | ||||||
| github.com/sirupsen/logrus v0.0.0-20180213143110-8c0189d9f6bb h1:eKjx20EiekBRT2tjZ0XEdKpftfPJQwiavtFshwTyqf0= | github.com/shurcooL/sanitized_anchor_name v0.0.0-20170918181015-86672fcb3f95 h1:/vdW8Cb7EXrkqWGufVMES1OH2sU9gKVb2n9/1y5NMBY= | ||||||
| github.com/sirupsen/logrus v0.0.0-20180213143110-8c0189d9f6bb/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc= | github.com/shurcooL/sanitized_anchor_name v0.0.0-20170918181015-86672fcb3f95/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= | ||||||
|  | github.com/sirupsen/logrus v1.2.0 h1:juTguoYk5qI21pwyTXY3B3Y5cOTH3ZUyZCg1v/mihuo= | ||||||
|  | github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||||
| github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 h1:lXQ+j+KwZcbwrbgU0Rp4Eglg3EJLHbuZU3BbOqAGBmg= | github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9 h1:lXQ+j+KwZcbwrbgU0Rp4Eglg3EJLHbuZU3BbOqAGBmg= | ||||||
| github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= | github.com/smartystreets/assertions v0.0.0-20180803164922-886ec427f6b9/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= | ||||||
| github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a h1:JSvGDIbmil4Ui/dDdFBExb7/cmkNjyX5F97oglmvCDo= | github.com/smartystreets/goconvey v0.0.0-20180222194500-ef6db91d284a h1:JSvGDIbmil4Ui/dDdFBExb7/cmkNjyX5F97oglmvCDo= | ||||||
| @@ -127,8 +129,10 @@ github.com/spf13/pflag v0.0.0-20180220143236-ee5fd03fd6ac h1:+uzyQ0TQ3aKorQxsOjc | |||||||
| github.com/spf13/pflag v0.0.0-20180220143236-ee5fd03fd6ac/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | github.com/spf13/pflag v0.0.0-20180220143236-ee5fd03fd6ac/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | ||||||
| github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7 h1:Wj4cg2M6Um7j1N7yD/mxsdy1/wrsdjzVha2eWdOhti8= | github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7 h1:Wj4cg2M6Um7j1N7yD/mxsdy1/wrsdjzVha2eWdOhti8= | ||||||
| github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7/go.mod h1:A8kyI5cUJhb8N+3pkfONlcEcZbueH6nhAm0Fq7SrnBM= | github.com/spf13/viper v0.0.0-20171227194143-aafc9e6bc7b7/go.mod h1:A8kyI5cUJhb8N+3pkfONlcEcZbueH6nhAm0Fq7SrnBM= | ||||||
| github.com/stretchr/testify v0.0.0-20170714215325-05e8a0eda380 h1:MsolbevHkd4SpbeG4dHLHj6I9jzoohyNI6EK6JvR5hE= | github.com/stretchr/objx v0.1.1 h1:2vfRuCMp5sSVIDSqO8oNnWJq7mPa6KVP3iPIwFBuy8A= | ||||||
| github.com/stretchr/testify v0.0.0-20170714215325-05e8a0eda380/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
|  | github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= | ||||||
|  | github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||||
| github.com/technoweenie/multipartstreamer v1.0.1 h1:XRztA5MXiR1TIRHxH2uNxXxaIkKQDeX7m2XsSOlQEnM= | github.com/technoweenie/multipartstreamer v1.0.1 h1:XRztA5MXiR1TIRHxH2uNxXxaIkKQDeX7m2XsSOlQEnM= | ||||||
| github.com/technoweenie/multipartstreamer v1.0.1/go.mod h1:jNVxdtShOxzAsukZwTSw6MDx5eUJoiEBsSvzDU9uzog= | github.com/technoweenie/multipartstreamer v1.0.1/go.mod h1:jNVxdtShOxzAsukZwTSw6MDx5eUJoiEBsSvzDU9uzog= | ||||||
| github.com/valyala/bytebufferpool v0.0.0-20160817181652-e746df99fe4a h1:AOcehBWpFhYPYw0ioDTppQzgI8pAAahVCiMSKTp9rbo= | github.com/valyala/bytebufferpool v0.0.0-20160817181652-e746df99fe4a h1:AOcehBWpFhYPYw0ioDTppQzgI8pAAahVCiMSKTp9rbo= | ||||||
| @@ -139,24 +143,24 @@ github.com/x-cray/logrus-prefixed-formatter v0.5.2 h1:00txxvfBM9muc0jiLIEAkAcIMJ | |||||||
| github.com/x-cray/logrus-prefixed-formatter v0.5.2/go.mod h1:2duySbKsL6M18s5GU7VPsoEPHyzalCE06qoARUCeBBE= | github.com/x-cray/logrus-prefixed-formatter v0.5.2/go.mod h1:2duySbKsL6M18s5GU7VPsoEPHyzalCE06qoARUCeBBE= | ||||||
| github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6 h1:/WULP+6asFz569UbOwg87f3iDT7T+GF5/vjLmL51Pdk= | github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6 h1:/WULP+6asFz569UbOwg87f3iDT7T+GF5/vjLmL51Pdk= | ||||||
| github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6/go.mod h1:0MsIttMJIF/8Y7x0XjonJP7K99t3sR6bjj4m5S4JmqU= | github.com/zfjagann/golang-ring v0.0.0-20141111230621-17637388c9f6/go.mod h1:0MsIttMJIF/8Y7x0XjonJP7K99t3sR6bjj4m5S4JmqU= | ||||||
| golang.org/x/crypto v0.0.0-20180228161326-91a49db82a88 h1:jLkAo/qlT9whgCLYC5GAJ9kcKrv3Wj8VCc4N+KJ4wpw= | golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||||
| golang.org/x/crypto v0.0.0-20180228161326-91a49db82a88/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16 h1:y6ce7gCWtnH+m3dCjzQ1PCuwl28DDIc3VNnvY29DlIA= | ||||||
|  | golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||||
|  | golang.org/x/crypto v0.0.0-20181112202954-3d3f9f413869 h1:kkXA53yGe04D0adEYJwEVQjeBppL01Exg+fnMjfUraU= | ||||||
|  | golang.org/x/crypto v0.0.0-20181112202954-3d3f9f413869/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||||
| golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37 h1:BkNcmLtAVeWe9h5k0jt24CQgaG5vb4x/doFbAiEC/Ho= | golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37 h1:BkNcmLtAVeWe9h5k0jt24CQgaG5vb4x/doFbAiEC/Ho= | ||||||
| golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | golang.org/x/net v0.0.0-20180108090419-434ec0c7fe37/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f h1:wMNYb4v58l5UBM7MYRLPG6ZhfOqbKu7X5eyFl8ZhKvA= | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f h1:wMNYb4v58l5UBM7MYRLPG6ZhfOqbKu7X5eyFl8ZhKvA= | ||||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sys v0.0.0-20171130163741-8b4580aae2a0 h1:x4M4WCms+ErQg/4VyECbP2kSNcDJ6nLwqEGov1QPtqk= | golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/sys v0.0.0-20171130163741-8b4580aae2a0/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sys v0.0.0-20181116161606-93218def8b18 h1:Wh+XCfg3kNpjhdq2LXrsiOProjtQZKme5XUx7VcxwAw= | ||||||
|  | golang.org/x/sys v0.0.0-20181116161606-93218def8b18/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978 h1:WNm0tmiuBMW4FJRuXKWOqaQfmKptHs0n8nTCyG0ayjc= | golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978 h1:WNm0tmiuBMW4FJRuXKWOqaQfmKptHs0n8nTCyG0ayjc= | ||||||
| golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | golang.org/x/text v0.0.0-20180511172408-5c1cf69b5978/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
| gopkg.in/airbrake/gobrake.v2 v2.0.9 h1:7z2uVWwn7oVeeugY1DtlPAy5H+KYgB1KeKTnqjNatLo= |  | ||||||
| gopkg.in/airbrake/gobrake.v2 v2.0.9/go.mod h1:/h5ZAUhDkGaJfjzjKLSjv6zCL6O0LLBxU4K+aSYdM/U= |  | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= | gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= | ||||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||||
| gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.1 h1:4buh9nXkpqc7+GLzDFHei0jwoU9wCQYfVB5Kfo58Yz0= |  | ||||||
| gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.1/go.mod h1:Xk6kEKp8OKb+X14hQBKWaSkCsqBpgog8nAV2xsGOxlo= |  | ||||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||||
| gopkg.in/yaml.v2 v2.0.0-20160301204022-a83829b6f129 h1:RBgb9aPUbZ9nu66ecQNIBNsA7j3mB5h8PNDIfhPjaJg= | gopkg.in/yaml.v2 v2.0.0-20160301204022-a83829b6f129 h1:RBgb9aPUbZ9nu66ecQNIBNsA7j3mB5h8PNDIfhPjaJg= | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								img/matterbridge-notext.gif
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 70 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-add-scopes.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 270 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-app-page.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 170 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-create-app.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 282 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-create-bot.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 204 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-finished.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-install-app.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 73 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/slack-setup-invite-bot.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 62 KiB | 
| @@ -14,7 +14,7 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	version = "1.11.3" | 	version = "1.12.1" | ||||||
| 	githash string | 	githash string | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -44,7 +44,7 @@ func main() { | |||||||
| 		flog.Println("WARNING: THIS IS A DEVELOPMENT VERSION. Things may break.") | 		flog.Println("WARNING: THIS IS A DEVELOPMENT VERSION. Things may break.") | ||||||
| 	} | 	} | ||||||
| 	cfg := config.NewConfig(*flagConfig) | 	cfg := config.NewConfig(*flagConfig) | ||||||
| 	cfg.General.Debug = *flagDebug | 	cfg.BridgeValues().General.Debug = *flagDebug | ||||||
| 	r, err := gateway.NewRouter(cfg) | 	r, err := gateway.NewRouter(cfg) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		flog.Fatalf("Starting gateway failed: %s", err) | 		flog.Fatalf("Starting gateway failed: %s", err) | ||||||
|   | |||||||
| @@ -129,12 +129,8 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #The string "{NOPINGNICK}" (case sensitive) will be replaced by the actual nick / username, but with a ZWSP inside the nick, so the irc user with the same nick won't get pinged. See https://github.com/42wim/matterbridge/issues/175 for more information | #The string "{NOPINGNICK}" (case sensitive) will be replaced by the actual nick / username, but with a ZWSP inside the nick, so the irc user with the same nick won't get pinged. See https://github.com/42wim/matterbridge/issues/175 for more information | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -227,11 +223,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -311,11 +303,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}/{BRIDGE}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}/{BRIDGE}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -455,11 +443,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -534,11 +518,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -656,11 +636,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -683,6 +659,18 @@ StripNick=false | |||||||
| #OPTIONAL (default false) | #OPTIONAL (default false) | ||||||
| ShowTopicChange=false | ShowTopicChange=false | ||||||
|  |  | ||||||
|  | #Opportunistically preserve threaded replies between Slack channels. | ||||||
|  | #This only works if the parent message is still in the cache. | ||||||
|  | #Cache is flushed between restarts. | ||||||
|  | #OPTIONAL (default false) | ||||||
|  | PreserveThreading=false | ||||||
|  |  | ||||||
|  | #Enable showing "user_typing" events from across gateway when available. | ||||||
|  | #Protip: Set your bot/user's "Full Name" to be "Someone (over chat bridge)", | ||||||
|  | #and so the message will say "Someone (over chat bridge) is typing". | ||||||
|  | #OPTIONAL (default false) | ||||||
|  | ShowUserTyping=false | ||||||
|  |  | ||||||
| ################################################################### | ################################################################### | ||||||
| #discord section | #discord section | ||||||
| ################################################################### | ################################################################### | ||||||
| @@ -759,11 +747,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -866,16 +850,11 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| # | # | ||||||
| #WARNING: if you have set MessageFormat="HTML" be sure that this format matches the guidelines | #WARNING: if you have set MessageFormat="HTML" be sure that this format matches the guidelines | ||||||
| #on https://core.telegram.org/bots/api#html-style otherwise the message will not go through to | #on https://core.telegram.org/bots/api#html-style otherwise the message will not go through to | ||||||
| #telegram! eg <{NICK}> should be <{NICK}> | #telegram! eg <{NICK}> should be <{NICK}> | ||||||
| # |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -969,11 +948,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -1059,11 +1034,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -1143,11 +1114,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -1227,11 +1194,7 @@ ReplaceNicks=[ ["user--","user"] ] | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| #Enable to show users joins/parts from other bridges  | #Enable to show users joins/parts from other bridges  | ||||||
| @@ -1263,6 +1226,7 @@ ShowTopicChange=false | |||||||
| BindAddress="127.0.0.1:4242" | BindAddress="127.0.0.1:4242" | ||||||
|  |  | ||||||
| #Amount of messages to keep in memory | #Amount of messages to keep in memory | ||||||
|  | #OPTIONAL (library default 10) | ||||||
| Buffer=1000 | Buffer=1000 | ||||||
|  |  | ||||||
| #Bearer token used for authentication | #Bearer token used for authentication | ||||||
| @@ -1275,11 +1239,7 @@ Token="mytoken" | |||||||
| Label="" | Label="" | ||||||
|  |  | ||||||
| #RemoteNickFormat defines how remote users appear on this bridge  | #RemoteNickFormat defines how remote users appear on this bridge  | ||||||
| #The string "{NICK}" (case sensitive) will be replaced by the actual nick / username. | #See [general] config section for default options | ||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge |  | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge |  | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge |  | ||||||
| #OPTIONAL (default empty) |  | ||||||
| RemoteNickFormat="{NICK}" | RemoteNickFormat="{NICK}" | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -1298,6 +1258,8 @@ RemoteNickFormat="{NICK}" | |||||||
| #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge | #The string "{BRIDGE}" (case sensitive) will be replaced by the sending bridge | ||||||
| #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge | #The string "{LABEL}" (case sensitive) will be replaced by label= field of the sending bridge | ||||||
| #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge | #The string "{PROTOCOL}" (case sensitive) will be replaced by the protocol used by the bridge | ||||||
|  | #The string "{GATEWAY}" (case sensitive) will be replaced by the origin gateway name that is replicating the message. | ||||||
|  | #The string "{CHANNEL}" (case sensitive) will be replaced by the origin channel name used by the bridge | ||||||
| #OPTIONAL (default empty) | #OPTIONAL (default empty) | ||||||
| RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | RemoteNickFormat="[{PROTOCOL}] <{NICK}> " | ||||||
|  |  | ||||||
| @@ -1427,7 +1389,7 @@ enable=true | |||||||
|  |  | ||||||
|         #OPTIONAL - webhookurl only works for discord (it needs a different URL for each cahnnel) |         #OPTIONAL - webhookurl only works for discord (it needs a different URL for each cahnnel) | ||||||
|         [gateway.inout.options] |         [gateway.inout.options] | ||||||
|         webhookurl=""https://discordapp.com/api/webhooks/123456789123456789/C9WPqExYWONPDZabcdef-def1434FGFjstasJX9pYht73y" |         webhookurl="https://discordapp.com/api/webhooks/123456789123456789/C9WPqExYWONPDZabcdef-def1434FGFjstasJX9pYht73y" | ||||||
|  |  | ||||||
|     #API example |     #API example | ||||||
|     #[[gateway.inout]] |     #[[gateway.inout]] | ||||||
|   | |||||||
| @@ -13,19 +13,20 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	prefixed "github.com/matterbridge/logrus-prefixed-formatter" |  | ||||||
| 	log "github.com/sirupsen/logrus" |  | ||||||
|  |  | ||||||
| 	"github.com/gorilla/websocket" | 	"github.com/gorilla/websocket" | ||||||
| 	"github.com/hashicorp/golang-lru" | 	"github.com/hashicorp/golang-lru" | ||||||
| 	"github.com/jpillora/backoff" | 	"github.com/jpillora/backoff" | ||||||
|  | 	prefixed "github.com/matterbridge/logrus-prefixed-formatter" | ||||||
| 	"github.com/mattermost/platform/model" | 	"github.com/mattermost/platform/model" | ||||||
|  | 	log "github.com/sirupsen/logrus" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Credentials struct { | type Credentials struct { | ||||||
| 	Login         string | 	Login         string | ||||||
| 	Team          string | 	Team          string | ||||||
| 	Pass          string | 	Pass          string | ||||||
|  | 	Token         string | ||||||
|  | 	CookieToken   bool | ||||||
| 	Server        string | 	Server        string | ||||||
| 	NoTLS         bool | 	NoTLS         bool | ||||||
| 	SkipTLSVerify bool | 	SkipTLSVerify bool | ||||||
| @@ -42,6 +43,7 @@ type Message struct { | |||||||
| 	UserID   string | 	UserID   string | ||||||
| } | } | ||||||
|  |  | ||||||
|  | //nolint:golint | ||||||
| type Team struct { | type Team struct { | ||||||
| 	Team         *model.Team | 	Team         *model.Team | ||||||
| 	Id           string | 	Id           string | ||||||
| @@ -117,6 +119,23 @@ func (m *MMClient) Login() error { | |||||||
| 	m.Client.HttpClient.Transport = &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, Proxy: http.ProxyFromEnvironment} | 	m.Client.HttpClient.Transport = &http.Transport{TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, Proxy: http.ProxyFromEnvironment} | ||||||
| 	m.Client.HttpClient.Timeout = time.Second * 10 | 	m.Client.HttpClient.Timeout = time.Second * 10 | ||||||
|  |  | ||||||
|  | 	if strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN) { | ||||||
|  | 		token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=") | ||||||
|  | 		if len(token) != 2 { | ||||||
|  | 			return errors.New("incorrect MMAUTHTOKEN. valid input is MMAUTHTOKEN=yourtoken") | ||||||
|  | 		} | ||||||
|  | 		m.Credentials.Token = token[1] | ||||||
|  | 		m.Credentials.CookieToken = true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if strings.Contains(m.Credentials.Pass, "token=") { | ||||||
|  | 		token := strings.Split(m.Credentials.Pass, "token=") | ||||||
|  | 		if len(token) != 2 { | ||||||
|  | 			return errors.New("incorrect personal token. valid input is token=yourtoken") | ||||||
|  | 		} | ||||||
|  | 		m.Credentials.Token = token[1] | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	for { | 	for { | ||||||
| 		d := b.Duration() | 		d := b.Duration() | ||||||
| 		// bogus call to get the serverversion | 		// bogus call to get the serverversion | ||||||
| @@ -144,22 +163,22 @@ func (m *MMClient) Login() error { | |||||||
| 	var logmsg = "trying login" | 	var logmsg = "trying login" | ||||||
| 	for { | 	for { | ||||||
| 		m.log.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server) | 		m.log.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server) | ||||||
| 		if strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN) { | 		if m.Credentials.Token != "" { | ||||||
| 			m.log.Debugf(logmsg + " with token") |  | ||||||
| 			token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=") |  | ||||||
| 			if len(token) != 2 { |  | ||||||
| 				return errors.New("incorrect MMAUTHTOKEN. valid input is MMAUTHTOKEN=yourtoken") |  | ||||||
| 			} |  | ||||||
| 			m.Client.HttpClient.Jar = m.createCookieJar(token[1]) |  | ||||||
| 			m.Client.AuthToken = token[1] |  | ||||||
| 			m.Client.AuthType = model.HEADER_BEARER | 			m.Client.AuthType = model.HEADER_BEARER | ||||||
|  | 			m.Client.AuthToken = m.Credentials.Token | ||||||
|  | 			if m.Credentials.CookieToken { | ||||||
|  | 				m.log.Debugf(logmsg + " with cookie (MMAUTH) token") | ||||||
|  | 				m.Client.HttpClient.Jar = m.createCookieJar(m.Credentials.Token) | ||||||
|  | 			} else { | ||||||
|  | 				m.log.Debugf(logmsg + " with personal token") | ||||||
|  | 			} | ||||||
| 			m.User, resp = m.Client.GetMe("") | 			m.User, resp = m.Client.GetMe("") | ||||||
| 			if resp.Error != nil { | 			if resp.Error != nil { | ||||||
| 				return resp.Error | 				return resp.Error | ||||||
| 			} | 			} | ||||||
| 			if m.User == nil { | 			if m.User == nil { | ||||||
| 				m.log.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass) | 				m.log.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass) | ||||||
| 				return errors.New("invalid " + model.SESSION_COOKIE_TOKEN) | 				return errors.New("invalid token") | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			m.User, resp = m.Client.Login(m.Credentials.Login, m.Credentials.Pass) | 			m.User, resp = m.Client.Login(m.Credentials.Login, m.Credentials.Pass) | ||||||
| @@ -315,6 +334,8 @@ func (m *MMClient) parseMessage(rmsg *Message) { | |||||||
| 		if _, ok := user["id"].(string); ok { | 		if _, ok := user["id"].(string); ok { | ||||||
| 			m.UpdateUser(user["id"].(string)) | 			m.UpdateUser(user["id"].(string)) | ||||||
| 		} | 		} | ||||||
|  | 	case "group_added": | ||||||
|  | 		m.UpdateChannels() | ||||||
| 		/* | 		/* | ||||||
| 			case model.ACTION_USER_REMOVED: | 			case model.ACTION_USER_REMOVED: | ||||||
| 				m.handleWsActionUserRemoved(&rmsg) | 				m.handleWsActionUserRemoved(&rmsg) | ||||||
| @@ -344,7 +365,8 @@ func (m *MMClient) parseActionPost(rmsg *Message) { | |||||||
| 	data := model.PostFromJson(strings.NewReader(rmsg.Raw.Data["post"].(string))) | 	data := model.PostFromJson(strings.NewReader(rmsg.Raw.Data["post"].(string))) | ||||||
| 	// we don't have the user, refresh the userlist | 	// we don't have the user, refresh the userlist | ||||||
| 	if m.GetUser(data.UserId) == nil { | 	if m.GetUser(data.UserId) == nil { | ||||||
| 		m.log.Infof("User %s is not known, ignoring message %s", data) | 		m.log.Infof("User '%v' is not known, ignoring message '%#v'", | ||||||
|  | 			data.UserId, data) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	rmsg.Username = m.GetUserName(data.UserId) | 	rmsg.Username = m.GetUserName(data.UserId) | ||||||
| @@ -402,7 +424,7 @@ func (m *MMClient) UpdateChannels() error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetChannelName(channelId string) string { | func (m *MMClient) GetChannelName(channelId string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	for _, t := range m.OtherTeams { | 	for _, t := range m.OtherTeams { | ||||||
| @@ -412,6 +434,11 @@ func (m *MMClient) GetChannelName(channelId string) string { | |||||||
| 		if t.Channels != nil { | 		if t.Channels != nil { | ||||||
| 			for _, channel := range t.Channels { | 			for _, channel := range t.Channels { | ||||||
| 				if channel.Id == channelId { | 				if channel.Id == channelId { | ||||||
|  | 					if channel.Type == model.CHANNEL_GROUP { | ||||||
|  | 						res := strings.Replace(channel.DisplayName, ", ", "-", -1) | ||||||
|  | 						res = strings.Replace(res, " ", "_", -1) | ||||||
|  | 						return res | ||||||
|  | 					} | ||||||
| 					return channel.Name | 					return channel.Name | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @@ -419,6 +446,11 @@ func (m *MMClient) GetChannelName(channelId string) string { | |||||||
| 		if t.MoreChannels != nil { | 		if t.MoreChannels != nil { | ||||||
| 			for _, channel := range t.MoreChannels { | 			for _, channel := range t.MoreChannels { | ||||||
| 				if channel.Id == channelId { | 				if channel.Id == channelId { | ||||||
|  | 					if channel.Type == model.CHANNEL_GROUP { | ||||||
|  | 						res := strings.Replace(channel.DisplayName, ", ", "-", -1) | ||||||
|  | 						res = strings.Replace(res, " ", "_", -1) | ||||||
|  | 						return res | ||||||
|  | 					} | ||||||
| 					return channel.Name | 					return channel.Name | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @@ -427,12 +459,24 @@ func (m *MMClient) GetChannelName(channelId string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetChannelId(name string, teamId string) string { | func (m *MMClient) GetChannelId(name string, teamId string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	if teamId == "" { | 	if teamId == "" { | ||||||
| 		teamId = m.Team.Id | 		for _, t := range m.OtherTeams { | ||||||
|  | 			for _, channel := range append(t.Channels, t.MoreChannels...) { | ||||||
|  | 				if channel.Type == model.CHANNEL_GROUP { | ||||||
|  | 					res := strings.Replace(channel.DisplayName, ", ", "-", -1) | ||||||
|  | 					res = strings.Replace(res, " ", "_", -1) | ||||||
|  | 					if res == name { | ||||||
|  | 						return channel.Id | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, t := range m.OtherTeams { | 	for _, t := range m.OtherTeams { | ||||||
| 		if t.Id == teamId { | 		if t.Id == teamId { | ||||||
| 			for _, channel := range append(t.Channels, t.MoreChannels...) { | 			for _, channel := range append(t.Channels, t.MoreChannels...) { | ||||||
| @@ -445,7 +489,7 @@ func (m *MMClient) GetChannelId(name string, teamId string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetChannelTeamId(id string) string { | func (m *MMClient) GetChannelTeamId(id string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	for _, t := range append(m.OtherTeams, m.Team) { | 	for _, t := range append(m.OtherTeams, m.Team) { | ||||||
| @@ -458,7 +502,7 @@ func (m *MMClient) GetChannelTeamId(id string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetChannelHeader(channelId string) string { | func (m *MMClient) GetChannelHeader(channelId string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	for _, t := range m.OtherTeams { | 	for _, t := range m.OtherTeams { | ||||||
| @@ -472,7 +516,7 @@ func (m *MMClient) GetChannelHeader(channelId string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) PostMessage(channelId string, text string) (string, error) { | func (m *MMClient) PostMessage(channelId string, text string) (string, error) { //nolint:golint | ||||||
| 	post := &model.Post{ChannelId: channelId, Message: text} | 	post := &model.Post{ChannelId: channelId, Message: text} | ||||||
| 	res, resp := m.Client.CreatePost(post) | 	res, resp := m.Client.CreatePost(post) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| @@ -481,7 +525,7 @@ func (m *MMClient) PostMessage(channelId string, text string) (string, error) { | |||||||
| 	return res.Id, nil | 	return res.Id, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) PostMessageWithFiles(channelId string, text string, fileIds []string) (string, error) { | func (m *MMClient) PostMessageWithFiles(channelId string, text string, fileIds []string) (string, error) { //nolint:golint | ||||||
| 	post := &model.Post{ChannelId: channelId, Message: text, FileIds: fileIds} | 	post := &model.Post{ChannelId: channelId, Message: text, FileIds: fileIds} | ||||||
| 	res, resp := m.Client.CreatePost(post) | 	res, resp := m.Client.CreatePost(post) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| @@ -490,7 +534,7 @@ func (m *MMClient) PostMessageWithFiles(channelId string, text string, fileIds [ | |||||||
| 	return res.Id, nil | 	return res.Id, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) EditMessage(postId string, text string) (string, error) { | func (m *MMClient) EditMessage(postId string, text string) (string, error) { //nolint:golint | ||||||
| 	post := &model.Post{Message: text} | 	post := &model.Post{Message: text} | ||||||
| 	res, resp := m.Client.UpdatePost(postId, post) | 	res, resp := m.Client.UpdatePost(postId, post) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| @@ -499,7 +543,7 @@ func (m *MMClient) EditMessage(postId string, text string) (string, error) { | |||||||
| 	return res.Id, nil | 	return res.Id, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) DeleteMessage(postId string) error { | func (m *MMClient) DeleteMessage(postId string) error { //nolint:golint | ||||||
| 	_, resp := m.Client.DeletePost(postId) | 	_, resp := m.Client.DeletePost(postId) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return resp.Error | 		return resp.Error | ||||||
| @@ -507,7 +551,7 @@ func (m *MMClient) DeleteMessage(postId string) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) JoinChannel(channelId string) error { | func (m *MMClient) JoinChannel(channelId string) error { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	for _, c := range m.Team.Channels { | 	for _, c := range m.Team.Channels { | ||||||
| @@ -524,7 +568,7 @@ func (m *MMClient) JoinChannel(channelId string) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetPostsSince(channelId string, time int64) *model.PostList { | func (m *MMClient) GetPostsSince(channelId string, time int64) *model.PostList { //nolint:golint | ||||||
| 	res, resp := m.Client.GetPostsSince(channelId, time) | 	res, resp := m.Client.GetPostsSince(channelId, time) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -540,7 +584,7 @@ func (m *MMClient) SearchPosts(query string) *model.PostList { | |||||||
| 	return res | 	return res | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetPosts(channelId string, limit int) *model.PostList { | func (m *MMClient) GetPosts(channelId string, limit int) *model.PostList { //nolint:golint | ||||||
| 	res, resp := m.Client.GetPostsForChannel(channelId, 0, limit, "") | 	res, resp := m.Client.GetPostsForChannel(channelId, 0, limit, "") | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -587,7 +631,7 @@ func (m *MMClient) GetFileLinks(filenames []string) []string { | |||||||
| 	return output | 	return output | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UpdateChannelHeader(channelId string, header string) { | func (m *MMClient) UpdateChannelHeader(channelId string, header string) { //nolint:golint | ||||||
| 	channel := &model.Channel{Id: channelId, Header: header} | 	channel := &model.Channel{Id: channelId, Header: header} | ||||||
| 	m.log.Debugf("updating channelheader %#v, %#v", channelId, header) | 	m.log.Debugf("updating channelheader %#v, %#v", channelId, header) | ||||||
| 	_, resp := m.Client.UpdateChannel(channel) | 	_, resp := m.Client.UpdateChannel(channel) | ||||||
| @@ -596,13 +640,15 @@ func (m *MMClient) UpdateChannelHeader(channelId string, header string) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UpdateLastViewed(channelId string) { | func (m *MMClient) UpdateLastViewed(channelId string) error { //nolint:golint | ||||||
| 	m.log.Debugf("posting lastview %#v", channelId) | 	m.log.Debugf("posting lastview %#v", channelId) | ||||||
| 	view := &model.ChannelView{ChannelId: channelId} | 	view := &model.ChannelView{ChannelId: channelId} | ||||||
| 	_, resp := m.Client.ViewChannel(m.User.Id, view) | 	_, resp := m.Client.ViewChannel(m.User.Id, view) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		m.log.Errorf("ChannelView update for %s failed: %s", channelId, resp.Error) | 		m.log.Errorf("ChannelView update for %s failed: %s", channelId, resp.Error) | ||||||
|  | 		return resp.Error | ||||||
| 	} | 	} | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UpdateUserNick(nick string) error { | func (m *MMClient) UpdateUserNick(nick string) error { | ||||||
| @@ -615,7 +661,7 @@ func (m *MMClient) UpdateUserNick(nick string) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UsernamesInChannel(channelId string) []string { | func (m *MMClient) UsernamesInChannel(channelId string) []string { //nolint:golint | ||||||
| 	res, resp := m.Client.GetChannelMembers(channelId, 0, 50000, "") | 	res, resp := m.Client.GetChannelMembers(channelId, 0, 50000, "") | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		m.log.Errorf("UsernamesInChannel(%s) failed: %s", channelId, resp.Error) | 		m.log.Errorf("UsernamesInChannel(%s) failed: %s", channelId, resp.Error) | ||||||
| @@ -645,7 +691,11 @@ func (m *MMClient) createCookieJar(token string) *cookiejar.Jar { | |||||||
| } | } | ||||||
|  |  | ||||||
| // SendDirectMessage sends a direct message to specified user | // SendDirectMessage sends a direct message to specified user | ||||||
| func (m *MMClient) SendDirectMessage(toUserId string, msg string) { | func (m *MMClient) SendDirectMessage(toUserId string, msg string) { //nolint:golint | ||||||
|  | 	m.SendDirectMessageProps(toUserId, msg, nil) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (m *MMClient) SendDirectMessageProps(toUserId string, msg string, props map[string]interface{}) { //nolint:golint | ||||||
| 	m.log.Debugf("SendDirectMessage to %s, msg %s", toUserId, msg) | 	m.log.Debugf("SendDirectMessage to %s, msg %s", toUserId, msg) | ||||||
| 	// create DM channel (only happens on first message) | 	// create DM channel (only happens on first message) | ||||||
| 	_, resp := m.Client.CreateDirectChannel(m.User.Id, toUserId) | 	_, resp := m.Client.CreateDirectChannel(m.User.Id, toUserId) | ||||||
| @@ -660,12 +710,12 @@ func (m *MMClient) SendDirectMessage(toUserId string, msg string) { | |||||||
|  |  | ||||||
| 	// build & send the message | 	// build & send the message | ||||||
| 	msg = strings.Replace(msg, "\r", "", -1) | 	msg = strings.Replace(msg, "\r", "", -1) | ||||||
| 	post := &model.Post{ChannelId: m.GetChannelId(channelName, ""), Message: msg} | 	post := &model.Post{ChannelId: m.GetChannelId(channelName, m.Team.Id), Message: msg, Props: props} | ||||||
| 	m.Client.CreatePost(post) | 	m.Client.CreatePost(post) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetTeamName returns the name of the specified teamId | // GetTeamName returns the name of the specified teamId | ||||||
| func (m *MMClient) GetTeamName(teamId string) string { | func (m *MMClient) GetTeamName(teamId string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	for _, t := range m.OtherTeams { | 	for _, t := range m.OtherTeams { | ||||||
| @@ -703,7 +753,7 @@ func (m *MMClient) GetMoreChannels() []*model.Channel { | |||||||
| } | } | ||||||
|  |  | ||||||
| // GetTeamFromChannel returns teamId belonging to channel (DM channels have no teamId). | // GetTeamFromChannel returns teamId belonging to channel (DM channels have no teamId). | ||||||
| func (m *MMClient) GetTeamFromChannel(channelId string) string { | func (m *MMClient) GetTeamFromChannel(channelId string) string { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	var channels []*model.Channel | 	var channels []*model.Channel | ||||||
| @@ -714,14 +764,18 @@ func (m *MMClient) GetTeamFromChannel(channelId string) string { | |||||||
| 		} | 		} | ||||||
| 		for _, c := range channels { | 		for _, c := range channels { | ||||||
| 			if c.Id == channelId { | 			if c.Id == channelId { | ||||||
|  | 				if c.Type == model.CHANNEL_GROUP { | ||||||
|  | 					return "G" | ||||||
|  | 				} | ||||||
| 				return t.Id | 				return t.Id | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 		channels = nil | ||||||
| 	} | 	} | ||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetLastViewedAt(channelId string) int64 { | func (m *MMClient) GetLastViewedAt(channelId string) int64 { //nolint:golint | ||||||
| 	m.RLock() | 	m.RLock() | ||||||
| 	defer m.RUnlock() | 	defer m.RUnlock() | ||||||
| 	res, resp := m.Client.GetChannelMember(channelId, m.User.Id, "") | 	res, resp := m.Client.GetChannelMember(channelId, m.User.Id, "") | ||||||
| @@ -741,7 +795,7 @@ func (m *MMClient) GetUsers() map[string]*model.User { | |||||||
| 	return users | 	return users | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetUser(userId string) *model.User { | func (m *MMClient) GetUser(userId string) *model.User { //nolint:golint | ||||||
| 	m.Lock() | 	m.Lock() | ||||||
| 	defer m.Unlock() | 	defer m.Unlock() | ||||||
| 	_, ok := m.Users[userId] | 	_, ok := m.Users[userId] | ||||||
| @@ -755,7 +809,7 @@ func (m *MMClient) GetUser(userId string) *model.User { | |||||||
| 	return m.Users[userId] | 	return m.Users[userId] | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UpdateUser(userId string) { | func (m *MMClient) UpdateUser(userId string) { //nolint:golint | ||||||
| 	m.Lock() | 	m.Lock() | ||||||
| 	defer m.Unlock() | 	defer m.Unlock() | ||||||
| 	res, resp := m.Client.GetUser(userId, "") | 	res, resp := m.Client.GetUser(userId, "") | ||||||
| @@ -765,7 +819,7 @@ func (m *MMClient) UpdateUser(userId string) { | |||||||
| 	m.Users[userId] = res | 	m.Users[userId] = res | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetUserName(userId string) string { | func (m *MMClient) GetUserName(userId string) string { //nolint:golint | ||||||
| 	user := m.GetUser(userId) | 	user := m.GetUser(userId) | ||||||
| 	if user != nil { | 	if user != nil { | ||||||
| 		return user.Username | 		return user.Username | ||||||
| @@ -773,7 +827,7 @@ func (m *MMClient) GetUserName(userId string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetNickName(userId string) string { | func (m *MMClient) GetNickName(userId string) string { //nolint:golint | ||||||
| 	user := m.GetUser(userId) | 	user := m.GetUser(userId) | ||||||
| 	if user != nil { | 	if user != nil { | ||||||
| 		return user.Nickname | 		return user.Nickname | ||||||
| @@ -781,7 +835,7 @@ func (m *MMClient) GetNickName(userId string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetStatus(userId string) string { | func (m *MMClient) GetStatus(userId string) string { //nolint:golint | ||||||
| 	res, resp := m.Client.GetUserStatus(userId, "") | 	res, resp := m.Client.GetUserStatus(userId, "") | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return "" | 		return "" | ||||||
| @@ -795,7 +849,7 @@ func (m *MMClient) GetStatus(userId string) string { | |||||||
| 	return "offline" | 	return "offline" | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UpdateStatus(userId string, status string) error { | func (m *MMClient) UpdateStatus(userId string, status string) error { //nolint:golint | ||||||
| 	_, resp := m.Client.UpdateUserStatus(userId, &model.Status{Status: status}) | 	_, resp := m.Client.UpdateUserStatus(userId, &model.Status{Status: status}) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return resp.Error | 		return resp.Error | ||||||
| @@ -825,11 +879,11 @@ func (m *MMClient) GetStatuses() map[string]string { | |||||||
| 	return statuses | 	return statuses | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) GetTeamId() string { | func (m *MMClient) GetTeamId() string { //nolint:golint | ||||||
| 	return m.Team.Id | 	return m.Team.Id | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *MMClient) UploadFile(data []byte, channelId string, filename string) (string, error) { | func (m *MMClient) UploadFile(data []byte, channelId string, filename string) (string, error) { //nolint:golint | ||||||
| 	f, resp := m.Client.UploadFile(data, channelId, filename) | 	f, resp := m.Client.UploadFile(data, channelId, filename) | ||||||
| 	if resp.Error != nil { | 	if resp.Error != nil { | ||||||
| 		return "", resp.Error | 		return "", resp.Error | ||||||
| @@ -843,14 +897,13 @@ func (m *MMClient) StatusLoop() { | |||||||
| 	if m.OnWsConnect != nil { | 	if m.OnWsConnect != nil { | ||||||
| 		m.OnWsConnect() | 		m.OnWsConnect() | ||||||
| 	} | 	} | ||||||
| 	m.log.Debug("StatusLoop:", m.OnWsConnect) | 	m.log.Debug("StatusLoop:", m.OnWsConnect != nil) | ||||||
| 	for { | 	for { | ||||||
| 		if m.WsQuit { | 		if m.WsQuit { | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		if m.WsConnected { | 		if m.WsConnected { | ||||||
| 			m.log.Debug("WS PING") | 			m.checkAlive() | ||||||
| 			m.sendWSRequest("ping", nil) |  | ||||||
| 			select { | 			select { | ||||||
| 			case <-m.WsPingChan: | 			case <-m.WsPingChan: | ||||||
| 				m.log.Debug("WS PONG received") | 				m.log.Debug("WS PONG received") | ||||||
| @@ -925,6 +978,16 @@ func (m *MMClient) initUser() error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (m *MMClient) checkAlive() error { | ||||||
|  | 	// check if session still is valid | ||||||
|  | 	_, resp := m.Client.GetMe("") | ||||||
|  | 	if resp.Error != nil { | ||||||
|  | 		return resp.Error | ||||||
|  | 	} | ||||||
|  | 	m.log.Debug("WS PING") | ||||||
|  | 	return m.sendWSRequest("ping", nil) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (m *MMClient) sendWSRequest(action string, data map[string]interface{}) error { | func (m *MMClient) sendWSRequest(action string, data map[string]interface{}) error { | ||||||
| 	req := &model.WebSocketRequest{} | 	req := &model.WebSocketRequest{} | ||||||
| 	req.Seq = m.WsSequence | 	req.Seq = m.WsSequence | ||||||
|   | |||||||
| @@ -41,9 +41,9 @@ type IMessage struct { | |||||||
| 	Timestamp   string `schema:"timestamp"` | 	Timestamp   string `schema:"timestamp"` | ||||||
| 	UserID      string `schema:"user_id"` | 	UserID      string `schema:"user_id"` | ||||||
| 	UserName    string `schema:"user_name"` | 	UserName    string `schema:"user_name"` | ||||||
| 	PostId      string `schema:"post_id"` | 	PostId      string `schema:"post_id"` //nolint:golint | ||||||
| 	RawText     string `schema:"raw_text"` | 	RawText     string `schema:"raw_text"` | ||||||
| 	ServiceId   string `schema:"service_id"` | 	ServiceId   string `schema:"service_id"` //nolint:golint | ||||||
| 	Text        string `schema:"text"` | 	Text        string `schema:"text"` | ||||||
| 	TriggerWord string `schema:"trigger_word"` | 	TriggerWord string `schema:"trigger_word"` | ||||||
| 	FileIDs     string `schema:"file_ids"` | 	FileIDs     string `schema:"file_ids"` | ||||||
| @@ -51,7 +51,8 @@ type IMessage struct { | |||||||
|  |  | ||||||
| // Client for Mattermost. | // Client for Mattermost. | ||||||
| type Client struct { | type Client struct { | ||||||
| 	Url        string // URL for incoming webhooks on mattermost. | 	// URL for incoming webhooks on mattermost. | ||||||
|  | 	Url        string // nolint:golint | ||||||
| 	In         chan IMessage | 	In         chan IMessage | ||||||
| 	Out        chan OMessage | 	Out        chan OMessage | ||||||
| 	httpclient *http.Client | 	httpclient *http.Client | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								vendor/github.com/BurntSushi/toml/.gitignore
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,5 +0,0 @@ | |||||||
| TAGS |  | ||||||
| tags |  | ||||||
| .*.swp |  | ||||||
| tomlcheck/tomlcheck |  | ||||||
| toml.test |  | ||||||
							
								
								
									
										15
									
								
								vendor/github.com/BurntSushi/toml/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,15 +0,0 @@ | |||||||
| language: go |  | ||||||
| go: |  | ||||||
|   - 1.1 |  | ||||||
|   - 1.2 |  | ||||||
|   - 1.3 |  | ||||||
|   - 1.4 |  | ||||||
|   - 1.5 |  | ||||||
|   - 1.6 |  | ||||||
|   - tip |  | ||||||
| install: |  | ||||||
|   - go install ./... |  | ||||||
|   - go get github.com/BurntSushi/toml-test |  | ||||||
| script: |  | ||||||
|   - export PATH="$PATH:$HOME/gopath/bin" |  | ||||||
|   - make test |  | ||||||
							
								
								
									
										3
									
								
								vendor/github.com/BurntSushi/toml/COMPATIBLE
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +0,0 @@ | |||||||
| Compatible with TOML version |  | ||||||
| [v0.4.0](https://github.com/toml-lang/toml/blob/v0.4.0/versions/en/toml-v0.4.0.md) |  | ||||||
|  |  | ||||||
							
								
								
									
										14
									
								
								vendor/github.com/BurntSushi/toml/COPYING
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,14 +0,0 @@ | |||||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE |  | ||||||
|                     Version 2, December 2004 |  | ||||||
|  |  | ||||||
|  Copyright (C) 2004 Sam Hocevar <sam@hocevar.net> |  | ||||||
|  |  | ||||||
|  Everyone is permitted to copy and distribute verbatim or modified |  | ||||||
|  copies of this license document, and changing it is allowed as long |  | ||||||
|  as the name is changed. |  | ||||||
|  |  | ||||||
|             DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE |  | ||||||
|    TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION |  | ||||||
|  |  | ||||||
|   0. You just DO WHAT THE FUCK YOU WANT TO. |  | ||||||
|  |  | ||||||
							
								
								
									
										19
									
								
								vendor/github.com/BurntSushi/toml/Makefile
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,19 +0,0 @@ | |||||||
| install: |  | ||||||
| 	go install ./... |  | ||||||
|  |  | ||||||
| test: install |  | ||||||
| 	go test -v |  | ||||||
| 	toml-test toml-test-decoder |  | ||||||
| 	toml-test -encoder toml-test-encoder |  | ||||||
|  |  | ||||||
| fmt: |  | ||||||
| 	gofmt -w *.go */*.go |  | ||||||
| 	colcheck *.go */*.go |  | ||||||
|  |  | ||||||
| tags: |  | ||||||
| 	find ./ -name '*.go' -print0 | xargs -0 gotags > TAGS |  | ||||||
|  |  | ||||||
| push: |  | ||||||
| 	git push origin master |  | ||||||
| 	git push github master |  | ||||||
|  |  | ||||||
							
								
								
									
										218
									
								
								vendor/github.com/BurntSushi/toml/README.md
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,218 +0,0 @@ | |||||||
| ## TOML parser and encoder for Go with reflection |  | ||||||
|  |  | ||||||
| TOML stands for Tom's Obvious, Minimal Language. This Go package provides a |  | ||||||
| reflection interface similar to Go's standard library `json` and `xml` |  | ||||||
| packages. This package also supports the `encoding.TextUnmarshaler` and |  | ||||||
| `encoding.TextMarshaler` interfaces so that you can define custom data |  | ||||||
| representations. (There is an example of this below.) |  | ||||||
|  |  | ||||||
| Spec: https://github.com/toml-lang/toml |  | ||||||
|  |  | ||||||
| Compatible with TOML version |  | ||||||
| [v0.4.0](https://github.com/toml-lang/toml/blob/master/versions/en/toml-v0.4.0.md) |  | ||||||
|  |  | ||||||
| Documentation: https://godoc.org/github.com/BurntSushi/toml |  | ||||||
|  |  | ||||||
| Installation: |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| go get github.com/BurntSushi/toml |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Try the toml validator: |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| go get github.com/BurntSushi/toml/cmd/tomlv |  | ||||||
| tomlv some-toml-file.toml |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| [](https://travis-ci.org/BurntSushi/toml) [](https://godoc.org/github.com/BurntSushi/toml) |  | ||||||
|  |  | ||||||
| ### Testing |  | ||||||
|  |  | ||||||
| This package passes all tests in |  | ||||||
| [toml-test](https://github.com/BurntSushi/toml-test) for both the decoder |  | ||||||
| and the encoder. |  | ||||||
|  |  | ||||||
| ### Examples |  | ||||||
|  |  | ||||||
| This package works similarly to how the Go standard library handles `XML` |  | ||||||
| and `JSON`. Namely, data is loaded into Go values via reflection. |  | ||||||
|  |  | ||||||
| For the simplest example, consider some TOML file as just a list of keys |  | ||||||
| and values: |  | ||||||
|  |  | ||||||
| ```toml |  | ||||||
| Age = 25 |  | ||||||
| Cats = [ "Cauchy", "Plato" ] |  | ||||||
| Pi = 3.14 |  | ||||||
| Perfection = [ 6, 28, 496, 8128 ] |  | ||||||
| DOB = 1987-07-05T05:45:00Z |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Which could be defined in Go as: |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| type Config struct { |  | ||||||
|   Age int |  | ||||||
|   Cats []string |  | ||||||
|   Pi float64 |  | ||||||
|   Perfection []int |  | ||||||
|   DOB time.Time // requires `import time` |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| And then decoded with: |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| var conf Config |  | ||||||
| if _, err := toml.Decode(tomlData, &conf); err != nil { |  | ||||||
|   // handle error |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| You can also use struct tags if your struct field name doesn't map to a TOML |  | ||||||
| key value directly: |  | ||||||
|  |  | ||||||
| ```toml |  | ||||||
| some_key_NAME = "wat" |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| type TOML struct { |  | ||||||
|   ObscureKey string `toml:"some_key_NAME"` |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Using the `encoding.TextUnmarshaler` interface |  | ||||||
|  |  | ||||||
| Here's an example that automatically parses duration strings into |  | ||||||
| `time.Duration` values: |  | ||||||
|  |  | ||||||
| ```toml |  | ||||||
| [[song]] |  | ||||||
| name = "Thunder Road" |  | ||||||
| duration = "4m49s" |  | ||||||
|  |  | ||||||
| [[song]] |  | ||||||
| name = "Stairway to Heaven" |  | ||||||
| duration = "8m03s" |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Which can be decoded with: |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| type song struct { |  | ||||||
|   Name     string |  | ||||||
|   Duration duration |  | ||||||
| } |  | ||||||
| type songs struct { |  | ||||||
|   Song []song |  | ||||||
| } |  | ||||||
| var favorites songs |  | ||||||
| if _, err := toml.Decode(blob, &favorites); err != nil { |  | ||||||
|   log.Fatal(err) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| for _, s := range favorites.Song { |  | ||||||
|   fmt.Printf("%s (%s)\n", s.Name, s.Duration) |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| And you'll also need a `duration` type that satisfies the |  | ||||||
| `encoding.TextUnmarshaler` interface: |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| type duration struct { |  | ||||||
| 	time.Duration |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (d *duration) UnmarshalText(text []byte) error { |  | ||||||
| 	var err error |  | ||||||
| 	d.Duration, err = time.ParseDuration(string(text)) |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### More complex usage |  | ||||||
|  |  | ||||||
| Here's an example of how to load the example from the official spec page: |  | ||||||
|  |  | ||||||
| ```toml |  | ||||||
| # This is a TOML document. Boom. |  | ||||||
|  |  | ||||||
| title = "TOML Example" |  | ||||||
|  |  | ||||||
| [owner] |  | ||||||
| name = "Tom Preston-Werner" |  | ||||||
| organization = "GitHub" |  | ||||||
| bio = "GitHub Cofounder & CEO\nLikes tater tots and beer." |  | ||||||
| dob = 1979-05-27T07:32:00Z # First class dates? Why not? |  | ||||||
|  |  | ||||||
| [database] |  | ||||||
| server = "192.168.1.1" |  | ||||||
| ports = [ 8001, 8001, 8002 ] |  | ||||||
| connection_max = 5000 |  | ||||||
| enabled = true |  | ||||||
|  |  | ||||||
| [servers] |  | ||||||
|  |  | ||||||
|   # You can indent as you please. Tabs or spaces. TOML don't care. |  | ||||||
|   [servers.alpha] |  | ||||||
|   ip = "10.0.0.1" |  | ||||||
|   dc = "eqdc10" |  | ||||||
|  |  | ||||||
|   [servers.beta] |  | ||||||
|   ip = "10.0.0.2" |  | ||||||
|   dc = "eqdc10" |  | ||||||
|  |  | ||||||
| [clients] |  | ||||||
| data = [ ["gamma", "delta"], [1, 2] ] # just an update to make sure parsers support it |  | ||||||
|  |  | ||||||
| # Line breaks are OK when inside arrays |  | ||||||
| hosts = [ |  | ||||||
|   "alpha", |  | ||||||
|   "omega" |  | ||||||
| ] |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| And the corresponding Go types are: |  | ||||||
|  |  | ||||||
| ```go |  | ||||||
| type tomlConfig struct { |  | ||||||
| 	Title string |  | ||||||
| 	Owner ownerInfo |  | ||||||
| 	DB database `toml:"database"` |  | ||||||
| 	Servers map[string]server |  | ||||||
| 	Clients clients |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type ownerInfo struct { |  | ||||||
| 	Name string |  | ||||||
| 	Org string `toml:"organization"` |  | ||||||
| 	Bio string |  | ||||||
| 	DOB time.Time |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type database struct { |  | ||||||
| 	Server string |  | ||||||
| 	Ports []int |  | ||||||
| 	ConnMax int `toml:"connection_max"` |  | ||||||
| 	Enabled bool |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type server struct { |  | ||||||
| 	IP string |  | ||||||
| 	DC string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type clients struct { |  | ||||||
| 	Data [][]interface{} |  | ||||||
| 	Hosts []string |  | ||||||
| } |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Note that a case insensitive match will be tried if an exact match can't be |  | ||||||
| found. |  | ||||||
|  |  | ||||||
| A working example of the above can be found in `_examples/example.{go,toml}`. |  | ||||||
							
								
								
									
										509
									
								
								vendor/github.com/BurntSushi/toml/decode.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,509 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"io/ioutil" |  | ||||||
| 	"math" |  | ||||||
| 	"reflect" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func e(format string, args ...interface{}) error { |  | ||||||
| 	return fmt.Errorf("toml: "+format, args...) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Unmarshaler is the interface implemented by objects that can unmarshal a |  | ||||||
| // TOML description of themselves. |  | ||||||
| type Unmarshaler interface { |  | ||||||
| 	UnmarshalTOML(interface{}) error |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Unmarshal decodes the contents of `p` in TOML format into a pointer `v`. |  | ||||||
| func Unmarshal(p []byte, v interface{}) error { |  | ||||||
| 	_, err := Decode(string(p), v) |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Primitive is a TOML value that hasn't been decoded into a Go value. |  | ||||||
| // When using the various `Decode*` functions, the type `Primitive` may |  | ||||||
| // be given to any value, and its decoding will be delayed. |  | ||||||
| // |  | ||||||
| // A `Primitive` value can be decoded using the `PrimitiveDecode` function. |  | ||||||
| // |  | ||||||
| // The underlying representation of a `Primitive` value is subject to change. |  | ||||||
| // Do not rely on it. |  | ||||||
| // |  | ||||||
| // N.B. Primitive values are still parsed, so using them will only avoid |  | ||||||
| // the overhead of reflection. They can be useful when you don't know the |  | ||||||
| // exact type of TOML data until run time. |  | ||||||
| type Primitive struct { |  | ||||||
| 	undecoded interface{} |  | ||||||
| 	context   Key |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // DEPRECATED! |  | ||||||
| // |  | ||||||
| // Use MetaData.PrimitiveDecode instead. |  | ||||||
| func PrimitiveDecode(primValue Primitive, v interface{}) error { |  | ||||||
| 	md := MetaData{decoded: make(map[string]bool)} |  | ||||||
| 	return md.unify(primValue.undecoded, rvalue(v)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // PrimitiveDecode is just like the other `Decode*` functions, except it |  | ||||||
| // decodes a TOML value that has already been parsed. Valid primitive values |  | ||||||
| // can *only* be obtained from values filled by the decoder functions, |  | ||||||
| // including this method. (i.e., `v` may contain more `Primitive` |  | ||||||
| // values.) |  | ||||||
| // |  | ||||||
| // Meta data for primitive values is included in the meta data returned by |  | ||||||
| // the `Decode*` functions with one exception: keys returned by the Undecoded |  | ||||||
| // method will only reflect keys that were decoded. Namely, any keys hidden |  | ||||||
| // behind a Primitive will be considered undecoded. Executing this method will |  | ||||||
| // update the undecoded keys in the meta data. (See the example.) |  | ||||||
| func (md *MetaData) PrimitiveDecode(primValue Primitive, v interface{}) error { |  | ||||||
| 	md.context = primValue.context |  | ||||||
| 	defer func() { md.context = nil }() |  | ||||||
| 	return md.unify(primValue.undecoded, rvalue(v)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Decode will decode the contents of `data` in TOML format into a pointer |  | ||||||
| // `v`. |  | ||||||
| // |  | ||||||
| // TOML hashes correspond to Go structs or maps. (Dealer's choice. They can be |  | ||||||
| // used interchangeably.) |  | ||||||
| // |  | ||||||
| // TOML arrays of tables correspond to either a slice of structs or a slice |  | ||||||
| // of maps. |  | ||||||
| // |  | ||||||
| // TOML datetimes correspond to Go `time.Time` values. |  | ||||||
| // |  | ||||||
| // All other TOML types (float, string, int, bool and array) correspond |  | ||||||
| // to the obvious Go types. |  | ||||||
| // |  | ||||||
| // An exception to the above rules is if a type implements the |  | ||||||
| // encoding.TextUnmarshaler interface. In this case, any primitive TOML value |  | ||||||
| // (floats, strings, integers, booleans and datetimes) will be converted to |  | ||||||
| // a byte string and given to the value's UnmarshalText method. See the |  | ||||||
| // Unmarshaler example for a demonstration with time duration strings. |  | ||||||
| // |  | ||||||
| // Key mapping |  | ||||||
| // |  | ||||||
| // TOML keys can map to either keys in a Go map or field names in a Go |  | ||||||
| // struct. The special `toml` struct tag may be used to map TOML keys to |  | ||||||
| // struct fields that don't match the key name exactly. (See the example.) |  | ||||||
| // A case insensitive match to struct names will be tried if an exact match |  | ||||||
| // can't be found. |  | ||||||
| // |  | ||||||
| // The mapping between TOML values and Go values is loose. That is, there |  | ||||||
| // may exist TOML values that cannot be placed into your representation, and |  | ||||||
| // there may be parts of your representation that do not correspond to |  | ||||||
| // TOML values. This loose mapping can be made stricter by using the IsDefined |  | ||||||
| // and/or Undecoded methods on the MetaData returned. |  | ||||||
| // |  | ||||||
| // This decoder will not handle cyclic types. If a cyclic type is passed, |  | ||||||
| // `Decode` will not terminate. |  | ||||||
| func Decode(data string, v interface{}) (MetaData, error) { |  | ||||||
| 	rv := reflect.ValueOf(v) |  | ||||||
| 	if rv.Kind() != reflect.Ptr { |  | ||||||
| 		return MetaData{}, e("Decode of non-pointer %s", reflect.TypeOf(v)) |  | ||||||
| 	} |  | ||||||
| 	if rv.IsNil() { |  | ||||||
| 		return MetaData{}, e("Decode of nil %s", reflect.TypeOf(v)) |  | ||||||
| 	} |  | ||||||
| 	p, err := parse(data) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return MetaData{}, err |  | ||||||
| 	} |  | ||||||
| 	md := MetaData{ |  | ||||||
| 		p.mapping, p.types, p.ordered, |  | ||||||
| 		make(map[string]bool, len(p.ordered)), nil, |  | ||||||
| 	} |  | ||||||
| 	return md, md.unify(p.mapping, indirect(rv)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // DecodeFile is just like Decode, except it will automatically read the |  | ||||||
| // contents of the file at `fpath` and decode it for you. |  | ||||||
| func DecodeFile(fpath string, v interface{}) (MetaData, error) { |  | ||||||
| 	bs, err := ioutil.ReadFile(fpath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return MetaData{}, err |  | ||||||
| 	} |  | ||||||
| 	return Decode(string(bs), v) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // DecodeReader is just like Decode, except it will consume all bytes |  | ||||||
| // from the reader and decode it for you. |  | ||||||
| func DecodeReader(r io.Reader, v interface{}) (MetaData, error) { |  | ||||||
| 	bs, err := ioutil.ReadAll(r) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return MetaData{}, err |  | ||||||
| 	} |  | ||||||
| 	return Decode(string(bs), v) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // unify performs a sort of type unification based on the structure of `rv`, |  | ||||||
| // which is the client representation. |  | ||||||
| // |  | ||||||
| // Any type mismatch produces an error. Finding a type that we don't know |  | ||||||
| // how to handle produces an unsupported type error. |  | ||||||
| func (md *MetaData) unify(data interface{}, rv reflect.Value) error { |  | ||||||
|  |  | ||||||
| 	// Special case. Look for a `Primitive` value. |  | ||||||
| 	if rv.Type() == reflect.TypeOf((*Primitive)(nil)).Elem() { |  | ||||||
| 		// Save the undecoded data and the key context into the primitive |  | ||||||
| 		// value. |  | ||||||
| 		context := make(Key, len(md.context)) |  | ||||||
| 		copy(context, md.context) |  | ||||||
| 		rv.Set(reflect.ValueOf(Primitive{ |  | ||||||
| 			undecoded: data, |  | ||||||
| 			context:   context, |  | ||||||
| 		})) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Special case. Unmarshaler Interface support. |  | ||||||
| 	if rv.CanAddr() { |  | ||||||
| 		if v, ok := rv.Addr().Interface().(Unmarshaler); ok { |  | ||||||
| 			return v.UnmarshalTOML(data) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Special case. Handle time.Time values specifically. |  | ||||||
| 	// TODO: Remove this code when we decide to drop support for Go 1.1. |  | ||||||
| 	// This isn't necessary in Go 1.2 because time.Time satisfies the encoding |  | ||||||
| 	// interfaces. |  | ||||||
| 	if rv.Type().AssignableTo(rvalue(time.Time{}).Type()) { |  | ||||||
| 		return md.unifyDatetime(data, rv) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Special case. Look for a value satisfying the TextUnmarshaler interface. |  | ||||||
| 	if v, ok := rv.Interface().(TextUnmarshaler); ok { |  | ||||||
| 		return md.unifyText(data, v) |  | ||||||
| 	} |  | ||||||
| 	// BUG(burntsushi) |  | ||||||
| 	// The behavior here is incorrect whenever a Go type satisfies the |  | ||||||
| 	// encoding.TextUnmarshaler interface but also corresponds to a TOML |  | ||||||
| 	// hash or array. In particular, the unmarshaler should only be applied |  | ||||||
| 	// to primitive TOML values. But at this point, it will be applied to |  | ||||||
| 	// all kinds of values and produce an incorrect error whenever those values |  | ||||||
| 	// are hashes or arrays (including arrays of tables). |  | ||||||
|  |  | ||||||
| 	k := rv.Kind() |  | ||||||
|  |  | ||||||
| 	// laziness |  | ||||||
| 	if k >= reflect.Int && k <= reflect.Uint64 { |  | ||||||
| 		return md.unifyInt(data, rv) |  | ||||||
| 	} |  | ||||||
| 	switch k { |  | ||||||
| 	case reflect.Ptr: |  | ||||||
| 		elem := reflect.New(rv.Type().Elem()) |  | ||||||
| 		err := md.unify(data, reflect.Indirect(elem)) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		rv.Set(elem) |  | ||||||
| 		return nil |  | ||||||
| 	case reflect.Struct: |  | ||||||
| 		return md.unifyStruct(data, rv) |  | ||||||
| 	case reflect.Map: |  | ||||||
| 		return md.unifyMap(data, rv) |  | ||||||
| 	case reflect.Array: |  | ||||||
| 		return md.unifyArray(data, rv) |  | ||||||
| 	case reflect.Slice: |  | ||||||
| 		return md.unifySlice(data, rv) |  | ||||||
| 	case reflect.String: |  | ||||||
| 		return md.unifyString(data, rv) |  | ||||||
| 	case reflect.Bool: |  | ||||||
| 		return md.unifyBool(data, rv) |  | ||||||
| 	case reflect.Interface: |  | ||||||
| 		// we only support empty interfaces. |  | ||||||
| 		if rv.NumMethod() > 0 { |  | ||||||
| 			return e("unsupported type %s", rv.Type()) |  | ||||||
| 		} |  | ||||||
| 		return md.unifyAnything(data, rv) |  | ||||||
| 	case reflect.Float32: |  | ||||||
| 		fallthrough |  | ||||||
| 	case reflect.Float64: |  | ||||||
| 		return md.unifyFloat64(data, rv) |  | ||||||
| 	} |  | ||||||
| 	return e("unsupported type %s", rv.Kind()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyStruct(mapping interface{}, rv reflect.Value) error { |  | ||||||
| 	tmap, ok := mapping.(map[string]interface{}) |  | ||||||
| 	if !ok { |  | ||||||
| 		if mapping == nil { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		return e("type mismatch for %s: expected table but found %T", |  | ||||||
| 			rv.Type().String(), mapping) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for key, datum := range tmap { |  | ||||||
| 		var f *field |  | ||||||
| 		fields := cachedTypeFields(rv.Type()) |  | ||||||
| 		for i := range fields { |  | ||||||
| 			ff := &fields[i] |  | ||||||
| 			if ff.name == key { |  | ||||||
| 				f = ff |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 			if f == nil && strings.EqualFold(ff.name, key) { |  | ||||||
| 				f = ff |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if f != nil { |  | ||||||
| 			subv := rv |  | ||||||
| 			for _, i := range f.index { |  | ||||||
| 				subv = indirect(subv.Field(i)) |  | ||||||
| 			} |  | ||||||
| 			if isUnifiable(subv) { |  | ||||||
| 				md.decoded[md.context.add(key).String()] = true |  | ||||||
| 				md.context = append(md.context, key) |  | ||||||
| 				if err := md.unify(datum, subv); err != nil { |  | ||||||
| 					return err |  | ||||||
| 				} |  | ||||||
| 				md.context = md.context[0 : len(md.context)-1] |  | ||||||
| 			} else if f.name != "" { |  | ||||||
| 				// Bad user! No soup for you! |  | ||||||
| 				return e("cannot write unexported field %s.%s", |  | ||||||
| 					rv.Type().String(), f.name) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyMap(mapping interface{}, rv reflect.Value) error { |  | ||||||
| 	tmap, ok := mapping.(map[string]interface{}) |  | ||||||
| 	if !ok { |  | ||||||
| 		if tmap == nil { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		return badtype("map", mapping) |  | ||||||
| 	} |  | ||||||
| 	if rv.IsNil() { |  | ||||||
| 		rv.Set(reflect.MakeMap(rv.Type())) |  | ||||||
| 	} |  | ||||||
| 	for k, v := range tmap { |  | ||||||
| 		md.decoded[md.context.add(k).String()] = true |  | ||||||
| 		md.context = append(md.context, k) |  | ||||||
|  |  | ||||||
| 		rvkey := indirect(reflect.New(rv.Type().Key())) |  | ||||||
| 		rvval := reflect.Indirect(reflect.New(rv.Type().Elem())) |  | ||||||
| 		if err := md.unify(v, rvval); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		md.context = md.context[0 : len(md.context)-1] |  | ||||||
|  |  | ||||||
| 		rvkey.SetString(k) |  | ||||||
| 		rv.SetMapIndex(rvkey, rvval) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyArray(data interface{}, rv reflect.Value) error { |  | ||||||
| 	datav := reflect.ValueOf(data) |  | ||||||
| 	if datav.Kind() != reflect.Slice { |  | ||||||
| 		if !datav.IsValid() { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		return badtype("slice", data) |  | ||||||
| 	} |  | ||||||
| 	sliceLen := datav.Len() |  | ||||||
| 	if sliceLen != rv.Len() { |  | ||||||
| 		return e("expected array length %d; got TOML array of length %d", |  | ||||||
| 			rv.Len(), sliceLen) |  | ||||||
| 	} |  | ||||||
| 	return md.unifySliceArray(datav, rv) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifySlice(data interface{}, rv reflect.Value) error { |  | ||||||
| 	datav := reflect.ValueOf(data) |  | ||||||
| 	if datav.Kind() != reflect.Slice { |  | ||||||
| 		if !datav.IsValid() { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 		return badtype("slice", data) |  | ||||||
| 	} |  | ||||||
| 	n := datav.Len() |  | ||||||
| 	if rv.IsNil() || rv.Cap() < n { |  | ||||||
| 		rv.Set(reflect.MakeSlice(rv.Type(), n, n)) |  | ||||||
| 	} |  | ||||||
| 	rv.SetLen(n) |  | ||||||
| 	return md.unifySliceArray(datav, rv) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifySliceArray(data, rv reflect.Value) error { |  | ||||||
| 	sliceLen := data.Len() |  | ||||||
| 	for i := 0; i < sliceLen; i++ { |  | ||||||
| 		v := data.Index(i).Interface() |  | ||||||
| 		sliceval := indirect(rv.Index(i)) |  | ||||||
| 		if err := md.unify(v, sliceval); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyDatetime(data interface{}, rv reflect.Value) error { |  | ||||||
| 	if _, ok := data.(time.Time); ok { |  | ||||||
| 		rv.Set(reflect.ValueOf(data)) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return badtype("time.Time", data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyString(data interface{}, rv reflect.Value) error { |  | ||||||
| 	if s, ok := data.(string); ok { |  | ||||||
| 		rv.SetString(s) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return badtype("string", data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyFloat64(data interface{}, rv reflect.Value) error { |  | ||||||
| 	if num, ok := data.(float64); ok { |  | ||||||
| 		switch rv.Kind() { |  | ||||||
| 		case reflect.Float32: |  | ||||||
| 			fallthrough |  | ||||||
| 		case reflect.Float64: |  | ||||||
| 			rv.SetFloat(num) |  | ||||||
| 		default: |  | ||||||
| 			panic("bug") |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return badtype("float", data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyInt(data interface{}, rv reflect.Value) error { |  | ||||||
| 	if num, ok := data.(int64); ok { |  | ||||||
| 		if rv.Kind() >= reflect.Int && rv.Kind() <= reflect.Int64 { |  | ||||||
| 			switch rv.Kind() { |  | ||||||
| 			case reflect.Int, reflect.Int64: |  | ||||||
| 				// No bounds checking necessary. |  | ||||||
| 			case reflect.Int8: |  | ||||||
| 				if num < math.MinInt8 || num > math.MaxInt8 { |  | ||||||
| 					return e("value %d is out of range for int8", num) |  | ||||||
| 				} |  | ||||||
| 			case reflect.Int16: |  | ||||||
| 				if num < math.MinInt16 || num > math.MaxInt16 { |  | ||||||
| 					return e("value %d is out of range for int16", num) |  | ||||||
| 				} |  | ||||||
| 			case reflect.Int32: |  | ||||||
| 				if num < math.MinInt32 || num > math.MaxInt32 { |  | ||||||
| 					return e("value %d is out of range for int32", num) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			rv.SetInt(num) |  | ||||||
| 		} else if rv.Kind() >= reflect.Uint && rv.Kind() <= reflect.Uint64 { |  | ||||||
| 			unum := uint64(num) |  | ||||||
| 			switch rv.Kind() { |  | ||||||
| 			case reflect.Uint, reflect.Uint64: |  | ||||||
| 				// No bounds checking necessary. |  | ||||||
| 			case reflect.Uint8: |  | ||||||
| 				if num < 0 || unum > math.MaxUint8 { |  | ||||||
| 					return e("value %d is out of range for uint8", num) |  | ||||||
| 				} |  | ||||||
| 			case reflect.Uint16: |  | ||||||
| 				if num < 0 || unum > math.MaxUint16 { |  | ||||||
| 					return e("value %d is out of range for uint16", num) |  | ||||||
| 				} |  | ||||||
| 			case reflect.Uint32: |  | ||||||
| 				if num < 0 || unum > math.MaxUint32 { |  | ||||||
| 					return e("value %d is out of range for uint32", num) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 			rv.SetUint(unum) |  | ||||||
| 		} else { |  | ||||||
| 			panic("unreachable") |  | ||||||
| 		} |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return badtype("integer", data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyBool(data interface{}, rv reflect.Value) error { |  | ||||||
| 	if b, ok := data.(bool); ok { |  | ||||||
| 		rv.SetBool(b) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return badtype("boolean", data) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyAnything(data interface{}, rv reflect.Value) error { |  | ||||||
| 	rv.Set(reflect.ValueOf(data)) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (md *MetaData) unifyText(data interface{}, v TextUnmarshaler) error { |  | ||||||
| 	var s string |  | ||||||
| 	switch sdata := data.(type) { |  | ||||||
| 	case TextMarshaler: |  | ||||||
| 		text, err := sdata.MarshalText() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		s = string(text) |  | ||||||
| 	case fmt.Stringer: |  | ||||||
| 		s = sdata.String() |  | ||||||
| 	case string: |  | ||||||
| 		s = sdata |  | ||||||
| 	case bool: |  | ||||||
| 		s = fmt.Sprintf("%v", sdata) |  | ||||||
| 	case int64: |  | ||||||
| 		s = fmt.Sprintf("%d", sdata) |  | ||||||
| 	case float64: |  | ||||||
| 		s = fmt.Sprintf("%f", sdata) |  | ||||||
| 	default: |  | ||||||
| 		return badtype("primitive (string-like)", data) |  | ||||||
| 	} |  | ||||||
| 	if err := v.UnmarshalText([]byte(s)); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // rvalue returns a reflect.Value of `v`. All pointers are resolved. |  | ||||||
| func rvalue(v interface{}) reflect.Value { |  | ||||||
| 	return indirect(reflect.ValueOf(v)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // indirect returns the value pointed to by a pointer. |  | ||||||
| // Pointers are followed until the value is not a pointer. |  | ||||||
| // New values are allocated for each nil pointer. |  | ||||||
| // |  | ||||||
| // An exception to this rule is if the value satisfies an interface of |  | ||||||
| // interest to us (like encoding.TextUnmarshaler). |  | ||||||
| func indirect(v reflect.Value) reflect.Value { |  | ||||||
| 	if v.Kind() != reflect.Ptr { |  | ||||||
| 		if v.CanSet() { |  | ||||||
| 			pv := v.Addr() |  | ||||||
| 			if _, ok := pv.Interface().(TextUnmarshaler); ok { |  | ||||||
| 				return pv |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return v |  | ||||||
| 	} |  | ||||||
| 	if v.IsNil() { |  | ||||||
| 		v.Set(reflect.New(v.Type().Elem())) |  | ||||||
| 	} |  | ||||||
| 	return indirect(reflect.Indirect(v)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isUnifiable(rv reflect.Value) bool { |  | ||||||
| 	if rv.CanSet() { |  | ||||||
| 		return true |  | ||||||
| 	} |  | ||||||
| 	if _, ok := rv.Interface().(TextUnmarshaler); ok { |  | ||||||
| 		return true |  | ||||||
| 	} |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func badtype(expected string, data interface{}) error { |  | ||||||
| 	return e("cannot load TOML value of type %T into a Go %s", data, expected) |  | ||||||
| } |  | ||||||
							
								
								
									
										121
									
								
								vendor/github.com/BurntSushi/toml/decode_meta.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,121 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| import "strings" |  | ||||||
|  |  | ||||||
| // MetaData allows access to meta information about TOML data that may not |  | ||||||
| // be inferrable via reflection. In particular, whether a key has been defined |  | ||||||
| // and the TOML type of a key. |  | ||||||
| type MetaData struct { |  | ||||||
| 	mapping map[string]interface{} |  | ||||||
| 	types   map[string]tomlType |  | ||||||
| 	keys    []Key |  | ||||||
| 	decoded map[string]bool |  | ||||||
| 	context Key // Used only during decoding. |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // IsDefined returns true if the key given exists in the TOML data. The key |  | ||||||
| // should be specified hierarchially. e.g., |  | ||||||
| // |  | ||||||
| //	// access the TOML key 'a.b.c' |  | ||||||
| //	IsDefined("a", "b", "c") |  | ||||||
| // |  | ||||||
| // IsDefined will return false if an empty key given. Keys are case sensitive. |  | ||||||
| func (md *MetaData) IsDefined(key ...string) bool { |  | ||||||
| 	if len(key) == 0 { |  | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var hash map[string]interface{} |  | ||||||
| 	var ok bool |  | ||||||
| 	var hashOrVal interface{} = md.mapping |  | ||||||
| 	for _, k := range key { |  | ||||||
| 		if hash, ok = hashOrVal.(map[string]interface{}); !ok { |  | ||||||
| 			return false |  | ||||||
| 		} |  | ||||||
| 		if hashOrVal, ok = hash[k]; !ok { |  | ||||||
| 			return false |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return true |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Type returns a string representation of the type of the key specified. |  | ||||||
| // |  | ||||||
| // Type will return the empty string if given an empty key or a key that |  | ||||||
| // does not exist. Keys are case sensitive. |  | ||||||
| func (md *MetaData) Type(key ...string) string { |  | ||||||
| 	fullkey := strings.Join(key, ".") |  | ||||||
| 	if typ, ok := md.types[fullkey]; ok { |  | ||||||
| 		return typ.typeString() |  | ||||||
| 	} |  | ||||||
| 	return "" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Key is the type of any TOML key, including key groups. Use (MetaData).Keys |  | ||||||
| // to get values of this type. |  | ||||||
| type Key []string |  | ||||||
|  |  | ||||||
| func (k Key) String() string { |  | ||||||
| 	return strings.Join(k, ".") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (k Key) maybeQuotedAll() string { |  | ||||||
| 	var ss []string |  | ||||||
| 	for i := range k { |  | ||||||
| 		ss = append(ss, k.maybeQuoted(i)) |  | ||||||
| 	} |  | ||||||
| 	return strings.Join(ss, ".") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (k Key) maybeQuoted(i int) string { |  | ||||||
| 	quote := false |  | ||||||
| 	for _, c := range k[i] { |  | ||||||
| 		if !isBareKeyChar(c) { |  | ||||||
| 			quote = true |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if quote { |  | ||||||
| 		return "\"" + strings.Replace(k[i], "\"", "\\\"", -1) + "\"" |  | ||||||
| 	} |  | ||||||
| 	return k[i] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (k Key) add(piece string) Key { |  | ||||||
| 	newKey := make(Key, len(k)+1) |  | ||||||
| 	copy(newKey, k) |  | ||||||
| 	newKey[len(k)] = piece |  | ||||||
| 	return newKey |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Keys returns a slice of every key in the TOML data, including key groups. |  | ||||||
| // Each key is itself a slice, where the first element is the top of the |  | ||||||
| // hierarchy and the last is the most specific. |  | ||||||
| // |  | ||||||
| // The list will have the same order as the keys appeared in the TOML data. |  | ||||||
| // |  | ||||||
| // All keys returned are non-empty. |  | ||||||
| func (md *MetaData) Keys() []Key { |  | ||||||
| 	return md.keys |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Undecoded returns all keys that have not been decoded in the order in which |  | ||||||
| // they appear in the original TOML document. |  | ||||||
| // |  | ||||||
| // This includes keys that haven't been decoded because of a Primitive value. |  | ||||||
| // Once the Primitive value is decoded, the keys will be considered decoded. |  | ||||||
| // |  | ||||||
| // Also note that decoding into an empty interface will result in no decoding, |  | ||||||
| // and so no keys will be considered decoded. |  | ||||||
| // |  | ||||||
| // In this sense, the Undecoded keys correspond to keys in the TOML document |  | ||||||
| // that do not have a concrete type in your representation. |  | ||||||
| func (md *MetaData) Undecoded() []Key { |  | ||||||
| 	undecoded := make([]Key, 0, len(md.keys)) |  | ||||||
| 	for _, key := range md.keys { |  | ||||||
| 		if !md.decoded[key.String()] { |  | ||||||
| 			undecoded = append(undecoded, key) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return undecoded |  | ||||||
| } |  | ||||||
							
								
								
									
										27
									
								
								vendor/github.com/BurntSushi/toml/doc.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,27 +0,0 @@ | |||||||
| /* |  | ||||||
| Package toml provides facilities for decoding and encoding TOML configuration |  | ||||||
| files via reflection. There is also support for delaying decoding with |  | ||||||
| the Primitive type, and querying the set of keys in a TOML document with the |  | ||||||
| MetaData type. |  | ||||||
|  |  | ||||||
| The specification implemented: https://github.com/toml-lang/toml |  | ||||||
|  |  | ||||||
| The sub-command github.com/BurntSushi/toml/cmd/tomlv can be used to verify |  | ||||||
| whether a file is a valid TOML document. It can also be used to print the |  | ||||||
| type of each key in a TOML document. |  | ||||||
|  |  | ||||||
| Testing |  | ||||||
|  |  | ||||||
| There are two important types of tests used for this package. The first is |  | ||||||
| contained inside '*_test.go' files and uses the standard Go unit testing |  | ||||||
| framework. These tests are primarily devoted to holistically testing the |  | ||||||
| decoder and encoder. |  | ||||||
|  |  | ||||||
| The second type of testing is used to verify the implementation's adherence |  | ||||||
| to the TOML specification. These tests have been factored into their own |  | ||||||
| project: https://github.com/BurntSushi/toml-test |  | ||||||
|  |  | ||||||
| The reason the tests are in a separate project is so that they can be used by |  | ||||||
| any implementation of TOML. Namely, it is language agnostic. |  | ||||||
| */ |  | ||||||
| package toml |  | ||||||
							
								
								
									
										568
									
								
								vendor/github.com/BurntSushi/toml/encode.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,568 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"bufio" |  | ||||||
| 	"errors" |  | ||||||
| 	"fmt" |  | ||||||
| 	"io" |  | ||||||
| 	"reflect" |  | ||||||
| 	"sort" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type tomlEncodeError struct{ error } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	errArrayMixedElementTypes = errors.New( |  | ||||||
| 		"toml: cannot encode array with mixed element types") |  | ||||||
| 	errArrayNilElement = errors.New( |  | ||||||
| 		"toml: cannot encode array with nil element") |  | ||||||
| 	errNonString = errors.New( |  | ||||||
| 		"toml: cannot encode a map with non-string key type") |  | ||||||
| 	errAnonNonStruct = errors.New( |  | ||||||
| 		"toml: cannot encode an anonymous field that is not a struct") |  | ||||||
| 	errArrayNoTable = errors.New( |  | ||||||
| 		"toml: TOML array element cannot contain a table") |  | ||||||
| 	errNoKey = errors.New( |  | ||||||
| 		"toml: top-level values must be Go maps or structs") |  | ||||||
| 	errAnything = errors.New("") // used in testing |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var quotedReplacer = strings.NewReplacer( |  | ||||||
| 	"\t", "\\t", |  | ||||||
| 	"\n", "\\n", |  | ||||||
| 	"\r", "\\r", |  | ||||||
| 	"\"", "\\\"", |  | ||||||
| 	"\\", "\\\\", |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Encoder controls the encoding of Go values to a TOML document to some |  | ||||||
| // io.Writer. |  | ||||||
| // |  | ||||||
| // The indentation level can be controlled with the Indent field. |  | ||||||
| type Encoder struct { |  | ||||||
| 	// A single indentation level. By default it is two spaces. |  | ||||||
| 	Indent string |  | ||||||
|  |  | ||||||
| 	// hasWritten is whether we have written any output to w yet. |  | ||||||
| 	hasWritten bool |  | ||||||
| 	w          *bufio.Writer |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // NewEncoder returns a TOML encoder that encodes Go values to the io.Writer |  | ||||||
| // given. By default, a single indentation level is 2 spaces. |  | ||||||
| func NewEncoder(w io.Writer) *Encoder { |  | ||||||
| 	return &Encoder{ |  | ||||||
| 		w:      bufio.NewWriter(w), |  | ||||||
| 		Indent: "  ", |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Encode writes a TOML representation of the Go value to the underlying |  | ||||||
| // io.Writer. If the value given cannot be encoded to a valid TOML document, |  | ||||||
| // then an error is returned. |  | ||||||
| // |  | ||||||
| // The mapping between Go values and TOML values should be precisely the same |  | ||||||
| // as for the Decode* functions. Similarly, the TextMarshaler interface is |  | ||||||
| // supported by encoding the resulting bytes as strings. (If you want to write |  | ||||||
| // arbitrary binary data then you will need to use something like base64 since |  | ||||||
| // TOML does not have any binary types.) |  | ||||||
| // |  | ||||||
| // When encoding TOML hashes (i.e., Go maps or structs), keys without any |  | ||||||
| // sub-hashes are encoded first. |  | ||||||
| // |  | ||||||
| // If a Go map is encoded, then its keys are sorted alphabetically for |  | ||||||
| // deterministic output. More control over this behavior may be provided if |  | ||||||
| // there is demand for it. |  | ||||||
| // |  | ||||||
| // Encoding Go values without a corresponding TOML representation---like map |  | ||||||
| // types with non-string keys---will cause an error to be returned. Similarly |  | ||||||
| // for mixed arrays/slices, arrays/slices with nil elements, embedded |  | ||||||
| // non-struct types and nested slices containing maps or structs. |  | ||||||
| // (e.g., [][]map[string]string is not allowed but []map[string]string is OK |  | ||||||
| // and so is []map[string][]string.) |  | ||||||
| func (enc *Encoder) Encode(v interface{}) error { |  | ||||||
| 	rv := eindirect(reflect.ValueOf(v)) |  | ||||||
| 	if err := enc.safeEncode(Key([]string{}), rv); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	return enc.w.Flush() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) safeEncode(key Key, rv reflect.Value) (err error) { |  | ||||||
| 	defer func() { |  | ||||||
| 		if r := recover(); r != nil { |  | ||||||
| 			if terr, ok := r.(tomlEncodeError); ok { |  | ||||||
| 				err = terr.error |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 			panic(r) |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
| 	enc.encode(key, rv) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) encode(key Key, rv reflect.Value) { |  | ||||||
| 	// Special case. Time needs to be in ISO8601 format. |  | ||||||
| 	// Special case. If we can marshal the type to text, then we used that. |  | ||||||
| 	// Basically, this prevents the encoder for handling these types as |  | ||||||
| 	// generic structs (or whatever the underlying type of a TextMarshaler is). |  | ||||||
| 	switch rv.Interface().(type) { |  | ||||||
| 	case time.Time, TextMarshaler: |  | ||||||
| 		enc.keyEqElement(key, rv) |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	k := rv.Kind() |  | ||||||
| 	switch k { |  | ||||||
| 	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, |  | ||||||
| 		reflect.Int64, |  | ||||||
| 		reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, |  | ||||||
| 		reflect.Uint64, |  | ||||||
| 		reflect.Float32, reflect.Float64, reflect.String, reflect.Bool: |  | ||||||
| 		enc.keyEqElement(key, rv) |  | ||||||
| 	case reflect.Array, reflect.Slice: |  | ||||||
| 		if typeEqual(tomlArrayHash, tomlTypeOfGo(rv)) { |  | ||||||
| 			enc.eArrayOfTables(key, rv) |  | ||||||
| 		} else { |  | ||||||
| 			enc.keyEqElement(key, rv) |  | ||||||
| 		} |  | ||||||
| 	case reflect.Interface: |  | ||||||
| 		if rv.IsNil() { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		enc.encode(key, rv.Elem()) |  | ||||||
| 	case reflect.Map: |  | ||||||
| 		if rv.IsNil() { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		enc.eTable(key, rv) |  | ||||||
| 	case reflect.Ptr: |  | ||||||
| 		if rv.IsNil() { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		enc.encode(key, rv.Elem()) |  | ||||||
| 	case reflect.Struct: |  | ||||||
| 		enc.eTable(key, rv) |  | ||||||
| 	default: |  | ||||||
| 		panic(e("unsupported type for key '%s': %s", key, k)) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // eElement encodes any value that can be an array element (primitives and |  | ||||||
| // arrays). |  | ||||||
| func (enc *Encoder) eElement(rv reflect.Value) { |  | ||||||
| 	switch v := rv.Interface().(type) { |  | ||||||
| 	case time.Time: |  | ||||||
| 		// Special case time.Time as a primitive. Has to come before |  | ||||||
| 		// TextMarshaler below because time.Time implements |  | ||||||
| 		// encoding.TextMarshaler, but we need to always use UTC. |  | ||||||
| 		enc.wf(v.UTC().Format("2006-01-02T15:04:05Z")) |  | ||||||
| 		return |  | ||||||
| 	case TextMarshaler: |  | ||||||
| 		// Special case. Use text marshaler if it's available for this value. |  | ||||||
| 		if s, err := v.MarshalText(); err != nil { |  | ||||||
| 			encPanic(err) |  | ||||||
| 		} else { |  | ||||||
| 			enc.writeQuoted(string(s)) |  | ||||||
| 		} |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	switch rv.Kind() { |  | ||||||
| 	case reflect.Bool: |  | ||||||
| 		enc.wf(strconv.FormatBool(rv.Bool())) |  | ||||||
| 	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, |  | ||||||
| 		reflect.Int64: |  | ||||||
| 		enc.wf(strconv.FormatInt(rv.Int(), 10)) |  | ||||||
| 	case reflect.Uint, reflect.Uint8, reflect.Uint16, |  | ||||||
| 		reflect.Uint32, reflect.Uint64: |  | ||||||
| 		enc.wf(strconv.FormatUint(rv.Uint(), 10)) |  | ||||||
| 	case reflect.Float32: |  | ||||||
| 		enc.wf(floatAddDecimal(strconv.FormatFloat(rv.Float(), 'f', -1, 32))) |  | ||||||
| 	case reflect.Float64: |  | ||||||
| 		enc.wf(floatAddDecimal(strconv.FormatFloat(rv.Float(), 'f', -1, 64))) |  | ||||||
| 	case reflect.Array, reflect.Slice: |  | ||||||
| 		enc.eArrayOrSliceElement(rv) |  | ||||||
| 	case reflect.Interface: |  | ||||||
| 		enc.eElement(rv.Elem()) |  | ||||||
| 	case reflect.String: |  | ||||||
| 		enc.writeQuoted(rv.String()) |  | ||||||
| 	default: |  | ||||||
| 		panic(e("unexpected primitive type: %s", rv.Kind())) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // By the TOML spec, all floats must have a decimal with at least one |  | ||||||
| // number on either side. |  | ||||||
| func floatAddDecimal(fstr string) string { |  | ||||||
| 	if !strings.Contains(fstr, ".") { |  | ||||||
| 		return fstr + ".0" |  | ||||||
| 	} |  | ||||||
| 	return fstr |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) writeQuoted(s string) { |  | ||||||
| 	enc.wf("\"%s\"", quotedReplacer.Replace(s)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eArrayOrSliceElement(rv reflect.Value) { |  | ||||||
| 	length := rv.Len() |  | ||||||
| 	enc.wf("[") |  | ||||||
| 	for i := 0; i < length; i++ { |  | ||||||
| 		elem := rv.Index(i) |  | ||||||
| 		enc.eElement(elem) |  | ||||||
| 		if i != length-1 { |  | ||||||
| 			enc.wf(", ") |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	enc.wf("]") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eArrayOfTables(key Key, rv reflect.Value) { |  | ||||||
| 	if len(key) == 0 { |  | ||||||
| 		encPanic(errNoKey) |  | ||||||
| 	} |  | ||||||
| 	for i := 0; i < rv.Len(); i++ { |  | ||||||
| 		trv := rv.Index(i) |  | ||||||
| 		if isNil(trv) { |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		panicIfInvalidKey(key) |  | ||||||
| 		enc.newline() |  | ||||||
| 		enc.wf("%s[[%s]]", enc.indentStr(key), key.maybeQuotedAll()) |  | ||||||
| 		enc.newline() |  | ||||||
| 		enc.eMapOrStruct(key, trv) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eTable(key Key, rv reflect.Value) { |  | ||||||
| 	panicIfInvalidKey(key) |  | ||||||
| 	if len(key) == 1 { |  | ||||||
| 		// Output an extra newline between top-level tables. |  | ||||||
| 		// (The newline isn't written if nothing else has been written though.) |  | ||||||
| 		enc.newline() |  | ||||||
| 	} |  | ||||||
| 	if len(key) > 0 { |  | ||||||
| 		enc.wf("%s[%s]", enc.indentStr(key), key.maybeQuotedAll()) |  | ||||||
| 		enc.newline() |  | ||||||
| 	} |  | ||||||
| 	enc.eMapOrStruct(key, rv) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eMapOrStruct(key Key, rv reflect.Value) { |  | ||||||
| 	switch rv := eindirect(rv); rv.Kind() { |  | ||||||
| 	case reflect.Map: |  | ||||||
| 		enc.eMap(key, rv) |  | ||||||
| 	case reflect.Struct: |  | ||||||
| 		enc.eStruct(key, rv) |  | ||||||
| 	default: |  | ||||||
| 		panic("eTable: unhandled reflect.Value Kind: " + rv.Kind().String()) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eMap(key Key, rv reflect.Value) { |  | ||||||
| 	rt := rv.Type() |  | ||||||
| 	if rt.Key().Kind() != reflect.String { |  | ||||||
| 		encPanic(errNonString) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Sort keys so that we have deterministic output. And write keys directly |  | ||||||
| 	// underneath this key first, before writing sub-structs or sub-maps. |  | ||||||
| 	var mapKeysDirect, mapKeysSub []string |  | ||||||
| 	for _, mapKey := range rv.MapKeys() { |  | ||||||
| 		k := mapKey.String() |  | ||||||
| 		if typeIsHash(tomlTypeOfGo(rv.MapIndex(mapKey))) { |  | ||||||
| 			mapKeysSub = append(mapKeysSub, k) |  | ||||||
| 		} else { |  | ||||||
| 			mapKeysDirect = append(mapKeysDirect, k) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	var writeMapKeys = func(mapKeys []string) { |  | ||||||
| 		sort.Strings(mapKeys) |  | ||||||
| 		for _, mapKey := range mapKeys { |  | ||||||
| 			mrv := rv.MapIndex(reflect.ValueOf(mapKey)) |  | ||||||
| 			if isNil(mrv) { |  | ||||||
| 				// Don't write anything for nil fields. |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			enc.encode(key.add(mapKey), mrv) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	writeMapKeys(mapKeysDirect) |  | ||||||
| 	writeMapKeys(mapKeysSub) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) eStruct(key Key, rv reflect.Value) { |  | ||||||
| 	// Write keys for fields directly under this key first, because if we write |  | ||||||
| 	// a field that creates a new table, then all keys under it will be in that |  | ||||||
| 	// table (not the one we're writing here). |  | ||||||
| 	rt := rv.Type() |  | ||||||
| 	var fieldsDirect, fieldsSub [][]int |  | ||||||
| 	var addFields func(rt reflect.Type, rv reflect.Value, start []int) |  | ||||||
| 	addFields = func(rt reflect.Type, rv reflect.Value, start []int) { |  | ||||||
| 		for i := 0; i < rt.NumField(); i++ { |  | ||||||
| 			f := rt.Field(i) |  | ||||||
| 			// skip unexported fields |  | ||||||
| 			if f.PkgPath != "" && !f.Anonymous { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			frv := rv.Field(i) |  | ||||||
| 			if f.Anonymous { |  | ||||||
| 				t := f.Type |  | ||||||
| 				switch t.Kind() { |  | ||||||
| 				case reflect.Struct: |  | ||||||
| 					// Treat anonymous struct fields with |  | ||||||
| 					// tag names as though they are not |  | ||||||
| 					// anonymous, like encoding/json does. |  | ||||||
| 					if getOptions(f.Tag).name == "" { |  | ||||||
| 						addFields(t, frv, f.Index) |  | ||||||
| 						continue |  | ||||||
| 					} |  | ||||||
| 				case reflect.Ptr: |  | ||||||
| 					if t.Elem().Kind() == reflect.Struct && |  | ||||||
| 						getOptions(f.Tag).name == "" { |  | ||||||
| 						if !frv.IsNil() { |  | ||||||
| 							addFields(t.Elem(), frv.Elem(), f.Index) |  | ||||||
| 						} |  | ||||||
| 						continue |  | ||||||
| 					} |  | ||||||
| 					// Fall through to the normal field encoding logic below |  | ||||||
| 					// for non-struct anonymous fields. |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if typeIsHash(tomlTypeOfGo(frv)) { |  | ||||||
| 				fieldsSub = append(fieldsSub, append(start, f.Index...)) |  | ||||||
| 			} else { |  | ||||||
| 				fieldsDirect = append(fieldsDirect, append(start, f.Index...)) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	addFields(rt, rv, nil) |  | ||||||
|  |  | ||||||
| 	var writeFields = func(fields [][]int) { |  | ||||||
| 		for _, fieldIndex := range fields { |  | ||||||
| 			sft := rt.FieldByIndex(fieldIndex) |  | ||||||
| 			sf := rv.FieldByIndex(fieldIndex) |  | ||||||
| 			if isNil(sf) { |  | ||||||
| 				// Don't write anything for nil fields. |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			opts := getOptions(sft.Tag) |  | ||||||
| 			if opts.skip { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			keyName := sft.Name |  | ||||||
| 			if opts.name != "" { |  | ||||||
| 				keyName = opts.name |  | ||||||
| 			} |  | ||||||
| 			if opts.omitempty && isEmpty(sf) { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			if opts.omitzero && isZero(sf) { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			enc.encode(key.add(keyName), sf) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	writeFields(fieldsDirect) |  | ||||||
| 	writeFields(fieldsSub) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // tomlTypeName returns the TOML type name of the Go value's type. It is |  | ||||||
| // used to determine whether the types of array elements are mixed (which is |  | ||||||
| // forbidden). If the Go value is nil, then it is illegal for it to be an array |  | ||||||
| // element, and valueIsNil is returned as true. |  | ||||||
|  |  | ||||||
| // Returns the TOML type of a Go value. The type may be `nil`, which means |  | ||||||
| // no concrete TOML type could be found. |  | ||||||
| func tomlTypeOfGo(rv reflect.Value) tomlType { |  | ||||||
| 	if isNil(rv) || !rv.IsValid() { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	switch rv.Kind() { |  | ||||||
| 	case reflect.Bool: |  | ||||||
| 		return tomlBool |  | ||||||
| 	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, |  | ||||||
| 		reflect.Int64, |  | ||||||
| 		reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, |  | ||||||
| 		reflect.Uint64: |  | ||||||
| 		return tomlInteger |  | ||||||
| 	case reflect.Float32, reflect.Float64: |  | ||||||
| 		return tomlFloat |  | ||||||
| 	case reflect.Array, reflect.Slice: |  | ||||||
| 		if typeEqual(tomlHash, tomlArrayType(rv)) { |  | ||||||
| 			return tomlArrayHash |  | ||||||
| 		} |  | ||||||
| 		return tomlArray |  | ||||||
| 	case reflect.Ptr, reflect.Interface: |  | ||||||
| 		return tomlTypeOfGo(rv.Elem()) |  | ||||||
| 	case reflect.String: |  | ||||||
| 		return tomlString |  | ||||||
| 	case reflect.Map: |  | ||||||
| 		return tomlHash |  | ||||||
| 	case reflect.Struct: |  | ||||||
| 		switch rv.Interface().(type) { |  | ||||||
| 		case time.Time: |  | ||||||
| 			return tomlDatetime |  | ||||||
| 		case TextMarshaler: |  | ||||||
| 			return tomlString |  | ||||||
| 		default: |  | ||||||
| 			return tomlHash |  | ||||||
| 		} |  | ||||||
| 	default: |  | ||||||
| 		panic("unexpected reflect.Kind: " + rv.Kind().String()) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // tomlArrayType returns the element type of a TOML array. The type returned |  | ||||||
| // may be nil if it cannot be determined (e.g., a nil slice or a zero length |  | ||||||
| // slize). This function may also panic if it finds a type that cannot be |  | ||||||
| // expressed in TOML (such as nil elements, heterogeneous arrays or directly |  | ||||||
| // nested arrays of tables). |  | ||||||
| func tomlArrayType(rv reflect.Value) tomlType { |  | ||||||
| 	if isNil(rv) || !rv.IsValid() || rv.Len() == 0 { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	firstType := tomlTypeOfGo(rv.Index(0)) |  | ||||||
| 	if firstType == nil { |  | ||||||
| 		encPanic(errArrayNilElement) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	rvlen := rv.Len() |  | ||||||
| 	for i := 1; i < rvlen; i++ { |  | ||||||
| 		elem := rv.Index(i) |  | ||||||
| 		switch elemType := tomlTypeOfGo(elem); { |  | ||||||
| 		case elemType == nil: |  | ||||||
| 			encPanic(errArrayNilElement) |  | ||||||
| 		case !typeEqual(firstType, elemType): |  | ||||||
| 			encPanic(errArrayMixedElementTypes) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	// If we have a nested array, then we must make sure that the nested |  | ||||||
| 	// array contains ONLY primitives. |  | ||||||
| 	// This checks arbitrarily nested arrays. |  | ||||||
| 	if typeEqual(firstType, tomlArray) || typeEqual(firstType, tomlArrayHash) { |  | ||||||
| 		nest := tomlArrayType(eindirect(rv.Index(0))) |  | ||||||
| 		if typeEqual(nest, tomlHash) || typeEqual(nest, tomlArrayHash) { |  | ||||||
| 			encPanic(errArrayNoTable) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return firstType |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type tagOptions struct { |  | ||||||
| 	skip      bool // "-" |  | ||||||
| 	name      string |  | ||||||
| 	omitempty bool |  | ||||||
| 	omitzero  bool |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func getOptions(tag reflect.StructTag) tagOptions { |  | ||||||
| 	t := tag.Get("toml") |  | ||||||
| 	if t == "-" { |  | ||||||
| 		return tagOptions{skip: true} |  | ||||||
| 	} |  | ||||||
| 	var opts tagOptions |  | ||||||
| 	parts := strings.Split(t, ",") |  | ||||||
| 	opts.name = parts[0] |  | ||||||
| 	for _, s := range parts[1:] { |  | ||||||
| 		switch s { |  | ||||||
| 		case "omitempty": |  | ||||||
| 			opts.omitempty = true |  | ||||||
| 		case "omitzero": |  | ||||||
| 			opts.omitzero = true |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return opts |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isZero(rv reflect.Value) bool { |  | ||||||
| 	switch rv.Kind() { |  | ||||||
| 	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: |  | ||||||
| 		return rv.Int() == 0 |  | ||||||
| 	case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |  | ||||||
| 		return rv.Uint() == 0 |  | ||||||
| 	case reflect.Float32, reflect.Float64: |  | ||||||
| 		return rv.Float() == 0.0 |  | ||||||
| 	} |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isEmpty(rv reflect.Value) bool { |  | ||||||
| 	switch rv.Kind() { |  | ||||||
| 	case reflect.Array, reflect.Slice, reflect.Map, reflect.String: |  | ||||||
| 		return rv.Len() == 0 |  | ||||||
| 	case reflect.Bool: |  | ||||||
| 		return !rv.Bool() |  | ||||||
| 	} |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) newline() { |  | ||||||
| 	if enc.hasWritten { |  | ||||||
| 		enc.wf("\n") |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) keyEqElement(key Key, val reflect.Value) { |  | ||||||
| 	if len(key) == 0 { |  | ||||||
| 		encPanic(errNoKey) |  | ||||||
| 	} |  | ||||||
| 	panicIfInvalidKey(key) |  | ||||||
| 	enc.wf("%s%s = ", enc.indentStr(key), key.maybeQuoted(len(key)-1)) |  | ||||||
| 	enc.eElement(val) |  | ||||||
| 	enc.newline() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) wf(format string, v ...interface{}) { |  | ||||||
| 	if _, err := fmt.Fprintf(enc.w, format, v...); err != nil { |  | ||||||
| 		encPanic(err) |  | ||||||
| 	} |  | ||||||
| 	enc.hasWritten = true |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (enc *Encoder) indentStr(key Key) string { |  | ||||||
| 	return strings.Repeat(enc.Indent, len(key)-1) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func encPanic(err error) { |  | ||||||
| 	panic(tomlEncodeError{err}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func eindirect(v reflect.Value) reflect.Value { |  | ||||||
| 	switch v.Kind() { |  | ||||||
| 	case reflect.Ptr, reflect.Interface: |  | ||||||
| 		return eindirect(v.Elem()) |  | ||||||
| 	default: |  | ||||||
| 		return v |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isNil(rv reflect.Value) bool { |  | ||||||
| 	switch rv.Kind() { |  | ||||||
| 	case reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice: |  | ||||||
| 		return rv.IsNil() |  | ||||||
| 	default: |  | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func panicIfInvalidKey(key Key) { |  | ||||||
| 	for _, k := range key { |  | ||||||
| 		if len(k) == 0 { |  | ||||||
| 			encPanic(e("Key '%s' is not a valid table name. Key names "+ |  | ||||||
| 				"cannot be empty.", key.maybeQuotedAll())) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isValidKeyName(s string) bool { |  | ||||||
| 	return len(s) != 0 |  | ||||||
| } |  | ||||||
							
								
								
									
										19
									
								
								vendor/github.com/BurntSushi/toml/encoding_types.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,19 +0,0 @@ | |||||||
| // +build go1.2 |  | ||||||
|  |  | ||||||
| package toml |  | ||||||
|  |  | ||||||
| // In order to support Go 1.1, we define our own TextMarshaler and |  | ||||||
| // TextUnmarshaler types. For Go 1.2+, we just alias them with the |  | ||||||
| // standard library interfaces. |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // TextMarshaler is a synonym for encoding.TextMarshaler. It is defined here |  | ||||||
| // so that Go 1.1 can be supported. |  | ||||||
| type TextMarshaler encoding.TextMarshaler |  | ||||||
|  |  | ||||||
| // TextUnmarshaler is a synonym for encoding.TextUnmarshaler. It is defined |  | ||||||
| // here so that Go 1.1 can be supported. |  | ||||||
| type TextUnmarshaler encoding.TextUnmarshaler |  | ||||||
							
								
								
									
										18
									
								
								vendor/github.com/BurntSushi/toml/encoding_types_1.1.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,18 +0,0 @@ | |||||||
| // +build !go1.2 |  | ||||||
|  |  | ||||||
| package toml |  | ||||||
|  |  | ||||||
| // These interfaces were introduced in Go 1.2, so we add them manually when |  | ||||||
| // compiling for Go 1.1. |  | ||||||
|  |  | ||||||
| // TextMarshaler is a synonym for encoding.TextMarshaler. It is defined here |  | ||||||
| // so that Go 1.1 can be supported. |  | ||||||
| type TextMarshaler interface { |  | ||||||
| 	MarshalText() (text []byte, err error) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // TextUnmarshaler is a synonym for encoding.TextUnmarshaler. It is defined |  | ||||||
| // here so that Go 1.1 can be supported. |  | ||||||
| type TextUnmarshaler interface { |  | ||||||
| 	UnmarshalText(text []byte) error |  | ||||||
| } |  | ||||||
							
								
								
									
										953
									
								
								vendor/github.com/BurntSushi/toml/lex.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,953 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"strings" |  | ||||||
| 	"unicode" |  | ||||||
| 	"unicode/utf8" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type itemType int |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	itemError itemType = iota |  | ||||||
| 	itemNIL            // used in the parser to indicate no type |  | ||||||
| 	itemEOF |  | ||||||
| 	itemText |  | ||||||
| 	itemString |  | ||||||
| 	itemRawString |  | ||||||
| 	itemMultilineString |  | ||||||
| 	itemRawMultilineString |  | ||||||
| 	itemBool |  | ||||||
| 	itemInteger |  | ||||||
| 	itemFloat |  | ||||||
| 	itemDatetime |  | ||||||
| 	itemArray // the start of an array |  | ||||||
| 	itemArrayEnd |  | ||||||
| 	itemTableStart |  | ||||||
| 	itemTableEnd |  | ||||||
| 	itemArrayTableStart |  | ||||||
| 	itemArrayTableEnd |  | ||||||
| 	itemKeyStart |  | ||||||
| 	itemCommentStart |  | ||||||
| 	itemInlineTableStart |  | ||||||
| 	itemInlineTableEnd |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	eof              = 0 |  | ||||||
| 	comma            = ',' |  | ||||||
| 	tableStart       = '[' |  | ||||||
| 	tableEnd         = ']' |  | ||||||
| 	arrayTableStart  = '[' |  | ||||||
| 	arrayTableEnd    = ']' |  | ||||||
| 	tableSep         = '.' |  | ||||||
| 	keySep           = '=' |  | ||||||
| 	arrayStart       = '[' |  | ||||||
| 	arrayEnd         = ']' |  | ||||||
| 	commentStart     = '#' |  | ||||||
| 	stringStart      = '"' |  | ||||||
| 	stringEnd        = '"' |  | ||||||
| 	rawStringStart   = '\'' |  | ||||||
| 	rawStringEnd     = '\'' |  | ||||||
| 	inlineTableStart = '{' |  | ||||||
| 	inlineTableEnd   = '}' |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type stateFn func(lx *lexer) stateFn |  | ||||||
|  |  | ||||||
| type lexer struct { |  | ||||||
| 	input string |  | ||||||
| 	start int |  | ||||||
| 	pos   int |  | ||||||
| 	line  int |  | ||||||
| 	state stateFn |  | ||||||
| 	items chan item |  | ||||||
|  |  | ||||||
| 	// Allow for backing up up to three runes. |  | ||||||
| 	// This is necessary because TOML contains 3-rune tokens (""" and '''). |  | ||||||
| 	prevWidths [3]int |  | ||||||
| 	nprev      int // how many of prevWidths are in use |  | ||||||
| 	// If we emit an eof, we can still back up, but it is not OK to call |  | ||||||
| 	// next again. |  | ||||||
| 	atEOF bool |  | ||||||
|  |  | ||||||
| 	// A stack of state functions used to maintain context. |  | ||||||
| 	// The idea is to reuse parts of the state machine in various places. |  | ||||||
| 	// For example, values can appear at the top level or within arbitrarily |  | ||||||
| 	// nested arrays. The last state on the stack is used after a value has |  | ||||||
| 	// been lexed. Similarly for comments. |  | ||||||
| 	stack []stateFn |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type item struct { |  | ||||||
| 	typ  itemType |  | ||||||
| 	val  string |  | ||||||
| 	line int |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) nextItem() item { |  | ||||||
| 	for { |  | ||||||
| 		select { |  | ||||||
| 		case item := <-lx.items: |  | ||||||
| 			return item |  | ||||||
| 		default: |  | ||||||
| 			lx.state = lx.state(lx) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lex(input string) *lexer { |  | ||||||
| 	lx := &lexer{ |  | ||||||
| 		input: input, |  | ||||||
| 		state: lexTop, |  | ||||||
| 		line:  1, |  | ||||||
| 		items: make(chan item, 10), |  | ||||||
| 		stack: make([]stateFn, 0, 10), |  | ||||||
| 	} |  | ||||||
| 	return lx |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) push(state stateFn) { |  | ||||||
| 	lx.stack = append(lx.stack, state) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) pop() stateFn { |  | ||||||
| 	if len(lx.stack) == 0 { |  | ||||||
| 		return lx.errorf("BUG in lexer: no states to pop") |  | ||||||
| 	} |  | ||||||
| 	last := lx.stack[len(lx.stack)-1] |  | ||||||
| 	lx.stack = lx.stack[0 : len(lx.stack)-1] |  | ||||||
| 	return last |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) current() string { |  | ||||||
| 	return lx.input[lx.start:lx.pos] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) emit(typ itemType) { |  | ||||||
| 	lx.items <- item{typ, lx.current(), lx.line} |  | ||||||
| 	lx.start = lx.pos |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) emitTrim(typ itemType) { |  | ||||||
| 	lx.items <- item{typ, strings.TrimSpace(lx.current()), lx.line} |  | ||||||
| 	lx.start = lx.pos |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (lx *lexer) next() (r rune) { |  | ||||||
| 	if lx.atEOF { |  | ||||||
| 		panic("next called after EOF") |  | ||||||
| 	} |  | ||||||
| 	if lx.pos >= len(lx.input) { |  | ||||||
| 		lx.atEOF = true |  | ||||||
| 		return eof |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if lx.input[lx.pos] == '\n' { |  | ||||||
| 		lx.line++ |  | ||||||
| 	} |  | ||||||
| 	lx.prevWidths[2] = lx.prevWidths[1] |  | ||||||
| 	lx.prevWidths[1] = lx.prevWidths[0] |  | ||||||
| 	if lx.nprev < 3 { |  | ||||||
| 		lx.nprev++ |  | ||||||
| 	} |  | ||||||
| 	r, w := utf8.DecodeRuneInString(lx.input[lx.pos:]) |  | ||||||
| 	lx.prevWidths[0] = w |  | ||||||
| 	lx.pos += w |  | ||||||
| 	return r |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ignore skips over the pending input before this point. |  | ||||||
| func (lx *lexer) ignore() { |  | ||||||
| 	lx.start = lx.pos |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // backup steps back one rune. Can be called only twice between calls to next. |  | ||||||
| func (lx *lexer) backup() { |  | ||||||
| 	if lx.atEOF { |  | ||||||
| 		lx.atEOF = false |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	if lx.nprev < 1 { |  | ||||||
| 		panic("backed up too far") |  | ||||||
| 	} |  | ||||||
| 	w := lx.prevWidths[0] |  | ||||||
| 	lx.prevWidths[0] = lx.prevWidths[1] |  | ||||||
| 	lx.prevWidths[1] = lx.prevWidths[2] |  | ||||||
| 	lx.nprev-- |  | ||||||
| 	lx.pos -= w |  | ||||||
| 	if lx.pos < len(lx.input) && lx.input[lx.pos] == '\n' { |  | ||||||
| 		lx.line-- |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // accept consumes the next rune if it's equal to `valid`. |  | ||||||
| func (lx *lexer) accept(valid rune) bool { |  | ||||||
| 	if lx.next() == valid { |  | ||||||
| 		return true |  | ||||||
| 	} |  | ||||||
| 	lx.backup() |  | ||||||
| 	return false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // peek returns but does not consume the next rune in the input. |  | ||||||
| func (lx *lexer) peek() rune { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	lx.backup() |  | ||||||
| 	return r |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // skip ignores all input that matches the given predicate. |  | ||||||
| func (lx *lexer) skip(pred func(rune) bool) { |  | ||||||
| 	for { |  | ||||||
| 		r := lx.next() |  | ||||||
| 		if pred(r) { |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		lx.backup() |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // errorf stops all lexing by emitting an error and returning `nil`. |  | ||||||
| // Note that any value that is a character is escaped if it's a special |  | ||||||
| // character (newlines, tabs, etc.). |  | ||||||
| func (lx *lexer) errorf(format string, values ...interface{}) stateFn { |  | ||||||
| 	lx.items <- item{ |  | ||||||
| 		itemError, |  | ||||||
| 		fmt.Sprintf(format, values...), |  | ||||||
| 		lx.line, |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexTop consumes elements at the top level of TOML data. |  | ||||||
| func lexTop(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isWhitespace(r) || isNL(r) { |  | ||||||
| 		return lexSkip(lx, lexTop) |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case commentStart: |  | ||||||
| 		lx.push(lexTop) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case tableStart: |  | ||||||
| 		return lexTableStart |  | ||||||
| 	case eof: |  | ||||||
| 		if lx.pos > lx.start { |  | ||||||
| 			return lx.errorf("unexpected EOF") |  | ||||||
| 		} |  | ||||||
| 		lx.emit(itemEOF) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// At this point, the only valid item can be a key, so we back up |  | ||||||
| 	// and let the key lexer do the rest. |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.push(lexTopEnd) |  | ||||||
| 	return lexKeyStart |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexTopEnd is entered whenever a top-level item has been consumed. (A value |  | ||||||
| // or a table.) It must see only whitespace, and will turn back to lexTop |  | ||||||
| // upon a newline. If it sees EOF, it will quit the lexer successfully. |  | ||||||
| func lexTopEnd(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case r == commentStart: |  | ||||||
| 		// a comment will read to a newline for us. |  | ||||||
| 		lx.push(lexTop) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexTopEnd |  | ||||||
| 	case isNL(r): |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lexTop |  | ||||||
| 	case r == eof: |  | ||||||
| 		lx.emit(itemEOF) |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("expected a top-level item to end with a newline, "+ |  | ||||||
| 		"comment, or EOF, but got %q instead", r) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexTable lexes the beginning of a table. Namely, it makes sure that |  | ||||||
| // it starts with a character other than '.' and ']'. |  | ||||||
| // It assumes that '[' has already been consumed. |  | ||||||
| // It also handles the case that this is an item in an array of tables. |  | ||||||
| // e.g., '[[name]]'. |  | ||||||
| func lexTableStart(lx *lexer) stateFn { |  | ||||||
| 	if lx.peek() == arrayTableStart { |  | ||||||
| 		lx.next() |  | ||||||
| 		lx.emit(itemArrayTableStart) |  | ||||||
| 		lx.push(lexArrayTableEnd) |  | ||||||
| 	} else { |  | ||||||
| 		lx.emit(itemTableStart) |  | ||||||
| 		lx.push(lexTableEnd) |  | ||||||
| 	} |  | ||||||
| 	return lexTableNameStart |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexTableEnd(lx *lexer) stateFn { |  | ||||||
| 	lx.emit(itemTableEnd) |  | ||||||
| 	return lexTopEnd |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexArrayTableEnd(lx *lexer) stateFn { |  | ||||||
| 	if r := lx.next(); r != arrayTableEnd { |  | ||||||
| 		return lx.errorf("expected end of table array name delimiter %q, "+ |  | ||||||
| 			"but got %q instead", arrayTableEnd, r) |  | ||||||
| 	} |  | ||||||
| 	lx.emit(itemArrayTableEnd) |  | ||||||
| 	return lexTopEnd |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexTableNameStart(lx *lexer) stateFn { |  | ||||||
| 	lx.skip(isWhitespace) |  | ||||||
| 	switch r := lx.peek(); { |  | ||||||
| 	case r == tableEnd || r == eof: |  | ||||||
| 		return lx.errorf("unexpected end of table name " + |  | ||||||
| 			"(table names cannot be empty)") |  | ||||||
| 	case r == tableSep: |  | ||||||
| 		return lx.errorf("unexpected table separator " + |  | ||||||
| 			"(table names cannot be empty)") |  | ||||||
| 	case r == stringStart || r == rawStringStart: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		lx.push(lexTableNameEnd) |  | ||||||
| 		return lexValue // reuse string lexing |  | ||||||
| 	default: |  | ||||||
| 		return lexBareTableName |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexBareTableName lexes the name of a table. It assumes that at least one |  | ||||||
| // valid character for the table has already been read. |  | ||||||
| func lexBareTableName(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isBareKeyChar(r) { |  | ||||||
| 		return lexBareTableName |  | ||||||
| 	} |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.emit(itemText) |  | ||||||
| 	return lexTableNameEnd |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexTableNameEnd reads the end of a piece of a table name, optionally |  | ||||||
| // consuming whitespace. |  | ||||||
| func lexTableNameEnd(lx *lexer) stateFn { |  | ||||||
| 	lx.skip(isWhitespace) |  | ||||||
| 	switch r := lx.next(); { |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexTableNameEnd |  | ||||||
| 	case r == tableSep: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lexTableNameStart |  | ||||||
| 	case r == tableEnd: |  | ||||||
| 		return lx.pop() |  | ||||||
| 	default: |  | ||||||
| 		return lx.errorf("expected '.' or ']' to end table name, "+ |  | ||||||
| 			"but got %q instead", r) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexKeyStart consumes a key name up until the first non-whitespace character. |  | ||||||
| // lexKeyStart will ignore whitespace. |  | ||||||
| func lexKeyStart(lx *lexer) stateFn { |  | ||||||
| 	r := lx.peek() |  | ||||||
| 	switch { |  | ||||||
| 	case r == keySep: |  | ||||||
| 		return lx.errorf("unexpected key separator %q", keySep) |  | ||||||
| 	case isWhitespace(r) || isNL(r): |  | ||||||
| 		lx.next() |  | ||||||
| 		return lexSkip(lx, lexKeyStart) |  | ||||||
| 	case r == stringStart || r == rawStringStart: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		lx.emit(itemKeyStart) |  | ||||||
| 		lx.push(lexKeyEnd) |  | ||||||
| 		return lexValue // reuse string lexing |  | ||||||
| 	default: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		lx.emit(itemKeyStart) |  | ||||||
| 		return lexBareKey |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexBareKey consumes the text of a bare key. Assumes that the first character |  | ||||||
| // (which is not whitespace) has not yet been consumed. |  | ||||||
| func lexBareKey(lx *lexer) stateFn { |  | ||||||
| 	switch r := lx.next(); { |  | ||||||
| 	case isBareKeyChar(r): |  | ||||||
| 		return lexBareKey |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		lx.backup() |  | ||||||
| 		lx.emit(itemText) |  | ||||||
| 		return lexKeyEnd |  | ||||||
| 	case r == keySep: |  | ||||||
| 		lx.backup() |  | ||||||
| 		lx.emit(itemText) |  | ||||||
| 		return lexKeyEnd |  | ||||||
| 	default: |  | ||||||
| 		return lx.errorf("bare keys cannot contain %q", r) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexKeyEnd consumes the end of a key and trims whitespace (up to the key |  | ||||||
| // separator). |  | ||||||
| func lexKeyEnd(lx *lexer) stateFn { |  | ||||||
| 	switch r := lx.next(); { |  | ||||||
| 	case r == keySep: |  | ||||||
| 		return lexSkip(lx, lexValue) |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexSkip(lx, lexKeyEnd) |  | ||||||
| 	default: |  | ||||||
| 		return lx.errorf("expected key separator %q, but got %q instead", |  | ||||||
| 			keySep, r) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexValue starts the consumption of a value anywhere a value is expected. |  | ||||||
| // lexValue will ignore whitespace. |  | ||||||
| // After a value is lexed, the last state on the next is popped and returned. |  | ||||||
| func lexValue(lx *lexer) stateFn { |  | ||||||
| 	// We allow whitespace to precede a value, but NOT newlines. |  | ||||||
| 	// In array syntax, the array states are responsible for ignoring newlines. |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexSkip(lx, lexValue) |  | ||||||
| 	case isDigit(r): |  | ||||||
| 		lx.backup() // avoid an extra state and use the same as above |  | ||||||
| 		return lexNumberOrDateStart |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case arrayStart: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		lx.emit(itemArray) |  | ||||||
| 		return lexArrayValue |  | ||||||
| 	case inlineTableStart: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		lx.emit(itemInlineTableStart) |  | ||||||
| 		return lexInlineTableValue |  | ||||||
| 	case stringStart: |  | ||||||
| 		if lx.accept(stringStart) { |  | ||||||
| 			if lx.accept(stringStart) { |  | ||||||
| 				lx.ignore() // Ignore """ |  | ||||||
| 				return lexMultilineString |  | ||||||
| 			} |  | ||||||
| 			lx.backup() |  | ||||||
| 		} |  | ||||||
| 		lx.ignore() // ignore the '"' |  | ||||||
| 		return lexString |  | ||||||
| 	case rawStringStart: |  | ||||||
| 		if lx.accept(rawStringStart) { |  | ||||||
| 			if lx.accept(rawStringStart) { |  | ||||||
| 				lx.ignore() // Ignore """ |  | ||||||
| 				return lexMultilineRawString |  | ||||||
| 			} |  | ||||||
| 			lx.backup() |  | ||||||
| 		} |  | ||||||
| 		lx.ignore() // ignore the "'" |  | ||||||
| 		return lexRawString |  | ||||||
| 	case '+', '-': |  | ||||||
| 		return lexNumberStart |  | ||||||
| 	case '.': // special error case, be kind to users |  | ||||||
| 		return lx.errorf("floats must start with a digit, not '.'") |  | ||||||
| 	} |  | ||||||
| 	if unicode.IsLetter(r) { |  | ||||||
| 		// Be permissive here; lexBool will give a nice error if the |  | ||||||
| 		// user wrote something like |  | ||||||
| 		//   x = foo |  | ||||||
| 		// (i.e. not 'true' or 'false' but is something else word-like.) |  | ||||||
| 		lx.backup() |  | ||||||
| 		return lexBool |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("expected value but found %q instead", r) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexArrayValue consumes one value in an array. It assumes that '[' or ',' |  | ||||||
| // have already been consumed. All whitespace and newlines are ignored. |  | ||||||
| func lexArrayValue(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case isWhitespace(r) || isNL(r): |  | ||||||
| 		return lexSkip(lx, lexArrayValue) |  | ||||||
| 	case r == commentStart: |  | ||||||
| 		lx.push(lexArrayValue) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case r == comma: |  | ||||||
| 		return lx.errorf("unexpected comma") |  | ||||||
| 	case r == arrayEnd: |  | ||||||
| 		// NOTE(caleb): The spec isn't clear about whether you can have |  | ||||||
| 		// a trailing comma or not, so we'll allow it. |  | ||||||
| 		return lexArrayEnd |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.push(lexArrayValueEnd) |  | ||||||
| 	return lexValue |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexArrayValueEnd consumes everything between the end of an array value and |  | ||||||
| // the next value (or the end of the array): it ignores whitespace and newlines |  | ||||||
| // and expects either a ',' or a ']'. |  | ||||||
| func lexArrayValueEnd(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case isWhitespace(r) || isNL(r): |  | ||||||
| 		return lexSkip(lx, lexArrayValueEnd) |  | ||||||
| 	case r == commentStart: |  | ||||||
| 		lx.push(lexArrayValueEnd) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case r == comma: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lexArrayValue // move on to the next value |  | ||||||
| 	case r == arrayEnd: |  | ||||||
| 		return lexArrayEnd |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf( |  | ||||||
| 		"expected a comma or array terminator %q, but got %q instead", |  | ||||||
| 		arrayEnd, r, |  | ||||||
| 	) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexArrayEnd finishes the lexing of an array. |  | ||||||
| // It assumes that a ']' has just been consumed. |  | ||||||
| func lexArrayEnd(lx *lexer) stateFn { |  | ||||||
| 	lx.ignore() |  | ||||||
| 	lx.emit(itemArrayEnd) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexInlineTableValue consumes one key/value pair in an inline table. |  | ||||||
| // It assumes that '{' or ',' have already been consumed. Whitespace is ignored. |  | ||||||
| func lexInlineTableValue(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexSkip(lx, lexInlineTableValue) |  | ||||||
| 	case isNL(r): |  | ||||||
| 		return lx.errorf("newlines not allowed within inline tables") |  | ||||||
| 	case r == commentStart: |  | ||||||
| 		lx.push(lexInlineTableValue) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case r == comma: |  | ||||||
| 		return lx.errorf("unexpected comma") |  | ||||||
| 	case r == inlineTableEnd: |  | ||||||
| 		return lexInlineTableEnd |  | ||||||
| 	} |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.push(lexInlineTableValueEnd) |  | ||||||
| 	return lexKeyStart |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexInlineTableValueEnd consumes everything between the end of an inline table |  | ||||||
| // key/value pair and the next pair (or the end of the table): |  | ||||||
| // it ignores whitespace and expects either a ',' or a '}'. |  | ||||||
| func lexInlineTableValueEnd(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case isWhitespace(r): |  | ||||||
| 		return lexSkip(lx, lexInlineTableValueEnd) |  | ||||||
| 	case isNL(r): |  | ||||||
| 		return lx.errorf("newlines not allowed within inline tables") |  | ||||||
| 	case r == commentStart: |  | ||||||
| 		lx.push(lexInlineTableValueEnd) |  | ||||||
| 		return lexCommentStart |  | ||||||
| 	case r == comma: |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lexInlineTableValue |  | ||||||
| 	case r == inlineTableEnd: |  | ||||||
| 		return lexInlineTableEnd |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("expected a comma or an inline table terminator %q, "+ |  | ||||||
| 		"but got %q instead", inlineTableEnd, r) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexInlineTableEnd finishes the lexing of an inline table. |  | ||||||
| // It assumes that a '}' has just been consumed. |  | ||||||
| func lexInlineTableEnd(lx *lexer) stateFn { |  | ||||||
| 	lx.ignore() |  | ||||||
| 	lx.emit(itemInlineTableEnd) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexString consumes the inner contents of a string. It assumes that the |  | ||||||
| // beginning '"' has already been consumed and ignored. |  | ||||||
| func lexString(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case r == eof: |  | ||||||
| 		return lx.errorf("unexpected EOF") |  | ||||||
| 	case isNL(r): |  | ||||||
| 		return lx.errorf("strings cannot contain newlines") |  | ||||||
| 	case r == '\\': |  | ||||||
| 		lx.push(lexString) |  | ||||||
| 		return lexStringEscape |  | ||||||
| 	case r == stringEnd: |  | ||||||
| 		lx.backup() |  | ||||||
| 		lx.emit(itemString) |  | ||||||
| 		lx.next() |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lx.pop() |  | ||||||
| 	} |  | ||||||
| 	return lexString |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexMultilineString consumes the inner contents of a string. It assumes that |  | ||||||
| // the beginning '"""' has already been consumed and ignored. |  | ||||||
| func lexMultilineString(lx *lexer) stateFn { |  | ||||||
| 	switch lx.next() { |  | ||||||
| 	case eof: |  | ||||||
| 		return lx.errorf("unexpected EOF") |  | ||||||
| 	case '\\': |  | ||||||
| 		return lexMultilineStringEscape |  | ||||||
| 	case stringEnd: |  | ||||||
| 		if lx.accept(stringEnd) { |  | ||||||
| 			if lx.accept(stringEnd) { |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.emit(itemMultilineString) |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.ignore() |  | ||||||
| 				return lx.pop() |  | ||||||
| 			} |  | ||||||
| 			lx.backup() |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return lexMultilineString |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexRawString consumes a raw string. Nothing can be escaped in such a string. |  | ||||||
| // It assumes that the beginning "'" has already been consumed and ignored. |  | ||||||
| func lexRawString(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch { |  | ||||||
| 	case r == eof: |  | ||||||
| 		return lx.errorf("unexpected EOF") |  | ||||||
| 	case isNL(r): |  | ||||||
| 		return lx.errorf("strings cannot contain newlines") |  | ||||||
| 	case r == rawStringEnd: |  | ||||||
| 		lx.backup() |  | ||||||
| 		lx.emit(itemRawString) |  | ||||||
| 		lx.next() |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return lx.pop() |  | ||||||
| 	} |  | ||||||
| 	return lexRawString |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexMultilineRawString consumes a raw string. Nothing can be escaped in such |  | ||||||
| // a string. It assumes that the beginning "'''" has already been consumed and |  | ||||||
| // ignored. |  | ||||||
| func lexMultilineRawString(lx *lexer) stateFn { |  | ||||||
| 	switch lx.next() { |  | ||||||
| 	case eof: |  | ||||||
| 		return lx.errorf("unexpected EOF") |  | ||||||
| 	case rawStringEnd: |  | ||||||
| 		if lx.accept(rawStringEnd) { |  | ||||||
| 			if lx.accept(rawStringEnd) { |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.backup() |  | ||||||
| 				lx.emit(itemRawMultilineString) |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.next() |  | ||||||
| 				lx.ignore() |  | ||||||
| 				return lx.pop() |  | ||||||
| 			} |  | ||||||
| 			lx.backup() |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return lexMultilineRawString |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexMultilineStringEscape consumes an escaped character. It assumes that the |  | ||||||
| // preceding '\\' has already been consumed. |  | ||||||
| func lexMultilineStringEscape(lx *lexer) stateFn { |  | ||||||
| 	// Handle the special case first: |  | ||||||
| 	if isNL(lx.next()) { |  | ||||||
| 		return lexMultilineString |  | ||||||
| 	} |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.push(lexMultilineString) |  | ||||||
| 	return lexStringEscape(lx) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexStringEscape(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	switch r { |  | ||||||
| 	case 'b': |  | ||||||
| 		fallthrough |  | ||||||
| 	case 't': |  | ||||||
| 		fallthrough |  | ||||||
| 	case 'n': |  | ||||||
| 		fallthrough |  | ||||||
| 	case 'f': |  | ||||||
| 		fallthrough |  | ||||||
| 	case 'r': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '"': |  | ||||||
| 		fallthrough |  | ||||||
| 	case '\\': |  | ||||||
| 		return lx.pop() |  | ||||||
| 	case 'u': |  | ||||||
| 		return lexShortUnicodeEscape |  | ||||||
| 	case 'U': |  | ||||||
| 		return lexLongUnicodeEscape |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("invalid escape character %q; only the following "+ |  | ||||||
| 		"escape characters are allowed: "+ |  | ||||||
| 		`\b, \t, \n, \f, \r, \", \\, \uXXXX, and \UXXXXXXXX`, r) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexShortUnicodeEscape(lx *lexer) stateFn { |  | ||||||
| 	var r rune |  | ||||||
| 	for i := 0; i < 4; i++ { |  | ||||||
| 		r = lx.next() |  | ||||||
| 		if !isHexadecimal(r) { |  | ||||||
| 			return lx.errorf(`expected four hexadecimal digits after '\u', `+ |  | ||||||
| 				"but got %q instead", lx.current()) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func lexLongUnicodeEscape(lx *lexer) stateFn { |  | ||||||
| 	var r rune |  | ||||||
| 	for i := 0; i < 8; i++ { |  | ||||||
| 		r = lx.next() |  | ||||||
| 		if !isHexadecimal(r) { |  | ||||||
| 			return lx.errorf(`expected eight hexadecimal digits after '\U', `+ |  | ||||||
| 				"but got %q instead", lx.current()) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexNumberOrDateStart consumes either an integer, a float, or datetime. |  | ||||||
| func lexNumberOrDateStart(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isDigit(r) { |  | ||||||
| 		return lexNumberOrDate |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case '_': |  | ||||||
| 		return lexNumber |  | ||||||
| 	case 'e', 'E': |  | ||||||
| 		return lexFloat |  | ||||||
| 	case '.': |  | ||||||
| 		return lx.errorf("floats must start with a digit, not '.'") |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("expected a digit but got %q", r) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexNumberOrDate consumes either an integer, float or datetime. |  | ||||||
| func lexNumberOrDate(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isDigit(r) { |  | ||||||
| 		return lexNumberOrDate |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case '-': |  | ||||||
| 		return lexDatetime |  | ||||||
| 	case '_': |  | ||||||
| 		return lexNumber |  | ||||||
| 	case '.', 'e', 'E': |  | ||||||
| 		return lexFloat |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.emit(itemInteger) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexDatetime consumes a Datetime, to a first approximation. |  | ||||||
| // The parser validates that it matches one of the accepted formats. |  | ||||||
| func lexDatetime(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isDigit(r) { |  | ||||||
| 		return lexDatetime |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case '-', 'T', ':', '.', 'Z': |  | ||||||
| 		return lexDatetime |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.emit(itemDatetime) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexNumberStart consumes either an integer or a float. It assumes that a sign |  | ||||||
| // has already been read, but that *no* digits have been consumed. |  | ||||||
| // lexNumberStart will move to the appropriate integer or float states. |  | ||||||
| func lexNumberStart(lx *lexer) stateFn { |  | ||||||
| 	// We MUST see a digit. Even floats have to start with a digit. |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if !isDigit(r) { |  | ||||||
| 		if r == '.' { |  | ||||||
| 			return lx.errorf("floats must start with a digit, not '.'") |  | ||||||
| 		} |  | ||||||
| 		return lx.errorf("expected a digit but got %q", r) |  | ||||||
| 	} |  | ||||||
| 	return lexNumber |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexNumber consumes an integer or a float after seeing the first digit. |  | ||||||
| func lexNumber(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isDigit(r) { |  | ||||||
| 		return lexNumber |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case '_': |  | ||||||
| 		return lexNumber |  | ||||||
| 	case '.', 'e', 'E': |  | ||||||
| 		return lexFloat |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.emit(itemInteger) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexFloat consumes the elements of a float. It allows any sequence of |  | ||||||
| // float-like characters, so floats emitted by the lexer are only a first |  | ||||||
| // approximation and must be validated by the parser. |  | ||||||
| func lexFloat(lx *lexer) stateFn { |  | ||||||
| 	r := lx.next() |  | ||||||
| 	if isDigit(r) { |  | ||||||
| 		return lexFloat |  | ||||||
| 	} |  | ||||||
| 	switch r { |  | ||||||
| 	case '_', '.', '-', '+', 'e', 'E': |  | ||||||
| 		return lexFloat |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	lx.backup() |  | ||||||
| 	lx.emit(itemFloat) |  | ||||||
| 	return lx.pop() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexBool consumes a bool string: 'true' or 'false. |  | ||||||
| func lexBool(lx *lexer) stateFn { |  | ||||||
| 	var rs []rune |  | ||||||
| 	for { |  | ||||||
| 		r := lx.next() |  | ||||||
| 		if r == eof || isWhitespace(r) || isNL(r) { |  | ||||||
| 			lx.backup() |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 		rs = append(rs, r) |  | ||||||
| 	} |  | ||||||
| 	s := string(rs) |  | ||||||
| 	switch s { |  | ||||||
| 	case "true", "false": |  | ||||||
| 		lx.emit(itemBool) |  | ||||||
| 		return lx.pop() |  | ||||||
| 	} |  | ||||||
| 	return lx.errorf("expected value but found %q instead", s) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexCommentStart begins the lexing of a comment. It will emit |  | ||||||
| // itemCommentStart and consume no characters, passing control to lexComment. |  | ||||||
| func lexCommentStart(lx *lexer) stateFn { |  | ||||||
| 	lx.ignore() |  | ||||||
| 	lx.emit(itemCommentStart) |  | ||||||
| 	return lexComment |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexComment lexes an entire comment. It assumes that '#' has been consumed. |  | ||||||
| // It will consume *up to* the first newline character, and pass control |  | ||||||
| // back to the last state on the stack. |  | ||||||
| func lexComment(lx *lexer) stateFn { |  | ||||||
| 	r := lx.peek() |  | ||||||
| 	if isNL(r) || r == eof { |  | ||||||
| 		lx.emit(itemText) |  | ||||||
| 		return lx.pop() |  | ||||||
| 	} |  | ||||||
| 	lx.next() |  | ||||||
| 	return lexComment |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // lexSkip ignores all slurped input and moves on to the next state. |  | ||||||
| func lexSkip(lx *lexer, nextState stateFn) stateFn { |  | ||||||
| 	return func(lx *lexer) stateFn { |  | ||||||
| 		lx.ignore() |  | ||||||
| 		return nextState |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // isWhitespace returns true if `r` is a whitespace character according |  | ||||||
| // to the spec. |  | ||||||
| func isWhitespace(r rune) bool { |  | ||||||
| 	return r == '\t' || r == ' ' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isNL(r rune) bool { |  | ||||||
| 	return r == '\n' || r == '\r' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isDigit(r rune) bool { |  | ||||||
| 	return r >= '0' && r <= '9' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isHexadecimal(r rune) bool { |  | ||||||
| 	return (r >= '0' && r <= '9') || |  | ||||||
| 		(r >= 'a' && r <= 'f') || |  | ||||||
| 		(r >= 'A' && r <= 'F') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isBareKeyChar(r rune) bool { |  | ||||||
| 	return (r >= 'A' && r <= 'Z') || |  | ||||||
| 		(r >= 'a' && r <= 'z') || |  | ||||||
| 		(r >= '0' && r <= '9') || |  | ||||||
| 		r == '_' || |  | ||||||
| 		r == '-' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (itype itemType) String() string { |  | ||||||
| 	switch itype { |  | ||||||
| 	case itemError: |  | ||||||
| 		return "Error" |  | ||||||
| 	case itemNIL: |  | ||||||
| 		return "NIL" |  | ||||||
| 	case itemEOF: |  | ||||||
| 		return "EOF" |  | ||||||
| 	case itemText: |  | ||||||
| 		return "Text" |  | ||||||
| 	case itemString, itemRawString, itemMultilineString, itemRawMultilineString: |  | ||||||
| 		return "String" |  | ||||||
| 	case itemBool: |  | ||||||
| 		return "Bool" |  | ||||||
| 	case itemInteger: |  | ||||||
| 		return "Integer" |  | ||||||
| 	case itemFloat: |  | ||||||
| 		return "Float" |  | ||||||
| 	case itemDatetime: |  | ||||||
| 		return "DateTime" |  | ||||||
| 	case itemTableStart: |  | ||||||
| 		return "TableStart" |  | ||||||
| 	case itemTableEnd: |  | ||||||
| 		return "TableEnd" |  | ||||||
| 	case itemKeyStart: |  | ||||||
| 		return "KeyStart" |  | ||||||
| 	case itemArray: |  | ||||||
| 		return "Array" |  | ||||||
| 	case itemArrayEnd: |  | ||||||
| 		return "ArrayEnd" |  | ||||||
| 	case itemCommentStart: |  | ||||||
| 		return "CommentStart" |  | ||||||
| 	} |  | ||||||
| 	panic(fmt.Sprintf("BUG: Unknown type '%d'.", int(itype))) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (item item) String() string { |  | ||||||
| 	return fmt.Sprintf("(%s, %s)", item.typ.String(), item.val) |  | ||||||
| } |  | ||||||
							
								
								
									
										592
									
								
								vendor/github.com/BurntSushi/toml/parse.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,592 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"fmt" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
| 	"unicode" |  | ||||||
| 	"unicode/utf8" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type parser struct { |  | ||||||
| 	mapping map[string]interface{} |  | ||||||
| 	types   map[string]tomlType |  | ||||||
| 	lx      *lexer |  | ||||||
|  |  | ||||||
| 	// A list of keys in the order that they appear in the TOML data. |  | ||||||
| 	ordered []Key |  | ||||||
|  |  | ||||||
| 	// the full key for the current hash in scope |  | ||||||
| 	context Key |  | ||||||
|  |  | ||||||
| 	// the base key name for everything except hashes |  | ||||||
| 	currentKey string |  | ||||||
|  |  | ||||||
| 	// rough approximation of line number |  | ||||||
| 	approxLine int |  | ||||||
|  |  | ||||||
| 	// A map of 'key.group.names' to whether they were created implicitly. |  | ||||||
| 	implicits map[string]bool |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type parseError string |  | ||||||
|  |  | ||||||
| func (pe parseError) Error() string { |  | ||||||
| 	return string(pe) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func parse(data string) (p *parser, err error) { |  | ||||||
| 	defer func() { |  | ||||||
| 		if r := recover(); r != nil { |  | ||||||
| 			var ok bool |  | ||||||
| 			if err, ok = r.(parseError); ok { |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 			panic(r) |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	p = &parser{ |  | ||||||
| 		mapping:   make(map[string]interface{}), |  | ||||||
| 		types:     make(map[string]tomlType), |  | ||||||
| 		lx:        lex(data), |  | ||||||
| 		ordered:   make([]Key, 0), |  | ||||||
| 		implicits: make(map[string]bool), |  | ||||||
| 	} |  | ||||||
| 	for { |  | ||||||
| 		item := p.next() |  | ||||||
| 		if item.typ == itemEOF { |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 		p.topLevel(item) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return p, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) panicf(format string, v ...interface{}) { |  | ||||||
| 	msg := fmt.Sprintf("Near line %d (last key parsed '%s'): %s", |  | ||||||
| 		p.approxLine, p.current(), fmt.Sprintf(format, v...)) |  | ||||||
| 	panic(parseError(msg)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) next() item { |  | ||||||
| 	it := p.lx.nextItem() |  | ||||||
| 	if it.typ == itemError { |  | ||||||
| 		p.panicf("%s", it.val) |  | ||||||
| 	} |  | ||||||
| 	return it |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) bug(format string, v ...interface{}) { |  | ||||||
| 	panic(fmt.Sprintf("BUG: "+format+"\n\n", v...)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) expect(typ itemType) item { |  | ||||||
| 	it := p.next() |  | ||||||
| 	p.assertEqual(typ, it.typ) |  | ||||||
| 	return it |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) assertEqual(expected, got itemType) { |  | ||||||
| 	if expected != got { |  | ||||||
| 		p.bug("Expected '%s' but got '%s'.", expected, got) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) topLevel(item item) { |  | ||||||
| 	switch item.typ { |  | ||||||
| 	case itemCommentStart: |  | ||||||
| 		p.approxLine = item.line |  | ||||||
| 		p.expect(itemText) |  | ||||||
| 	case itemTableStart: |  | ||||||
| 		kg := p.next() |  | ||||||
| 		p.approxLine = kg.line |  | ||||||
|  |  | ||||||
| 		var key Key |  | ||||||
| 		for ; kg.typ != itemTableEnd && kg.typ != itemEOF; kg = p.next() { |  | ||||||
| 			key = append(key, p.keyString(kg)) |  | ||||||
| 		} |  | ||||||
| 		p.assertEqual(itemTableEnd, kg.typ) |  | ||||||
|  |  | ||||||
| 		p.establishContext(key, false) |  | ||||||
| 		p.setType("", tomlHash) |  | ||||||
| 		p.ordered = append(p.ordered, key) |  | ||||||
| 	case itemArrayTableStart: |  | ||||||
| 		kg := p.next() |  | ||||||
| 		p.approxLine = kg.line |  | ||||||
|  |  | ||||||
| 		var key Key |  | ||||||
| 		for ; kg.typ != itemArrayTableEnd && kg.typ != itemEOF; kg = p.next() { |  | ||||||
| 			key = append(key, p.keyString(kg)) |  | ||||||
| 		} |  | ||||||
| 		p.assertEqual(itemArrayTableEnd, kg.typ) |  | ||||||
|  |  | ||||||
| 		p.establishContext(key, true) |  | ||||||
| 		p.setType("", tomlArrayHash) |  | ||||||
| 		p.ordered = append(p.ordered, key) |  | ||||||
| 	case itemKeyStart: |  | ||||||
| 		kname := p.next() |  | ||||||
| 		p.approxLine = kname.line |  | ||||||
| 		p.currentKey = p.keyString(kname) |  | ||||||
|  |  | ||||||
| 		val, typ := p.value(p.next()) |  | ||||||
| 		p.setValue(p.currentKey, val) |  | ||||||
| 		p.setType(p.currentKey, typ) |  | ||||||
| 		p.ordered = append(p.ordered, p.context.add(p.currentKey)) |  | ||||||
| 		p.currentKey = "" |  | ||||||
| 	default: |  | ||||||
| 		p.bug("Unexpected type at top level: %s", item.typ) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Gets a string for a key (or part of a key in a table name). |  | ||||||
| func (p *parser) keyString(it item) string { |  | ||||||
| 	switch it.typ { |  | ||||||
| 	case itemText: |  | ||||||
| 		return it.val |  | ||||||
| 	case itemString, itemMultilineString, |  | ||||||
| 		itemRawString, itemRawMultilineString: |  | ||||||
| 		s, _ := p.value(it) |  | ||||||
| 		return s.(string) |  | ||||||
| 	default: |  | ||||||
| 		p.bug("Unexpected key type: %s", it.typ) |  | ||||||
| 		panic("unreachable") |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // value translates an expected value from the lexer into a Go value wrapped |  | ||||||
| // as an empty interface. |  | ||||||
| func (p *parser) value(it item) (interface{}, tomlType) { |  | ||||||
| 	switch it.typ { |  | ||||||
| 	case itemString: |  | ||||||
| 		return p.replaceEscapes(it.val), p.typeOfPrimitive(it) |  | ||||||
| 	case itemMultilineString: |  | ||||||
| 		trimmed := stripFirstNewline(stripEscapedWhitespace(it.val)) |  | ||||||
| 		return p.replaceEscapes(trimmed), p.typeOfPrimitive(it) |  | ||||||
| 	case itemRawString: |  | ||||||
| 		return it.val, p.typeOfPrimitive(it) |  | ||||||
| 	case itemRawMultilineString: |  | ||||||
| 		return stripFirstNewline(it.val), p.typeOfPrimitive(it) |  | ||||||
| 	case itemBool: |  | ||||||
| 		switch it.val { |  | ||||||
| 		case "true": |  | ||||||
| 			return true, p.typeOfPrimitive(it) |  | ||||||
| 		case "false": |  | ||||||
| 			return false, p.typeOfPrimitive(it) |  | ||||||
| 		} |  | ||||||
| 		p.bug("Expected boolean value, but got '%s'.", it.val) |  | ||||||
| 	case itemInteger: |  | ||||||
| 		if !numUnderscoresOK(it.val) { |  | ||||||
| 			p.panicf("Invalid integer %q: underscores must be surrounded by digits", |  | ||||||
| 				it.val) |  | ||||||
| 		} |  | ||||||
| 		val := strings.Replace(it.val, "_", "", -1) |  | ||||||
| 		num, err := strconv.ParseInt(val, 10, 64) |  | ||||||
| 		if err != nil { |  | ||||||
| 			// Distinguish integer values. Normally, it'd be a bug if the lexer |  | ||||||
| 			// provides an invalid integer, but it's possible that the number is |  | ||||||
| 			// out of range of valid values (which the lexer cannot determine). |  | ||||||
| 			// So mark the former as a bug but the latter as a legitimate user |  | ||||||
| 			// error. |  | ||||||
| 			if e, ok := err.(*strconv.NumError); ok && |  | ||||||
| 				e.Err == strconv.ErrRange { |  | ||||||
|  |  | ||||||
| 				p.panicf("Integer '%s' is out of the range of 64-bit "+ |  | ||||||
| 					"signed integers.", it.val) |  | ||||||
| 			} else { |  | ||||||
| 				p.bug("Expected integer value, but got '%s'.", it.val) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return num, p.typeOfPrimitive(it) |  | ||||||
| 	case itemFloat: |  | ||||||
| 		parts := strings.FieldsFunc(it.val, func(r rune) bool { |  | ||||||
| 			switch r { |  | ||||||
| 			case '.', 'e', 'E': |  | ||||||
| 				return true |  | ||||||
| 			} |  | ||||||
| 			return false |  | ||||||
| 		}) |  | ||||||
| 		for _, part := range parts { |  | ||||||
| 			if !numUnderscoresOK(part) { |  | ||||||
| 				p.panicf("Invalid float %q: underscores must be "+ |  | ||||||
| 					"surrounded by digits", it.val) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if !numPeriodsOK(it.val) { |  | ||||||
| 			// As a special case, numbers like '123.' or '1.e2', |  | ||||||
| 			// which are valid as far as Go/strconv are concerned, |  | ||||||
| 			// must be rejected because TOML says that a fractional |  | ||||||
| 			// part consists of '.' followed by 1+ digits. |  | ||||||
| 			p.panicf("Invalid float %q: '.' must be followed "+ |  | ||||||
| 				"by one or more digits", it.val) |  | ||||||
| 		} |  | ||||||
| 		val := strings.Replace(it.val, "_", "", -1) |  | ||||||
| 		num, err := strconv.ParseFloat(val, 64) |  | ||||||
| 		if err != nil { |  | ||||||
| 			if e, ok := err.(*strconv.NumError); ok && |  | ||||||
| 				e.Err == strconv.ErrRange { |  | ||||||
|  |  | ||||||
| 				p.panicf("Float '%s' is out of the range of 64-bit "+ |  | ||||||
| 					"IEEE-754 floating-point numbers.", it.val) |  | ||||||
| 			} else { |  | ||||||
| 				p.panicf("Invalid float value: %q", it.val) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return num, p.typeOfPrimitive(it) |  | ||||||
| 	case itemDatetime: |  | ||||||
| 		var t time.Time |  | ||||||
| 		var ok bool |  | ||||||
| 		var err error |  | ||||||
| 		for _, format := range []string{ |  | ||||||
| 			"2006-01-02T15:04:05Z07:00", |  | ||||||
| 			"2006-01-02T15:04:05", |  | ||||||
| 			"2006-01-02", |  | ||||||
| 		} { |  | ||||||
| 			t, err = time.ParseInLocation(format, it.val, time.Local) |  | ||||||
| 			if err == nil { |  | ||||||
| 				ok = true |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if !ok { |  | ||||||
| 			p.panicf("Invalid TOML Datetime: %q.", it.val) |  | ||||||
| 		} |  | ||||||
| 		return t, p.typeOfPrimitive(it) |  | ||||||
| 	case itemArray: |  | ||||||
| 		array := make([]interface{}, 0) |  | ||||||
| 		types := make([]tomlType, 0) |  | ||||||
|  |  | ||||||
| 		for it = p.next(); it.typ != itemArrayEnd; it = p.next() { |  | ||||||
| 			if it.typ == itemCommentStart { |  | ||||||
| 				p.expect(itemText) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			val, typ := p.value(it) |  | ||||||
| 			array = append(array, val) |  | ||||||
| 			types = append(types, typ) |  | ||||||
| 		} |  | ||||||
| 		return array, p.typeOfArray(types) |  | ||||||
| 	case itemInlineTableStart: |  | ||||||
| 		var ( |  | ||||||
| 			hash         = make(map[string]interface{}) |  | ||||||
| 			outerContext = p.context |  | ||||||
| 			outerKey     = p.currentKey |  | ||||||
| 		) |  | ||||||
|  |  | ||||||
| 		p.context = append(p.context, p.currentKey) |  | ||||||
| 		p.currentKey = "" |  | ||||||
| 		for it := p.next(); it.typ != itemInlineTableEnd; it = p.next() { |  | ||||||
| 			if it.typ != itemKeyStart { |  | ||||||
| 				p.bug("Expected key start but instead found %q, around line %d", |  | ||||||
| 					it.val, p.approxLine) |  | ||||||
| 			} |  | ||||||
| 			if it.typ == itemCommentStart { |  | ||||||
| 				p.expect(itemText) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			// retrieve key |  | ||||||
| 			k := p.next() |  | ||||||
| 			p.approxLine = k.line |  | ||||||
| 			kname := p.keyString(k) |  | ||||||
|  |  | ||||||
| 			// retrieve value |  | ||||||
| 			p.currentKey = kname |  | ||||||
| 			val, typ := p.value(p.next()) |  | ||||||
| 			// make sure we keep metadata up to date |  | ||||||
| 			p.setType(kname, typ) |  | ||||||
| 			p.ordered = append(p.ordered, p.context.add(p.currentKey)) |  | ||||||
| 			hash[kname] = val |  | ||||||
| 		} |  | ||||||
| 		p.context = outerContext |  | ||||||
| 		p.currentKey = outerKey |  | ||||||
| 		return hash, tomlHash |  | ||||||
| 	} |  | ||||||
| 	p.bug("Unexpected value type: %s", it.typ) |  | ||||||
| 	panic("unreachable") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // numUnderscoresOK checks whether each underscore in s is surrounded by |  | ||||||
| // characters that are not underscores. |  | ||||||
| func numUnderscoresOK(s string) bool { |  | ||||||
| 	accept := false |  | ||||||
| 	for _, r := range s { |  | ||||||
| 		if r == '_' { |  | ||||||
| 			if !accept { |  | ||||||
| 				return false |  | ||||||
| 			} |  | ||||||
| 			accept = false |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		accept = true |  | ||||||
| 	} |  | ||||||
| 	return accept |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // numPeriodsOK checks whether every period in s is followed by a digit. |  | ||||||
| func numPeriodsOK(s string) bool { |  | ||||||
| 	period := false |  | ||||||
| 	for _, r := range s { |  | ||||||
| 		if period && !isDigit(r) { |  | ||||||
| 			return false |  | ||||||
| 		} |  | ||||||
| 		period = r == '.' |  | ||||||
| 	} |  | ||||||
| 	return !period |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // establishContext sets the current context of the parser, |  | ||||||
| // where the context is either a hash or an array of hashes. Which one is |  | ||||||
| // set depends on the value of the `array` parameter. |  | ||||||
| // |  | ||||||
| // Establishing the context also makes sure that the key isn't a duplicate, and |  | ||||||
| // will create implicit hashes automatically. |  | ||||||
| func (p *parser) establishContext(key Key, array bool) { |  | ||||||
| 	var ok bool |  | ||||||
|  |  | ||||||
| 	// Always start at the top level and drill down for our context. |  | ||||||
| 	hashContext := p.mapping |  | ||||||
| 	keyContext := make(Key, 0) |  | ||||||
|  |  | ||||||
| 	// We only need implicit hashes for key[0:-1] |  | ||||||
| 	for _, k := range key[0 : len(key)-1] { |  | ||||||
| 		_, ok = hashContext[k] |  | ||||||
| 		keyContext = append(keyContext, k) |  | ||||||
|  |  | ||||||
| 		// No key? Make an implicit hash and move on. |  | ||||||
| 		if !ok { |  | ||||||
| 			p.addImplicit(keyContext) |  | ||||||
| 			hashContext[k] = make(map[string]interface{}) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// If the hash context is actually an array of tables, then set |  | ||||||
| 		// the hash context to the last element in that array. |  | ||||||
| 		// |  | ||||||
| 		// Otherwise, it better be a table, since this MUST be a key group (by |  | ||||||
| 		// virtue of it not being the last element in a key). |  | ||||||
| 		switch t := hashContext[k].(type) { |  | ||||||
| 		case []map[string]interface{}: |  | ||||||
| 			hashContext = t[len(t)-1] |  | ||||||
| 		case map[string]interface{}: |  | ||||||
| 			hashContext = t |  | ||||||
| 		default: |  | ||||||
| 			p.panicf("Key '%s' was already created as a hash.", keyContext) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	p.context = keyContext |  | ||||||
| 	if array { |  | ||||||
| 		// If this is the first element for this array, then allocate a new |  | ||||||
| 		// list of tables for it. |  | ||||||
| 		k := key[len(key)-1] |  | ||||||
| 		if _, ok := hashContext[k]; !ok { |  | ||||||
| 			hashContext[k] = make([]map[string]interface{}, 0, 5) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Add a new table. But make sure the key hasn't already been used |  | ||||||
| 		// for something else. |  | ||||||
| 		if hash, ok := hashContext[k].([]map[string]interface{}); ok { |  | ||||||
| 			hashContext[k] = append(hash, make(map[string]interface{})) |  | ||||||
| 		} else { |  | ||||||
| 			p.panicf("Key '%s' was already created and cannot be used as "+ |  | ||||||
| 				"an array.", keyContext) |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		p.setValue(key[len(key)-1], make(map[string]interface{})) |  | ||||||
| 	} |  | ||||||
| 	p.context = append(p.context, key[len(key)-1]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // setValue sets the given key to the given value in the current context. |  | ||||||
| // It will make sure that the key hasn't already been defined, account for |  | ||||||
| // implicit key groups. |  | ||||||
| func (p *parser) setValue(key string, value interface{}) { |  | ||||||
| 	var tmpHash interface{} |  | ||||||
| 	var ok bool |  | ||||||
|  |  | ||||||
| 	hash := p.mapping |  | ||||||
| 	keyContext := make(Key, 0) |  | ||||||
| 	for _, k := range p.context { |  | ||||||
| 		keyContext = append(keyContext, k) |  | ||||||
| 		if tmpHash, ok = hash[k]; !ok { |  | ||||||
| 			p.bug("Context for key '%s' has not been established.", keyContext) |  | ||||||
| 		} |  | ||||||
| 		switch t := tmpHash.(type) { |  | ||||||
| 		case []map[string]interface{}: |  | ||||||
| 			// The context is a table of hashes. Pick the most recent table |  | ||||||
| 			// defined as the current hash. |  | ||||||
| 			hash = t[len(t)-1] |  | ||||||
| 		case map[string]interface{}: |  | ||||||
| 			hash = t |  | ||||||
| 		default: |  | ||||||
| 			p.bug("Expected hash to have type 'map[string]interface{}', but "+ |  | ||||||
| 				"it has '%T' instead.", tmpHash) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	keyContext = append(keyContext, key) |  | ||||||
|  |  | ||||||
| 	if _, ok := hash[key]; ok { |  | ||||||
| 		// Typically, if the given key has already been set, then we have |  | ||||||
| 		// to raise an error since duplicate keys are disallowed. However, |  | ||||||
| 		// it's possible that a key was previously defined implicitly. In this |  | ||||||
| 		// case, it is allowed to be redefined concretely. (See the |  | ||||||
| 		// `tests/valid/implicit-and-explicit-after.toml` test in `toml-test`.) |  | ||||||
| 		// |  | ||||||
| 		// But we have to make sure to stop marking it as an implicit. (So that |  | ||||||
| 		// another redefinition provokes an error.) |  | ||||||
| 		// |  | ||||||
| 		// Note that since it has already been defined (as a hash), we don't |  | ||||||
| 		// want to overwrite it. So our business is done. |  | ||||||
| 		if p.isImplicit(keyContext) { |  | ||||||
| 			p.removeImplicit(keyContext) |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// Otherwise, we have a concrete key trying to override a previous |  | ||||||
| 		// key, which is *always* wrong. |  | ||||||
| 		p.panicf("Key '%s' has already been defined.", keyContext) |  | ||||||
| 	} |  | ||||||
| 	hash[key] = value |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // setType sets the type of a particular value at a given key. |  | ||||||
| // It should be called immediately AFTER setValue. |  | ||||||
| // |  | ||||||
| // Note that if `key` is empty, then the type given will be applied to the |  | ||||||
| // current context (which is either a table or an array of tables). |  | ||||||
| func (p *parser) setType(key string, typ tomlType) { |  | ||||||
| 	keyContext := make(Key, 0, len(p.context)+1) |  | ||||||
| 	for _, k := range p.context { |  | ||||||
| 		keyContext = append(keyContext, k) |  | ||||||
| 	} |  | ||||||
| 	if len(key) > 0 { // allow type setting for hashes |  | ||||||
| 		keyContext = append(keyContext, key) |  | ||||||
| 	} |  | ||||||
| 	p.types[keyContext.String()] = typ |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // addImplicit sets the given Key as having been created implicitly. |  | ||||||
| func (p *parser) addImplicit(key Key) { |  | ||||||
| 	p.implicits[key.String()] = true |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // removeImplicit stops tagging the given key as having been implicitly |  | ||||||
| // created. |  | ||||||
| func (p *parser) removeImplicit(key Key) { |  | ||||||
| 	p.implicits[key.String()] = false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // isImplicit returns true if the key group pointed to by the key was created |  | ||||||
| // implicitly. |  | ||||||
| func (p *parser) isImplicit(key Key) bool { |  | ||||||
| 	return p.implicits[key.String()] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // current returns the full key name of the current context. |  | ||||||
| func (p *parser) current() string { |  | ||||||
| 	if len(p.currentKey) == 0 { |  | ||||||
| 		return p.context.String() |  | ||||||
| 	} |  | ||||||
| 	if len(p.context) == 0 { |  | ||||||
| 		return p.currentKey |  | ||||||
| 	} |  | ||||||
| 	return fmt.Sprintf("%s.%s", p.context, p.currentKey) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func stripFirstNewline(s string) string { |  | ||||||
| 	if len(s) == 0 || s[0] != '\n' { |  | ||||||
| 		return s |  | ||||||
| 	} |  | ||||||
| 	return s[1:] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func stripEscapedWhitespace(s string) string { |  | ||||||
| 	esc := strings.Split(s, "\\\n") |  | ||||||
| 	if len(esc) > 1 { |  | ||||||
| 		for i := 1; i < len(esc); i++ { |  | ||||||
| 			esc[i] = strings.TrimLeftFunc(esc[i], unicode.IsSpace) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return strings.Join(esc, "") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) replaceEscapes(str string) string { |  | ||||||
| 	var replaced []rune |  | ||||||
| 	s := []byte(str) |  | ||||||
| 	r := 0 |  | ||||||
| 	for r < len(s) { |  | ||||||
| 		if s[r] != '\\' { |  | ||||||
| 			c, size := utf8.DecodeRune(s[r:]) |  | ||||||
| 			r += size |  | ||||||
| 			replaced = append(replaced, c) |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		r += 1 |  | ||||||
| 		if r >= len(s) { |  | ||||||
| 			p.bug("Escape sequence at end of string.") |  | ||||||
| 			return "" |  | ||||||
| 		} |  | ||||||
| 		switch s[r] { |  | ||||||
| 		default: |  | ||||||
| 			p.bug("Expected valid escape code after \\, but got %q.", s[r]) |  | ||||||
| 			return "" |  | ||||||
| 		case 'b': |  | ||||||
| 			replaced = append(replaced, rune(0x0008)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case 't': |  | ||||||
| 			replaced = append(replaced, rune(0x0009)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case 'n': |  | ||||||
| 			replaced = append(replaced, rune(0x000A)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case 'f': |  | ||||||
| 			replaced = append(replaced, rune(0x000C)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case 'r': |  | ||||||
| 			replaced = append(replaced, rune(0x000D)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case '"': |  | ||||||
| 			replaced = append(replaced, rune(0x0022)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case '\\': |  | ||||||
| 			replaced = append(replaced, rune(0x005C)) |  | ||||||
| 			r += 1 |  | ||||||
| 		case 'u': |  | ||||||
| 			// At this point, we know we have a Unicode escape of the form |  | ||||||
| 			// `uXXXX` at [r, r+5). (Because the lexer guarantees this |  | ||||||
| 			// for us.) |  | ||||||
| 			escaped := p.asciiEscapeToUnicode(s[r+1 : r+5]) |  | ||||||
| 			replaced = append(replaced, escaped) |  | ||||||
| 			r += 5 |  | ||||||
| 		case 'U': |  | ||||||
| 			// At this point, we know we have a Unicode escape of the form |  | ||||||
| 			// `uXXXX` at [r, r+9). (Because the lexer guarantees this |  | ||||||
| 			// for us.) |  | ||||||
| 			escaped := p.asciiEscapeToUnicode(s[r+1 : r+9]) |  | ||||||
| 			replaced = append(replaced, escaped) |  | ||||||
| 			r += 9 |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return string(replaced) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (p *parser) asciiEscapeToUnicode(bs []byte) rune { |  | ||||||
| 	s := string(bs) |  | ||||||
| 	hex, err := strconv.ParseUint(strings.ToLower(s), 16, 32) |  | ||||||
| 	if err != nil { |  | ||||||
| 		p.bug("Could not parse '%s' as a hexadecimal number, but the "+ |  | ||||||
| 			"lexer claims it's OK: %s", s, err) |  | ||||||
| 	} |  | ||||||
| 	if !utf8.ValidRune(rune(hex)) { |  | ||||||
| 		p.panicf("Escaped character '\\u%s' is not valid UTF-8.", s) |  | ||||||
| 	} |  | ||||||
| 	return rune(hex) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func isStringType(ty itemType) bool { |  | ||||||
| 	return ty == itemString || ty == itemMultilineString || |  | ||||||
| 		ty == itemRawString || ty == itemRawMultilineString |  | ||||||
| } |  | ||||||
							
								
								
									
										1
									
								
								vendor/github.com/BurntSushi/toml/session.vim
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1 +0,0 @@ | |||||||
| au BufWritePost *.go silent!make tags > /dev/null 2>&1 |  | ||||||
							
								
								
									
										91
									
								
								vendor/github.com/BurntSushi/toml/type_check.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,91 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| // tomlType represents any Go type that corresponds to a TOML type. |  | ||||||
| // While the first draft of the TOML spec has a simplistic type system that |  | ||||||
| // probably doesn't need this level of sophistication, we seem to be militating |  | ||||||
| // toward adding real composite types. |  | ||||||
| type tomlType interface { |  | ||||||
| 	typeString() string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // typeEqual accepts any two types and returns true if they are equal. |  | ||||||
| func typeEqual(t1, t2 tomlType) bool { |  | ||||||
| 	if t1 == nil || t2 == nil { |  | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
| 	return t1.typeString() == t2.typeString() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func typeIsHash(t tomlType) bool { |  | ||||||
| 	return typeEqual(t, tomlHash) || typeEqual(t, tomlArrayHash) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type tomlBaseType string |  | ||||||
|  |  | ||||||
| func (btype tomlBaseType) typeString() string { |  | ||||||
| 	return string(btype) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (btype tomlBaseType) String() string { |  | ||||||
| 	return btype.typeString() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	tomlInteger   tomlBaseType = "Integer" |  | ||||||
| 	tomlFloat     tomlBaseType = "Float" |  | ||||||
| 	tomlDatetime  tomlBaseType = "Datetime" |  | ||||||
| 	tomlString    tomlBaseType = "String" |  | ||||||
| 	tomlBool      tomlBaseType = "Bool" |  | ||||||
| 	tomlArray     tomlBaseType = "Array" |  | ||||||
| 	tomlHash      tomlBaseType = "Hash" |  | ||||||
| 	tomlArrayHash tomlBaseType = "ArrayHash" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // typeOfPrimitive returns a tomlType of any primitive value in TOML. |  | ||||||
| // Primitive values are: Integer, Float, Datetime, String and Bool. |  | ||||||
| // |  | ||||||
| // Passing a lexer item other than the following will cause a BUG message |  | ||||||
| // to occur: itemString, itemBool, itemInteger, itemFloat, itemDatetime. |  | ||||||
| func (p *parser) typeOfPrimitive(lexItem item) tomlType { |  | ||||||
| 	switch lexItem.typ { |  | ||||||
| 	case itemInteger: |  | ||||||
| 		return tomlInteger |  | ||||||
| 	case itemFloat: |  | ||||||
| 		return tomlFloat |  | ||||||
| 	case itemDatetime: |  | ||||||
| 		return tomlDatetime |  | ||||||
| 	case itemString: |  | ||||||
| 		return tomlString |  | ||||||
| 	case itemMultilineString: |  | ||||||
| 		return tomlString |  | ||||||
| 	case itemRawString: |  | ||||||
| 		return tomlString |  | ||||||
| 	case itemRawMultilineString: |  | ||||||
| 		return tomlString |  | ||||||
| 	case itemBool: |  | ||||||
| 		return tomlBool |  | ||||||
| 	} |  | ||||||
| 	p.bug("Cannot infer primitive type of lex item '%s'.", lexItem) |  | ||||||
| 	panic("unreachable") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // typeOfArray returns a tomlType for an array given a list of types of its |  | ||||||
| // values. |  | ||||||
| // |  | ||||||
| // In the current spec, if an array is homogeneous, then its type is always |  | ||||||
| // "Array". If the array is not homogeneous, an error is generated. |  | ||||||
| func (p *parser) typeOfArray(types []tomlType) tomlType { |  | ||||||
| 	// Empty arrays are cool. |  | ||||||
| 	if len(types) == 0 { |  | ||||||
| 		return tomlArray |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	theType := types[0] |  | ||||||
| 	for _, t := range types[1:] { |  | ||||||
| 		if !typeEqual(theType, t) { |  | ||||||
| 			p.panicf("Array contains values of type '%s' and '%s', but "+ |  | ||||||
| 				"arrays must be homogeneous.", theType, t) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return tomlArray |  | ||||||
| } |  | ||||||
							
								
								
									
										242
									
								
								vendor/github.com/BurntSushi/toml/type_fields.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,242 +0,0 @@ | |||||||
| package toml |  | ||||||
|  |  | ||||||
| // Struct field handling is adapted from code in encoding/json: |  | ||||||
| // |  | ||||||
| // Copyright 2010 The Go Authors.  All rights reserved. |  | ||||||
| // Use of this source code is governed by a BSD-style |  | ||||||
| // license that can be found in the Go distribution. |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"reflect" |  | ||||||
| 	"sort" |  | ||||||
| 	"sync" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // A field represents a single field found in a struct. |  | ||||||
| type field struct { |  | ||||||
| 	name  string       // the name of the field (`toml` tag included) |  | ||||||
| 	tag   bool         // whether field has a `toml` tag |  | ||||||
| 	index []int        // represents the depth of an anonymous field |  | ||||||
| 	typ   reflect.Type // the type of the field |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // byName sorts field by name, breaking ties with depth, |  | ||||||
| // then breaking ties with "name came from toml tag", then |  | ||||||
| // breaking ties with index sequence. |  | ||||||
| type byName []field |  | ||||||
|  |  | ||||||
| func (x byName) Len() int { return len(x) } |  | ||||||
|  |  | ||||||
| func (x byName) Swap(i, j int) { x[i], x[j] = x[j], x[i] } |  | ||||||
|  |  | ||||||
| func (x byName) Less(i, j int) bool { |  | ||||||
| 	if x[i].name != x[j].name { |  | ||||||
| 		return x[i].name < x[j].name |  | ||||||
| 	} |  | ||||||
| 	if len(x[i].index) != len(x[j].index) { |  | ||||||
| 		return len(x[i].index) < len(x[j].index) |  | ||||||
| 	} |  | ||||||
| 	if x[i].tag != x[j].tag { |  | ||||||
| 		return x[i].tag |  | ||||||
| 	} |  | ||||||
| 	return byIndex(x).Less(i, j) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // byIndex sorts field by index sequence. |  | ||||||
| type byIndex []field |  | ||||||
|  |  | ||||||
| func (x byIndex) Len() int { return len(x) } |  | ||||||
|  |  | ||||||
| func (x byIndex) Swap(i, j int) { x[i], x[j] = x[j], x[i] } |  | ||||||
|  |  | ||||||
| func (x byIndex) Less(i, j int) bool { |  | ||||||
| 	for k, xik := range x[i].index { |  | ||||||
| 		if k >= len(x[j].index) { |  | ||||||
| 			return false |  | ||||||
| 		} |  | ||||||
| 		if xik != x[j].index[k] { |  | ||||||
| 			return xik < x[j].index[k] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return len(x[i].index) < len(x[j].index) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // typeFields returns a list of fields that TOML should recognize for the given |  | ||||||
| // type. The algorithm is breadth-first search over the set of structs to |  | ||||||
| // include - the top struct and then any reachable anonymous structs. |  | ||||||
| func typeFields(t reflect.Type) []field { |  | ||||||
| 	// Anonymous fields to explore at the current level and the next. |  | ||||||
| 	current := []field{} |  | ||||||
| 	next := []field{{typ: t}} |  | ||||||
|  |  | ||||||
| 	// Count of queued names for current level and the next. |  | ||||||
| 	count := map[reflect.Type]int{} |  | ||||||
| 	nextCount := map[reflect.Type]int{} |  | ||||||
|  |  | ||||||
| 	// Types already visited at an earlier level. |  | ||||||
| 	visited := map[reflect.Type]bool{} |  | ||||||
|  |  | ||||||
| 	// Fields found. |  | ||||||
| 	var fields []field |  | ||||||
|  |  | ||||||
| 	for len(next) > 0 { |  | ||||||
| 		current, next = next, current[:0] |  | ||||||
| 		count, nextCount = nextCount, map[reflect.Type]int{} |  | ||||||
|  |  | ||||||
| 		for _, f := range current { |  | ||||||
| 			if visited[f.typ] { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			visited[f.typ] = true |  | ||||||
|  |  | ||||||
| 			// Scan f.typ for fields to include. |  | ||||||
| 			for i := 0; i < f.typ.NumField(); i++ { |  | ||||||
| 				sf := f.typ.Field(i) |  | ||||||
| 				if sf.PkgPath != "" && !sf.Anonymous { // unexported |  | ||||||
| 					continue |  | ||||||
| 				} |  | ||||||
| 				opts := getOptions(sf.Tag) |  | ||||||
| 				if opts.skip { |  | ||||||
| 					continue |  | ||||||
| 				} |  | ||||||
| 				index := make([]int, len(f.index)+1) |  | ||||||
| 				copy(index, f.index) |  | ||||||
| 				index[len(f.index)] = i |  | ||||||
|  |  | ||||||
| 				ft := sf.Type |  | ||||||
| 				if ft.Name() == "" && ft.Kind() == reflect.Ptr { |  | ||||||
| 					// Follow pointer. |  | ||||||
| 					ft = ft.Elem() |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				// Record found field and index sequence. |  | ||||||
| 				if opts.name != "" || !sf.Anonymous || ft.Kind() != reflect.Struct { |  | ||||||
| 					tagged := opts.name != "" |  | ||||||
| 					name := opts.name |  | ||||||
| 					if name == "" { |  | ||||||
| 						name = sf.Name |  | ||||||
| 					} |  | ||||||
| 					fields = append(fields, field{name, tagged, index, ft}) |  | ||||||
| 					if count[f.typ] > 1 { |  | ||||||
| 						// If there were multiple instances, add a second, |  | ||||||
| 						// so that the annihilation code will see a duplicate. |  | ||||||
| 						// It only cares about the distinction between 1 or 2, |  | ||||||
| 						// so don't bother generating any more copies. |  | ||||||
| 						fields = append(fields, fields[len(fields)-1]) |  | ||||||
| 					} |  | ||||||
| 					continue |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				// Record new anonymous struct to explore in next round. |  | ||||||
| 				nextCount[ft]++ |  | ||||||
| 				if nextCount[ft] == 1 { |  | ||||||
| 					f := field{name: ft.Name(), index: index, typ: ft} |  | ||||||
| 					next = append(next, f) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	sort.Sort(byName(fields)) |  | ||||||
|  |  | ||||||
| 	// Delete all fields that are hidden by the Go rules for embedded fields, |  | ||||||
| 	// except that fields with TOML tags are promoted. |  | ||||||
|  |  | ||||||
| 	// The fields are sorted in primary order of name, secondary order |  | ||||||
| 	// of field index length. Loop over names; for each name, delete |  | ||||||
| 	// hidden fields by choosing the one dominant field that survives. |  | ||||||
| 	out := fields[:0] |  | ||||||
| 	for advance, i := 0, 0; i < len(fields); i += advance { |  | ||||||
| 		// One iteration per name. |  | ||||||
| 		// Find the sequence of fields with the name of this first field. |  | ||||||
| 		fi := fields[i] |  | ||||||
| 		name := fi.name |  | ||||||
| 		for advance = 1; i+advance < len(fields); advance++ { |  | ||||||
| 			fj := fields[i+advance] |  | ||||||
| 			if fj.name != name { |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if advance == 1 { // Only one field with this name |  | ||||||
| 			out = append(out, fi) |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		dominant, ok := dominantField(fields[i : i+advance]) |  | ||||||
| 		if ok { |  | ||||||
| 			out = append(out, dominant) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	fields = out |  | ||||||
| 	sort.Sort(byIndex(fields)) |  | ||||||
|  |  | ||||||
| 	return fields |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // dominantField looks through the fields, all of which are known to |  | ||||||
| // have the same name, to find the single field that dominates the |  | ||||||
| // others using Go's embedding rules, modified by the presence of |  | ||||||
| // TOML tags. If there are multiple top-level fields, the boolean |  | ||||||
| // will be false: This condition is an error in Go and we skip all |  | ||||||
| // the fields. |  | ||||||
| func dominantField(fields []field) (field, bool) { |  | ||||||
| 	// The fields are sorted in increasing index-length order. The winner |  | ||||||
| 	// must therefore be one with the shortest index length. Drop all |  | ||||||
| 	// longer entries, which is easy: just truncate the slice. |  | ||||||
| 	length := len(fields[0].index) |  | ||||||
| 	tagged := -1 // Index of first tagged field. |  | ||||||
| 	for i, f := range fields { |  | ||||||
| 		if len(f.index) > length { |  | ||||||
| 			fields = fields[:i] |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 		if f.tag { |  | ||||||
| 			if tagged >= 0 { |  | ||||||
| 				// Multiple tagged fields at the same level: conflict. |  | ||||||
| 				// Return no field. |  | ||||||
| 				return field{}, false |  | ||||||
| 			} |  | ||||||
| 			tagged = i |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if tagged >= 0 { |  | ||||||
| 		return fields[tagged], true |  | ||||||
| 	} |  | ||||||
| 	// All remaining fields have the same length. If there's more than one, |  | ||||||
| 	// we have a conflict (two fields named "X" at the same level) and we |  | ||||||
| 	// return no field. |  | ||||||
| 	if len(fields) > 1 { |  | ||||||
| 		return field{}, false |  | ||||||
| 	} |  | ||||||
| 	return fields[0], true |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var fieldCache struct { |  | ||||||
| 	sync.RWMutex |  | ||||||
| 	m map[reflect.Type][]field |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // cachedTypeFields is like typeFields but uses a cache to avoid repeated work. |  | ||||||
| func cachedTypeFields(t reflect.Type) []field { |  | ||||||
| 	fieldCache.RLock() |  | ||||||
| 	f := fieldCache.m[t] |  | ||||||
| 	fieldCache.RUnlock() |  | ||||||
| 	if f != nil { |  | ||||||
| 		return f |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Compute fields without lock. |  | ||||||
| 	// Might duplicate effort but won't hold other computations back. |  | ||||||
| 	f = typeFields(t) |  | ||||||
| 	if f == nil { |  | ||||||
| 		f = []field{} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	fieldCache.Lock() |  | ||||||
| 	if fieldCache.m == nil { |  | ||||||
| 		fieldCache.m = map[reflect.Type][]field{} |  | ||||||
| 	} |  | ||||||
| 	fieldCache.m[t] = f |  | ||||||
| 	fieldCache.Unlock() |  | ||||||
| 	return f |  | ||||||
| } |  | ||||||
| @@ -1,12 +1,12 @@ | |||||||
| language: go | language: go | ||||||
| go: | go: | ||||||
|     - 1.7.x |  | ||||||
|     - 1.8.x |  | ||||||
|     - 1.9.x |     - 1.9.x | ||||||
|  |     - 1.10.x | ||||||
|  |     - 1.11.x | ||||||
| install: | install: | ||||||
|     - go get github.com/bwmarrin/discordgo |     - go get github.com/bwmarrin/discordgo | ||||||
|     - go get -v . |     - go get -v . | ||||||
|     - go get -v github.com/golang/lint/golint |     - go get -v golang.org/x/lint/golint | ||||||
| script: | script: | ||||||
|     - diff <(gofmt -d .) <(echo -n) |     - diff <(gofmt -d .) <(echo -n) | ||||||
|     - go vet -x ./... |     - go vet -x ./... | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # DiscordGo  | # DiscordGo | ||||||
| 
 | 
 | ||||||
| [](https://godoc.org/github.com/bwmarrin/discordgo) [](http://goreportcard.com/report/bwmarrin/discordgo) [](https://travis-ci.org/bwmarrin/discordgo) [](https://discord.gg/0f1SbxBZjYoCtNPP) [](https://discord.gg/0SBTUU1wZTWT6sqd) | [](https://godoc.org/github.com/bwmarrin/discordgo) [](http://goreportcard.com/report/bwmarrin/discordgo) [](https://travis-ci.org/bwmarrin/discordgo) [](https://discord.gg/0f1SbxBZjYoCtNPP) [](https://discord.gg/0SBTUU1wZTWT6sqd) | ||||||
| 
 | 
 | ||||||
| @@ -15,11 +15,11 @@ to add the official DiscordGo test bot **dgo** to your server. This provides | |||||||
| indispensable help to this project. | indispensable help to this project. | ||||||
| 
 | 
 | ||||||
| * See [dgVoice](https://github.com/bwmarrin/dgvoice) package for an example of | * See [dgVoice](https://github.com/bwmarrin/dgvoice) package for an example of | ||||||
| additional voice helper functions and features for DiscordGo | additional voice helper functions and features for DiscordGo. | ||||||
| 
 | 
 | ||||||
| * See [dca](https://github.com/bwmarrin/dca) for an **experimental** stand alone | * See [dca](https://github.com/bwmarrin/dca) for an **experimental** stand alone | ||||||
| tool that wraps `ffmpeg` to create opus encoded audio appropriate for use with | tool that wraps `ffmpeg` to create opus encoded audio appropriate for use with | ||||||
| Discord (and DiscordGo) | Discord (and DiscordGo). | ||||||
| 
 | 
 | ||||||
| **For help with this package or general Go discussion, please join the [Discord  | **For help with this package or general Go discussion, please join the [Discord  | ||||||
| Gophers](https://discord.gg/0f1SbxBZjYq9jLBk) chat server.** | Gophers](https://discord.gg/0f1SbxBZjYq9jLBk) chat server.** | ||||||
| @@ -39,9 +39,9 @@ the breaking changes get documented before pushing to master. | |||||||
| 
 | 
 | ||||||
| *So, what should you use?* | *So, what should you use?* | ||||||
| 
 | 
 | ||||||
| If you can accept the constant changing nature of *develop* then it is the  | If you can accept the constant changing nature of *develop*, it is the  | ||||||
| recommended branch to use.  Otherwise, if you want to tail behind development | recommended branch to use.  Otherwise, if you want to tail behind development | ||||||
| slightly and have a more stable package with documented releases then use *master* | slightly and have a more stable package with documented releases, use *master*. | ||||||
| 
 | 
 | ||||||
| ### Installing | ### Installing | ||||||
| 
 | 
 | ||||||
| @@ -96,10 +96,10 @@ that information in a nice format. | |||||||
| ## Examples | ## Examples | ||||||
| 
 | 
 | ||||||
| Below is a list of examples and other projects using DiscordGo.  Please submit  | Below is a list of examples and other projects using DiscordGo.  Please submit  | ||||||
| an issue if you would like your project added or removed from this list  | an issue if you would like your project added or removed from this list.  | ||||||
| 
 | 
 | ||||||
| - [DiscordGo Examples](https://github.com/bwmarrin/discordgo/tree/master/examples) A collection of example programs written with DiscordGo | - [DiscordGo Examples](https://github.com/bwmarrin/discordgo/tree/master/examples) - A collection of example programs written with DiscordGo | ||||||
| - [Awesome DiscordGo](https://github.com/bwmarrin/discordgo/wiki/Awesome-DiscordGo) A curated list of high quality projects using DiscordGo | - [Awesome DiscordGo](https://github.com/bwmarrin/discordgo/wiki/Awesome-DiscordGo) - A curated list of high quality projects using DiscordGo | ||||||
| 
 | 
 | ||||||
| ## Troubleshooting | ## Troubleshooting | ||||||
| For help with common problems please reference the  | For help with common problems please reference the  | ||||||
| @@ -114,7 +114,7 @@ Contributions are very welcomed, however please follow the below guidelines. | |||||||
| discussed.   | discussed.   | ||||||
| - Fork the develop branch and make your changes.   | - Fork the develop branch and make your changes.   | ||||||
| - Try to match current naming conventions as closely as possible.   | - Try to match current naming conventions as closely as possible.   | ||||||
| - This package is intended to be a low level direct mapping of the Discord API  | - This package is intended to be a low level direct mapping of the Discord API,  | ||||||
| so please avoid adding enhancements outside of that scope without first  | so please avoid adding enhancements outside of that scope without first  | ||||||
| discussing it. | discussing it. | ||||||
| - Create a Pull Request with your changes against the develop branch. | - Create a Pull Request with your changes against the develop branch. | ||||||
| @@ -127,4 +127,4 @@ comparison and list of other Discord API libraries. | |||||||
| 
 | 
 | ||||||
| ## Special Thanks | ## Special Thanks | ||||||
| 
 | 
 | ||||||
| [Chris Rhodes](https://github.com/iopred) - For the DiscordGo logo and tons of PRs | [Chris Rhodes](https://github.com/iopred) - For the DiscordGo logo and tons of PRs. | ||||||
| @@ -6,8 +6,8 @@ | |||||||
| // license that can be found in the LICENSE file. | // license that can be found in the LICENSE file. | ||||||
| 
 | 
 | ||||||
| // This file contains high level helper functions and easy entry points for the | // This file contains high level helper functions and easy entry points for the | ||||||
| // entire discordgo package.  These functions are beling developed and are very | // entire discordgo package.  These functions are being developed and are very | ||||||
| // experimental at this point.  They will most likley change so please use the | // experimental at this point.  They will most likely change so please use the | ||||||
| // low level functions if that's a problem. | // low level functions if that's a problem. | ||||||
| 
 | 
 | ||||||
| // Package discordgo provides Discord binding for Go | // Package discordgo provides Discord binding for Go | ||||||
| @@ -21,7 +21,7 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // VERSION of DiscordGo, follows Semantic Versioning. (http://semver.org/) | // VERSION of DiscordGo, follows Semantic Versioning. (http://semver.org/) | ||||||
| const VERSION = "0.18.0" | const VERSION = "0.19.0" | ||||||
| 
 | 
 | ||||||
| // ErrMFA will be risen by New when the user has 2FA. | // ErrMFA will be risen by New when the user has 2FA. | ||||||
| var ErrMFA = errors.New("account has 2FA enabled") | var ErrMFA = errors.New("account has 2FA enabled") | ||||||
| @@ -11,6 +11,8 @@ | |||||||
| 
 | 
 | ||||||
| package discordgo | package discordgo | ||||||
| 
 | 
 | ||||||
|  | import "strconv" | ||||||
|  | 
 | ||||||
| // APIVersion is the Discord API version used for the REST and Websocket API. | // APIVersion is the Discord API version used for the REST and Websocket API. | ||||||
| var APIVersion = "6" | var APIVersion = "6" | ||||||
| 
 | 
 | ||||||
| @@ -61,14 +63,18 @@ var ( | |||||||
| 	EndpointUser               = func(uID string) string { return EndpointUsers + uID } | 	EndpointUser               = func(uID string) string { return EndpointUsers + uID } | ||||||
| 	EndpointUserAvatar         = func(uID, aID string) string { return EndpointCDNAvatars + uID + "/" + aID + ".png" } | 	EndpointUserAvatar         = func(uID, aID string) string { return EndpointCDNAvatars + uID + "/" + aID + ".png" } | ||||||
| 	EndpointUserAvatarAnimated = func(uID, aID string) string { return EndpointCDNAvatars + uID + "/" + aID + ".gif" } | 	EndpointUserAvatarAnimated = func(uID, aID string) string { return EndpointCDNAvatars + uID + "/" + aID + ".gif" } | ||||||
| 	EndpointUserSettings       = func(uID string) string { return EndpointUsers + uID + "/settings" } | 	EndpointDefaultUserAvatar  = func(uDiscriminator string) string { | ||||||
| 	EndpointUserGuilds         = func(uID string) string { return EndpointUsers + uID + "/guilds" } | 		uDiscriminatorInt, _ := strconv.Atoi(uDiscriminator) | ||||||
| 	EndpointUserGuild          = func(uID, gID string) string { return EndpointUsers + uID + "/guilds/" + gID } | 		return EndpointCDN + "embed/avatars/" + strconv.Itoa(uDiscriminatorInt%5) + ".png" | ||||||
| 	EndpointUserGuildSettings  = func(uID, gID string) string { return EndpointUsers + uID + "/guilds/" + gID + "/settings" } | 	} | ||||||
| 	EndpointUserChannels       = func(uID string) string { return EndpointUsers + uID + "/channels" } | 	EndpointUserSettings      = func(uID string) string { return EndpointUsers + uID + "/settings" } | ||||||
| 	EndpointUserDevices        = func(uID string) string { return EndpointUsers + uID + "/devices" } | 	EndpointUserGuilds        = func(uID string) string { return EndpointUsers + uID + "/guilds" } | ||||||
| 	EndpointUserConnections    = func(uID string) string { return EndpointUsers + uID + "/connections" } | 	EndpointUserGuild         = func(uID, gID string) string { return EndpointUsers + uID + "/guilds/" + gID } | ||||||
| 	EndpointUserNotes          = func(uID string) string { return EndpointUsers + "@me/notes/" + uID } | 	EndpointUserGuildSettings = func(uID, gID string) string { return EndpointUsers + uID + "/guilds/" + gID + "/settings" } | ||||||
|  | 	EndpointUserChannels      = func(uID string) string { return EndpointUsers + uID + "/channels" } | ||||||
|  | 	EndpointUserDevices       = func(uID string) string { return EndpointUsers + uID + "/devices" } | ||||||
|  | 	EndpointUserConnections   = func(uID string) string { return EndpointUsers + uID + "/connections" } | ||||||
|  | 	EndpointUserNotes         = func(uID string) string { return EndpointUsers + "@me/notes/" + uID } | ||||||
| 
 | 
 | ||||||
| 	EndpointGuild                = func(gID string) string { return EndpointGuilds + gID } | 	EndpointGuild                = func(gID string) string { return EndpointGuilds + gID } | ||||||
| 	EndpointGuildChannels        = func(gID string) string { return EndpointGuilds + gID + "/channels" } | 	EndpointGuildChannels        = func(gID string) string { return EndpointGuilds + gID + "/channels" } | ||||||
| @@ -88,6 +94,9 @@ var ( | |||||||
| 	EndpointGuildIcon            = func(gID, hash string) string { return EndpointCDNIcons + gID + "/" + hash + ".png" } | 	EndpointGuildIcon            = func(gID, hash string) string { return EndpointCDNIcons + gID + "/" + hash + ".png" } | ||||||
| 	EndpointGuildSplash          = func(gID, hash string) string { return EndpointCDNSplashes + gID + "/" + hash + ".png" } | 	EndpointGuildSplash          = func(gID, hash string) string { return EndpointCDNSplashes + gID + "/" + hash + ".png" } | ||||||
| 	EndpointGuildWebhooks        = func(gID string) string { return EndpointGuilds + gID + "/webhooks" } | 	EndpointGuildWebhooks        = func(gID string) string { return EndpointGuilds + gID + "/webhooks" } | ||||||
|  | 	EndpointGuildAuditLogs       = func(gID string) string { return EndpointGuilds + gID + "/audit-logs" } | ||||||
|  | 	EndpointGuildEmojis          = func(gID string) string { return EndpointGuilds + gID + "/emojis" } | ||||||
|  | 	EndpointGuildEmoji           = func(gID, eID string) string { return EndpointGuilds + gID + "/emojis/" + eID } | ||||||
| 
 | 
 | ||||||
| 	EndpointChannel                   = func(cID string) string { return EndpointChannels + cID } | 	EndpointChannel                   = func(cID string) string { return EndpointChannels + cID } | ||||||
| 	EndpointChannelPermissions        = func(cID string) string { return EndpointChannels + cID + "/permissions" } | 	EndpointChannelPermissions        = func(cID string) string { return EndpointChannels + cID + "/permissions" } | ||||||
| @@ -127,7 +136,8 @@ var ( | |||||||
| 
 | 
 | ||||||
| 	EndpointIntegrationsJoin = func(iID string) string { return EndpointAPI + "integrations/" + iID + "/join" } | 	EndpointIntegrationsJoin = func(iID string) string { return EndpointAPI + "integrations/" + iID + "/join" } | ||||||
| 
 | 
 | ||||||
| 	EndpointEmoji = func(eID string) string { return EndpointAPI + "emojis/" + eID + ".png" } | 	EndpointEmoji         = func(eID string) string { return EndpointAPI + "emojis/" + eID + ".png" } | ||||||
|  | 	EndpointEmojiAnimated = func(eID string) string { return EndpointAPI + "emojis/" + eID + ".gif" } | ||||||
| 
 | 
 | ||||||
| 	EndpointOauth2          = EndpointAPI + "oauth2/" | 	EndpointOauth2          = EndpointAPI + "oauth2/" | ||||||
| 	EndpointApplications    = EndpointOauth2 + "applications" | 	EndpointApplications    = EndpointOauth2 + "applications" | ||||||
| @@ -98,7 +98,9 @@ func (s *Session) addEventHandlerOnce(eventHandler EventHandler) func() { | |||||||
| 
 | 
 | ||||||
| // AddHandler allows you to add an event handler that will be fired anytime | // AddHandler allows you to add an event handler that will be fired anytime | ||||||
| // the Discord WSAPI event that matches the function fires. | // the Discord WSAPI event that matches the function fires. | ||||||
| // events.go contains all the Discord WSAPI events that can be fired. | // The first parameter is a *Session, and the second parameter is a pointer | ||||||
|  | // to a struct corresponding to the event for which you want to listen. | ||||||
|  | // | ||||||
| // eg: | // eg: | ||||||
| //     Session.AddHandler(func(s *discordgo.Session, m *discordgo.MessageCreate) { | //     Session.AddHandler(func(s *discordgo.Session, m *discordgo.MessageCreate) { | ||||||
| //     }) | //     }) | ||||||
| @@ -106,6 +108,13 @@ func (s *Session) addEventHandlerOnce(eventHandler EventHandler) func() { | |||||||
| // or: | // or: | ||||||
| //     Session.AddHandler(func(s *discordgo.Session, m *discordgo.PresenceUpdate) { | //     Session.AddHandler(func(s *discordgo.Session, m *discordgo.PresenceUpdate) { | ||||||
| //     }) | //     }) | ||||||
|  | // | ||||||
|  | // List of events can be found at this page, with corresponding names in the | ||||||
|  | // library for each event: https://discordapp.com/developers/docs/topics/gateway#event-names | ||||||
|  | // There are also synthetic events fired by the library internally which are | ||||||
|  | // available for handling, like Connect, Disconnect, and RateLimit. | ||||||
|  | // events.go contains all of the Discord WSAPI and synthetic events that can be handled. | ||||||
|  | // | ||||||
| // The return value of this method is a function, that when called will remove the | // The return value of this method is a function, that when called will remove the | ||||||
| // event handler. | // event handler. | ||||||
| func (s *Session) AddHandler(handler interface{}) func() { | func (s *Session) AddHandler(handler interface{}) func() { | ||||||
| @@ -50,6 +50,7 @@ const ( | |||||||
| 	userUpdateEventType               = "USER_UPDATE" | 	userUpdateEventType               = "USER_UPDATE" | ||||||
| 	voiceServerUpdateEventType        = "VOICE_SERVER_UPDATE" | 	voiceServerUpdateEventType        = "VOICE_SERVER_UPDATE" | ||||||
| 	voiceStateUpdateEventType         = "VOICE_STATE_UPDATE" | 	voiceStateUpdateEventType         = "VOICE_STATE_UPDATE" | ||||||
|  | 	webhooksUpdateEventType           = "WEBHOOKS_UPDATE" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // channelCreateEventHandler is an event handler for ChannelCreate events. | // channelCreateEventHandler is an event handler for ChannelCreate events. | ||||||
| @@ -892,6 +893,26 @@ func (eh voiceStateUpdateEventHandler) Handle(s *Session, i interface{}) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // webhooksUpdateEventHandler is an event handler for WebhooksUpdate events. | ||||||
|  | type webhooksUpdateEventHandler func(*Session, *WebhooksUpdate) | ||||||
|  | 
 | ||||||
|  | // Type returns the event type for WebhooksUpdate events. | ||||||
|  | func (eh webhooksUpdateEventHandler) Type() string { | ||||||
|  | 	return webhooksUpdateEventType | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // New returns a new instance of WebhooksUpdate. | ||||||
|  | func (eh webhooksUpdateEventHandler) New() interface{} { | ||||||
|  | 	return &WebhooksUpdate{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Handle is the handler for WebhooksUpdate events. | ||||||
|  | func (eh webhooksUpdateEventHandler) Handle(s *Session, i interface{}) { | ||||||
|  | 	if t, ok := i.(*WebhooksUpdate); ok { | ||||||
|  | 		eh(s, t) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func handlerForInterface(handler interface{}) EventHandler { | func handlerForInterface(handler interface{}) EventHandler { | ||||||
| 	switch v := handler.(type) { | 	switch v := handler.(type) { | ||||||
| 	case func(*Session, interface{}): | 	case func(*Session, interface{}): | ||||||
| @@ -982,6 +1003,8 @@ func handlerForInterface(handler interface{}) EventHandler { | |||||||
| 		return voiceServerUpdateEventHandler(v) | 		return voiceServerUpdateEventHandler(v) | ||||||
| 	case func(*Session, *VoiceStateUpdate): | 	case func(*Session, *VoiceStateUpdate): | ||||||
| 		return voiceStateUpdateEventHandler(v) | 		return voiceStateUpdateEventHandler(v) | ||||||
|  | 	case func(*Session, *WebhooksUpdate): | ||||||
|  | 		return webhooksUpdateEventHandler(v) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return nil | 	return nil | ||||||
| @@ -1027,4 +1050,5 @@ func init() { | |||||||
| 	registerInterfaceProvider(userUpdateEventHandler(nil)) | 	registerInterfaceProvider(userUpdateEventHandler(nil)) | ||||||
| 	registerInterfaceProvider(voiceServerUpdateEventHandler(nil)) | 	registerInterfaceProvider(voiceServerUpdateEventHandler(nil)) | ||||||
| 	registerInterfaceProvider(voiceStateUpdateEventHandler(nil)) | 	registerInterfaceProvider(voiceStateUpdateEventHandler(nil)) | ||||||
|  | 	registerInterfaceProvider(webhooksUpdateEventHandler(nil)) | ||||||
| } | } | ||||||
| @@ -70,6 +70,7 @@ type ChannelDelete struct { | |||||||
| type ChannelPinsUpdate struct { | type ChannelPinsUpdate struct { | ||||||
| 	LastPinTimestamp string `json:"last_pin_timestamp"` | 	LastPinTimestamp string `json:"last_pin_timestamp"` | ||||||
| 	ChannelID        string `json:"channel_id"` | 	ChannelID        string `json:"channel_id"` | ||||||
|  | 	GuildID          string `json:"guild_id,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GuildCreate is the data for a GuildCreate event. | // GuildCreate is the data for a GuildCreate event. | ||||||
| @@ -212,6 +213,7 @@ type RelationshipRemove struct { | |||||||
| type TypingStart struct { | type TypingStart struct { | ||||||
| 	UserID    string `json:"user_id"` | 	UserID    string `json:"user_id"` | ||||||
| 	ChannelID string `json:"channel_id"` | 	ChannelID string `json:"channel_id"` | ||||||
|  | 	GuildID   string `json:"guild_id,omitempty"` | ||||||
| 	Timestamp int    `json:"timestamp"` | 	Timestamp int    `json:"timestamp"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -250,4 +252,11 @@ type VoiceStateUpdate struct { | |||||||
| type MessageDeleteBulk struct { | type MessageDeleteBulk struct { | ||||||
| 	Messages  []string `json:"ids"` | 	Messages  []string `json:"ids"` | ||||||
| 	ChannelID string   `json:"channel_id"` | 	ChannelID string   `json:"channel_id"` | ||||||
|  | 	GuildID   string   `json:"guild_id"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WebhooksUpdate is the data for a WebhooksUpdate event | ||||||
|  | type WebhooksUpdate struct { | ||||||
|  | 	GuildID   string `json:"guild_id"` | ||||||
|  | 	ChannelID string `json:"channel_id"` | ||||||
| } | } | ||||||
							
								
								
									
										6
									
								
								vendor/github.com/bwmarrin/discordgo/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | |||||||
|  | module github.com/bwmarrin/discordgo | ||||||
|  |  | ||||||
|  | require ( | ||||||
|  | 	github.com/gorilla/websocket v1.4.0 | ||||||
|  | 	golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16 | ||||||
|  | ) | ||||||
							
								
								
									
										4
									
								
								vendor/github.com/bwmarrin/discordgo/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | github.com/gorilla/websocket v1.4.0 h1:WDFjx/TMzVgy9VdMMQi2K2Emtwi2QcUQsztZ/zLaH/Q= | ||||||
|  | github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= | ||||||
|  | golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16 h1:y6ce7gCWtnH+m3dCjzQ1PCuwl28DDIc3VNnvY29DlIA= | ||||||
|  | golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||||
| @@ -32,20 +32,59 @@ const ( | |||||||
| 
 | 
 | ||||||
| // A Message stores all data related to a specific Discord message. | // A Message stores all data related to a specific Discord message. | ||||||
| type Message struct { | type Message struct { | ||||||
| 	ID              string               `json:"id"` | 	// The ID of the message. | ||||||
| 	ChannelID       string               `json:"channel_id"` | 	ID string `json:"id"` | ||||||
| 	Content         string               `json:"content"` | 
 | ||||||
| 	Timestamp       Timestamp            `json:"timestamp"` | 	// The ID of the channel in which the message was sent. | ||||||
| 	EditedTimestamp Timestamp            `json:"edited_timestamp"` | 	ChannelID string `json:"channel_id"` | ||||||
| 	MentionRoles    []string             `json:"mention_roles"` | 
 | ||||||
| 	Tts             bool                 `json:"tts"` | 	// The ID of the guild in which the message was sent. | ||||||
| 	MentionEveryone bool                 `json:"mention_everyone"` | 	GuildID string `json:"guild_id,omitempty"` | ||||||
| 	Author          *User                `json:"author"` | 
 | ||||||
| 	Attachments     []*MessageAttachment `json:"attachments"` | 	// The content of the message. | ||||||
| 	Embeds          []*MessageEmbed      `json:"embeds"` | 	Content string `json:"content"` | ||||||
| 	Mentions        []*User              `json:"mentions"` | 
 | ||||||
| 	Reactions       []*MessageReactions  `json:"reactions"` | 	// The time at which the messsage was sent. | ||||||
| 	Type            MessageType          `json:"type"` | 	// CAUTION: this field may be removed in a | ||||||
|  | 	// future API version; it is safer to calculate | ||||||
|  | 	// the creation time via the ID. | ||||||
|  | 	Timestamp Timestamp `json:"timestamp"` | ||||||
|  | 
 | ||||||
|  | 	// The time at which the last edit of the message | ||||||
|  | 	// occurred, if it has been edited. | ||||||
|  | 	EditedTimestamp Timestamp `json:"edited_timestamp"` | ||||||
|  | 
 | ||||||
|  | 	// The roles mentioned in the message. | ||||||
|  | 	MentionRoles []string `json:"mention_roles"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the message is text-to-speech. | ||||||
|  | 	Tts bool `json:"tts"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the message mentions everyone. | ||||||
|  | 	MentionEveryone bool `json:"mention_everyone"` | ||||||
|  | 
 | ||||||
|  | 	// The author of the message. This is not guaranteed to be a | ||||||
|  | 	// valid user (webhook-sent messages do not possess a full author). | ||||||
|  | 	Author *User `json:"author"` | ||||||
|  | 
 | ||||||
|  | 	// A list of attachments present in the message. | ||||||
|  | 	Attachments []*MessageAttachment `json:"attachments"` | ||||||
|  | 
 | ||||||
|  | 	// A list of embeds present in the message. Multiple | ||||||
|  | 	// embeds can currently only be sent by webhooks. | ||||||
|  | 	Embeds []*MessageEmbed `json:"embeds"` | ||||||
|  | 
 | ||||||
|  | 	// A list of users mentioned in the message. | ||||||
|  | 	Mentions []*User `json:"mentions"` | ||||||
|  | 
 | ||||||
|  | 	// A list of reactions to the message. | ||||||
|  | 	Reactions []*MessageReactions `json:"reactions"` | ||||||
|  | 
 | ||||||
|  | 	// The type of the message. | ||||||
|  | 	Type MessageType `json:"type"` | ||||||
|  | 
 | ||||||
|  | 	// The webhook ID of the message, if it was generated by a webhook | ||||||
|  | 	WebhookID string `json:"webhook_id"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // File stores info about files you e.g. send in messages. | // File stores info about files you e.g. send in messages. | ||||||
| @@ -38,6 +38,7 @@ var ( | |||||||
| 	ErrPruneDaysBounds         = errors.New("the number of days should be more than or equal to 1") | 	ErrPruneDaysBounds         = errors.New("the number of days should be more than or equal to 1") | ||||||
| 	ErrGuildNoIcon             = errors.New("guild does not have an icon set") | 	ErrGuildNoIcon             = errors.New("guild does not have an icon set") | ||||||
| 	ErrGuildNoSplash           = errors.New("guild does not have a splash set") | 	ErrGuildNoSplash           = errors.New("guild does not have a splash set") | ||||||
|  | 	ErrUnauthorized            = errors.New("HTTP request was unauthorized. This could be because the provided token was not a bot token. Please add \"Bot \" to the start of your token. https://discordapp.com/developers/docs/reference#authentication-example-bot-token-authorization-header") | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // Request is the same as RequestWithBucketID but the bucket id is the same as the urlStr | // Request is the same as RequestWithBucketID but the bucket id is the same as the urlStr | ||||||
| @@ -89,7 +90,7 @@ func (s *Session) RequestWithLockedBucket(method, urlStr, contentType string, b | |||||||
| 
 | 
 | ||||||
| 	req.Header.Set("Content-Type", contentType) | 	req.Header.Set("Content-Type", contentType) | ||||||
| 	// TODO: Make a configurable static variable. | 	// TODO: Make a configurable static variable. | ||||||
| 	req.Header.Set("User-Agent", fmt.Sprintf("DiscordBot (https://github.com/bwmarrin/discordgo, v%s)", VERSION)) | 	req.Header.Set("User-Agent", "DiscordBot (https://github.com/bwmarrin/discordgo, v"+VERSION+")") | ||||||
| 
 | 
 | ||||||
| 	if s.Debug { | 	if s.Debug { | ||||||
| 		for k, v := range req.Header { | 		for k, v := range req.Header { | ||||||
| @@ -129,13 +130,9 @@ func (s *Session) RequestWithLockedBucket(method, urlStr, contentType string, b | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	switch resp.StatusCode { | 	switch resp.StatusCode { | ||||||
| 
 |  | ||||||
| 	case http.StatusOK: | 	case http.StatusOK: | ||||||
| 	case http.StatusCreated: | 	case http.StatusCreated: | ||||||
| 	case http.StatusNoContent: | 	case http.StatusNoContent: | ||||||
| 
 |  | ||||||
| 		// TODO check for 401 response, invalidate token if we get one. |  | ||||||
| 
 |  | ||||||
| 	case http.StatusBadGateway: | 	case http.StatusBadGateway: | ||||||
| 		// Retry sending request if possible | 		// Retry sending request if possible | ||||||
| 		if sequence < s.MaxRestRetries { | 		if sequence < s.MaxRestRetries { | ||||||
| @@ -145,7 +142,6 @@ func (s *Session) RequestWithLockedBucket(method, urlStr, contentType string, b | |||||||
| 		} else { | 		} else { | ||||||
| 			err = fmt.Errorf("Exceeded Max retries HTTP %s, %s", resp.Status, response) | 			err = fmt.Errorf("Exceeded Max retries HTTP %s, %s", resp.Status, response) | ||||||
| 		} | 		} | ||||||
| 
 |  | ||||||
| 	case 429: // TOO MANY REQUESTS - Rate limiting | 	case 429: // TOO MANY REQUESTS - Rate limiting | ||||||
| 		rl := TooManyRequests{} | 		rl := TooManyRequests{} | ||||||
| 		err = json.Unmarshal(response, &rl) | 		err = json.Unmarshal(response, &rl) | ||||||
| @@ -161,7 +157,12 @@ func (s *Session) RequestWithLockedBucket(method, urlStr, contentType string, b | |||||||
| 		// this method can cause longer delays than required | 		// this method can cause longer delays than required | ||||||
| 
 | 
 | ||||||
| 		response, err = s.RequestWithLockedBucket(method, urlStr, contentType, b, s.Ratelimiter.LockBucketObject(bucket), sequence) | 		response, err = s.RequestWithLockedBucket(method, urlStr, contentType, b, s.Ratelimiter.LockBucketObject(bucket), sequence) | ||||||
| 
 | 	case http.StatusUnauthorized: | ||||||
|  | 		if strings.Index(s.Token, "Bot ") != 0 { | ||||||
|  | 			s.log(LogInformational, ErrUnauthorized.Error()) | ||||||
|  | 			err = ErrUnauthorized | ||||||
|  | 		} | ||||||
|  | 		fallthrough | ||||||
| 	default: // Error condition | 	default: // Error condition | ||||||
| 		err = newRestError(req, resp, response) | 		err = newRestError(req, resp, response) | ||||||
| 	} | 	} | ||||||
| @@ -249,7 +250,7 @@ func (s *Session) Register(username string) (token string, err error) { | |||||||
| // even use. | // even use. | ||||||
| func (s *Session) Logout() (err error) { | func (s *Session) Logout() (err error) { | ||||||
| 
 | 
 | ||||||
| 	//  _, err = s.Request("POST", LOGOUT, fmt.Sprintf(`{"token": "%s"}`, s.Token)) | 	//  _, err = s.Request("POST", LOGOUT, `{"token": "` + s.Token + `"}`) | ||||||
| 
 | 
 | ||||||
| 	if s.Token == "" { | 	if s.Token == "" { | ||||||
| 		return | 		return | ||||||
| @@ -361,6 +362,21 @@ func (s *Session) UserUpdateStatus(status Status) (st *Settings, err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // UserConnections returns the user's connections | ||||||
|  | func (s *Session) UserConnections() (conn []*UserConnection, err error) { | ||||||
|  | 	response, err := s.RequestWithBucketID("GET", EndpointUserConnections("@me"), nil, EndpointUserConnections("@me")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = unmarshal(response, &conn) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // UserChannels returns an array of Channel structures for all private | // UserChannels returns an array of Channel structures for all private | ||||||
| // channels. | // channels. | ||||||
| func (s *Session) UserChannels() (st []*Channel, err error) { | func (s *Session) UserChannels() (st []*Channel, err error) { | ||||||
| @@ -412,7 +428,7 @@ func (s *Session) UserGuilds(limit int, beforeID, afterID string) (st []*UserGui | |||||||
| 	uri := EndpointUserGuilds("@me") | 	uri := EndpointUserGuilds("@me") | ||||||
| 
 | 
 | ||||||
| 	if len(v) > 0 { | 	if len(v) > 0 { | ||||||
| 		uri = fmt.Sprintf("%s?%s", uri, v.Encode()) | 		uri += "?" + v.Encode() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointUserGuilds("")) | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointUserGuilds("")) | ||||||
| @@ -565,7 +581,7 @@ func (s *Session) Guild(guildID string) (st *Guild, err error) { | |||||||
| 	if s.StateEnabled { | 	if s.StateEnabled { | ||||||
| 		// Attempt to grab the guild from State first. | 		// Attempt to grab the guild from State first. | ||||||
| 		st, err = s.State.Guild(guildID) | 		st, err = s.State.Guild(guildID) | ||||||
| 		if err == nil { | 		if err == nil && !st.Unavailable { | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -735,7 +751,7 @@ func (s *Session) GuildMembers(guildID string, after string, limit int) (st []*M | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if len(v) > 0 { | 	if len(v) > 0 { | ||||||
| 		uri = fmt.Sprintf("%s?%s", uri, v.Encode()) | 		uri += "?" + v.Encode() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointGuildMembers(guildID)) | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointGuildMembers(guildID)) | ||||||
| @@ -761,6 +777,32 @@ func (s *Session) GuildMember(guildID, userID string) (st *Member, err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GuildMemberAdd force joins a user to the guild. | ||||||
|  | //  accessToken   : Valid access_token for the user. | ||||||
|  | //  guildID       : The ID of a Guild. | ||||||
|  | //  userID        : The ID of a User. | ||||||
|  | //  nick          : Value to set users nickname to | ||||||
|  | //  roles         : A list of role ID's to set on the member. | ||||||
|  | //  mute          : If the user is muted. | ||||||
|  | //  deaf          : If the user is deafened. | ||||||
|  | func (s *Session) GuildMemberAdd(accessToken, guildID, userID, nick string, roles []string, mute, deaf bool) (err error) { | ||||||
|  | 
 | ||||||
|  | 	data := struct { | ||||||
|  | 		AccessToken string   `json:"access_token"` | ||||||
|  | 		Nick        string   `json:"nick,omitempty"` | ||||||
|  | 		Roles       []string `json:"roles,omitempty"` | ||||||
|  | 		Mute        bool     `json:"mute,omitempty"` | ||||||
|  | 		Deaf        bool     `json:"deaf,omitempty"` | ||||||
|  | 	}{accessToken, nick, roles, mute, deaf} | ||||||
|  | 
 | ||||||
|  | 	_, err = s.RequestWithBucketID("PUT", EndpointGuildMember(guildID, userID), data, EndpointGuildMember(guildID, "")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GuildMemberDelete removes the given user from the given guild. | // GuildMemberDelete removes the given user from the given guild. | ||||||
| // guildID   : The ID of a Guild. | // guildID   : The ID of a Guild. | ||||||
| // userID    : The ID of a User | // userID    : The ID of a User | ||||||
| @@ -877,17 +919,22 @@ func (s *Session) GuildChannels(guildID string) (st []*Channel, err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GuildChannelCreate creates a new channel in the given guild | // GuildChannelCreateData is provided to GuildChannelCreateComplex | ||||||
| // guildID   : The ID of a Guild. | type GuildChannelCreateData struct { | ||||||
| // name      : Name of the channel (2-100 chars length) | 	Name                 string                 `json:"name"` | ||||||
| // ctype     : Tpye of the channel (voice or text) | 	Type                 ChannelType            `json:"type"` | ||||||
| func (s *Session) GuildChannelCreate(guildID, name, ctype string) (st *Channel, err error) { | 	Topic                string                 `json:"topic,omitempty"` | ||||||
| 
 | 	Bitrate              int                    `json:"bitrate,omitempty"` | ||||||
| 	data := struct { | 	UserLimit            int                    `json:"user_limit,omitempty"` | ||||||
| 		Name string `json:"name"` | 	PermissionOverwrites []*PermissionOverwrite `json:"permission_overwrites,omitempty"` | ||||||
| 		Type string `json:"type"` | 	ParentID             string                 `json:"parent_id,omitempty"` | ||||||
| 	}{name, ctype} | 	NSFW                 bool                   `json:"nsfw,omitempty"` | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|  | // GuildChannelCreateComplex creates a new channel in the given guild | ||||||
|  | // guildID      : The ID of a Guild | ||||||
|  | // data         : A data struct describing the new Channel, Name and Type are mandatory, other fields depending on the type | ||||||
|  | func (s *Session) GuildChannelCreateComplex(guildID string, data GuildChannelCreateData) (st *Channel, err error) { | ||||||
| 	body, err := s.RequestWithBucketID("POST", EndpointGuildChannels(guildID), data, EndpointGuildChannels(guildID)) | 	body, err := s.RequestWithBucketID("POST", EndpointGuildChannels(guildID), data, EndpointGuildChannels(guildID)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return | 		return | ||||||
| @@ -897,12 +944,33 @@ func (s *Session) GuildChannelCreate(guildID, name, ctype string) (st *Channel, | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GuildChannelCreate creates a new channel in the given guild | ||||||
|  | // guildID   : The ID of a Guild. | ||||||
|  | // name      : Name of the channel (2-100 chars length) | ||||||
|  | // ctype     : Type of the channel | ||||||
|  | func (s *Session) GuildChannelCreate(guildID, name string, ctype ChannelType) (st *Channel, err error) { | ||||||
|  | 	return s.GuildChannelCreateComplex(guildID, GuildChannelCreateData{ | ||||||
|  | 		Name: name, | ||||||
|  | 		Type: ctype, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GuildChannelsReorder updates the order of channels in a guild | // GuildChannelsReorder updates the order of channels in a guild | ||||||
| // guildID   : The ID of a Guild. | // guildID   : The ID of a Guild. | ||||||
| // channels  : Updated channels. | // channels  : Updated channels. | ||||||
| func (s *Session) GuildChannelsReorder(guildID string, channels []*Channel) (err error) { | func (s *Session) GuildChannelsReorder(guildID string, channels []*Channel) (err error) { | ||||||
| 
 | 
 | ||||||
| 	_, err = s.RequestWithBucketID("PATCH", EndpointGuildChannels(guildID), channels, EndpointGuildChannels(guildID)) | 	data := make([]struct { | ||||||
|  | 		ID       string `json:"id"` | ||||||
|  | 		Position int    `json:"position"` | ||||||
|  | 	}, len(channels)) | ||||||
|  | 
 | ||||||
|  | 	for i, c := range channels { | ||||||
|  | 		data[i].ID = c.ID | ||||||
|  | 		data[i].Position = c.Position | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	_, err = s.RequestWithBucketID("PATCH", EndpointGuildChannels(guildID), data, EndpointGuildChannels(guildID)) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @@ -1021,7 +1089,7 @@ func (s *Session) GuildPruneCount(guildID string, days uint32) (count uint32, er | |||||||
| 		Pruned uint32 `json:"pruned"` | 		Pruned uint32 `json:"pruned"` | ||||||
| 	}{} | 	}{} | ||||||
| 
 | 
 | ||||||
| 	uri := EndpointGuildPrune(guildID) + fmt.Sprintf("?days=%d", days) | 	uri := EndpointGuildPrune(guildID) + "?days=" + strconv.FormatUint(uint64(days), 10) | ||||||
| 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointGuildPrune(guildID)) | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointGuildPrune(guildID)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return | 		return | ||||||
| @@ -1075,7 +1143,7 @@ func (s *Session) GuildPrune(guildID string, days uint32) (count uint32, err err | |||||||
| 
 | 
 | ||||||
| // GuildIntegrations returns an array of Integrations for a guild. | // GuildIntegrations returns an array of Integrations for a guild. | ||||||
| // guildID   : The ID of a Guild. | // guildID   : The ID of a Guild. | ||||||
| func (s *Session) GuildIntegrations(guildID string) (st []*GuildIntegration, err error) { | func (s *Session) GuildIntegrations(guildID string) (st []*Integration, err error) { | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("GET", EndpointGuildIntegrations(guildID), nil, EndpointGuildIntegrations(guildID)) | 	body, err := s.RequestWithBucketID("GET", EndpointGuildIntegrations(guildID), nil, EndpointGuildIntegrations(guildID)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -1206,6 +1274,94 @@ func (s *Session) GuildEmbedEdit(guildID string, enabled bool, channelID string) | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GuildAuditLog returns the audit log for a Guild. | ||||||
|  | // guildID     : The ID of a Guild. | ||||||
|  | // userID      : If provided the log will be filtered for the given ID. | ||||||
|  | // beforeID    : If provided all log entries returned will be before the given ID. | ||||||
|  | // actionType  : If provided the log will be filtered for the given Action Type. | ||||||
|  | // limit       : The number messages that can be returned. (default 50, min 1, max 100) | ||||||
|  | func (s *Session) GuildAuditLog(guildID, userID, beforeID string, actionType, limit int) (st *GuildAuditLog, err error) { | ||||||
|  | 
 | ||||||
|  | 	uri := EndpointGuildAuditLogs(guildID) | ||||||
|  | 
 | ||||||
|  | 	v := url.Values{} | ||||||
|  | 	if userID != "" { | ||||||
|  | 		v.Set("user_id", userID) | ||||||
|  | 	} | ||||||
|  | 	if beforeID != "" { | ||||||
|  | 		v.Set("before", beforeID) | ||||||
|  | 	} | ||||||
|  | 	if actionType > 0 { | ||||||
|  | 		v.Set("action_type", strconv.Itoa(actionType)) | ||||||
|  | 	} | ||||||
|  | 	if limit > 0 { | ||||||
|  | 		v.Set("limit", strconv.Itoa(limit)) | ||||||
|  | 	} | ||||||
|  | 	if len(v) > 0 { | ||||||
|  | 		uri = fmt.Sprintf("%s?%s", uri, v.Encode()) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointGuildAuditLogs(guildID)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = unmarshal(body, &st) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GuildEmojiCreate creates a new emoji | ||||||
|  | // guildID : The ID of a Guild. | ||||||
|  | // name    : The Name of the Emoji. | ||||||
|  | // image   : The base64 encoded emoji image, has to be smaller than 256KB. | ||||||
|  | // roles   : The roles for which this emoji will be whitelisted, can be nil. | ||||||
|  | func (s *Session) GuildEmojiCreate(guildID, name, image string, roles []string) (emoji *Emoji, err error) { | ||||||
|  | 
 | ||||||
|  | 	data := struct { | ||||||
|  | 		Name  string   `json:"name"` | ||||||
|  | 		Image string   `json:"image"` | ||||||
|  | 		Roles []string `json:"roles,omitempty"` | ||||||
|  | 	}{name, image, roles} | ||||||
|  | 
 | ||||||
|  | 	body, err := s.RequestWithBucketID("POST", EndpointGuildEmojis(guildID), data, EndpointGuildEmojis(guildID)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = unmarshal(body, &emoji) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GuildEmojiEdit modifies an emoji | ||||||
|  | // guildID : The ID of a Guild. | ||||||
|  | // emojiID : The ID of an Emoji. | ||||||
|  | // name    : The Name of the Emoji. | ||||||
|  | // roles   : The roles for which this emoji will be whitelisted, can be nil. | ||||||
|  | func (s *Session) GuildEmojiEdit(guildID, emojiID, name string, roles []string) (emoji *Emoji, err error) { | ||||||
|  | 
 | ||||||
|  | 	data := struct { | ||||||
|  | 		Name  string   `json:"name"` | ||||||
|  | 		Roles []string `json:"roles,omitempty"` | ||||||
|  | 	}{name, roles} | ||||||
|  | 
 | ||||||
|  | 	body, err := s.RequestWithBucketID("PATCH", EndpointGuildEmoji(guildID, emojiID), data, EndpointGuildEmojis(guildID)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = unmarshal(body, &emoji) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GuildEmojiDelete deletes an Emoji. | ||||||
|  | // guildID : The ID of a Guild. | ||||||
|  | // emojiID : The ID of an Emoji. | ||||||
|  | func (s *Session) GuildEmojiDelete(guildID, emojiID string) (err error) { | ||||||
|  | 
 | ||||||
|  | 	_, err = s.RequestWithBucketID("DELETE", EndpointGuildEmoji(guildID, emojiID), nil, EndpointGuildEmojis(guildID)) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // ------------------------------------------------------------------------------------------------ | // ------------------------------------------------------------------------------------------------ | ||||||
| // Functions specific to Discord Channels | // Functions specific to Discord Channels | ||||||
| // ------------------------------------------------------------------------------------------------ | // ------------------------------------------------------------------------------------------------ | ||||||
| @@ -1291,7 +1447,7 @@ func (s *Session) ChannelMessages(channelID string, limit int, beforeID, afterID | |||||||
| 		v.Set("around", aroundID) | 		v.Set("around", aroundID) | ||||||
| 	} | 	} | ||||||
| 	if len(v) > 0 { | 	if len(v) > 0 { | ||||||
| 		uri = fmt.Sprintf("%s?%s", uri, v.Encode()) | 		uri += "?" + v.Encode() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointChannelMessages(channelID)) | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointChannelMessages(channelID)) | ||||||
| @@ -1586,7 +1742,8 @@ func (s *Session) ChannelInviteCreate(channelID string, i Invite) (st *Invite, e | |||||||
| 		MaxAge    int  `json:"max_age"` | 		MaxAge    int  `json:"max_age"` | ||||||
| 		MaxUses   int  `json:"max_uses"` | 		MaxUses   int  `json:"max_uses"` | ||||||
| 		Temporary bool `json:"temporary"` | 		Temporary bool `json:"temporary"` | ||||||
| 	}{i.MaxAge, i.MaxUses, i.Temporary} | 		Unique    bool `json:"unique"` | ||||||
|  | 	}{i.MaxAge, i.MaxUses, i.Temporary, i.Unique} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("POST", EndpointChannelInvites(channelID), data, EndpointChannelInvites(channelID)) | 	body, err := s.RequestWithBucketID("POST", EndpointChannelInvites(channelID), data, EndpointChannelInvites(channelID)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -1638,6 +1795,19 @@ func (s *Session) Invite(inviteID string) (st *Invite, err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // InviteWithCounts returns an Invite structure of the given invite including approximate member counts | ||||||
|  | // inviteID : The invite code | ||||||
|  | func (s *Session) InviteWithCounts(inviteID string) (st *Invite, err error) { | ||||||
|  | 
 | ||||||
|  | 	body, err := s.RequestWithBucketID("GET", EndpointInvite(inviteID)+"?with_counts=true", nil, EndpointInvite("")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = unmarshal(body, &st) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // InviteDelete deletes an existing invite | // InviteDelete deletes an existing invite | ||||||
| // inviteID   : the code of an invite | // inviteID   : the code of an invite | ||||||
| func (s *Session) InviteDelete(inviteID string) (st *Invite, err error) { | func (s *Session) InviteDelete(inviteID string) (st *Invite, err error) { | ||||||
| @@ -1830,12 +2000,13 @@ func (s *Session) WebhookWithToken(webhookID, token string) (st *Webhook, err er | |||||||
| // webhookID: The ID of a webhook. | // webhookID: The ID of a webhook. | ||||||
| // name     : The name of the webhook. | // name     : The name of the webhook. | ||||||
| // avatar   : The avatar of the webhook. | // avatar   : The avatar of the webhook. | ||||||
| func (s *Session) WebhookEdit(webhookID, name, avatar string) (st *Role, err error) { | func (s *Session) WebhookEdit(webhookID, name, avatar, channelID string) (st *Role, err error) { | ||||||
| 
 | 
 | ||||||
| 	data := struct { | 	data := struct { | ||||||
| 		Name   string `json:"name,omitempty"` | 		Name      string `json:"name,omitempty"` | ||||||
| 		Avatar string `json:"avatar,omitempty"` | 		Avatar    string `json:"avatar,omitempty"` | ||||||
| 	}{name, avatar} | 		ChannelID string `json:"channel_id,omitempty"` | ||||||
|  | 	}{name, avatar, channelID} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("PATCH", EndpointWebhook(webhookID), data, EndpointWebhooks) | 	body, err := s.RequestWithBucketID("PATCH", EndpointWebhook(webhookID), data, EndpointWebhooks) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -1956,7 +2127,7 @@ func (s *Session) MessageReactions(channelID, messageID, emojiID string, limit i | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if len(v) > 0 { | 	if len(v) > 0 { | ||||||
| 		uri = fmt.Sprintf("%s?%s", uri, v.Encode()) | 		uri += "?" + v.Encode() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointMessageReaction(channelID, "", "", "")) | 	body, err := s.RequestWithBucketID("GET", uri, nil, EndpointMessageReaction(channelID, "", "", "")) | ||||||
| @@ -32,6 +32,7 @@ type State struct { | |||||||
| 	sync.RWMutex | 	sync.RWMutex | ||||||
| 	Ready | 	Ready | ||||||
| 
 | 
 | ||||||
|  | 	// MaxMessageCount represents how many messages per channel the state will store. | ||||||
| 	MaxMessageCount int | 	MaxMessageCount int | ||||||
| 	TrackChannels   bool | 	TrackChannels   bool | ||||||
| 	TrackEmojis     bool | 	TrackEmojis     bool | ||||||
| @@ -98,6 +99,9 @@ func (s *State) GuildAdd(guild *Guild) error { | |||||||
| 	if g, ok := s.guildMap[guild.ID]; ok { | 	if g, ok := s.guildMap[guild.ID]; ok { | ||||||
| 		// We are about to replace `g` in the state with `guild`, but first we need to | 		// We are about to replace `g` in the state with `guild`, but first we need to | ||||||
| 		// make sure we preserve any fields that the `guild` doesn't contain from `g`. | 		// make sure we preserve any fields that the `guild` doesn't contain from `g`. | ||||||
|  | 		if guild.MemberCount == 0 { | ||||||
|  | 			guild.MemberCount = g.MemberCount | ||||||
|  | 		} | ||||||
| 		if guild.Roles == nil { | 		if guild.Roles == nil { | ||||||
| 			guild.Roles = g.Roles | 			guild.Roles = g.Roles | ||||||
| 		} | 		} | ||||||
| @@ -299,7 +303,12 @@ func (s *State) MemberAdd(member *Member) error { | |||||||
| 		members[member.User.ID] = member | 		members[member.User.ID] = member | ||||||
| 		guild.Members = append(guild.Members, member) | 		guild.Members = append(guild.Members, member) | ||||||
| 	} else { | 	} else { | ||||||
| 		*m = *member // Update the actual data, which will also update the member pointer in the slice | 		// We are about to replace `m` in the state with `member`, but first we need to | ||||||
|  | 		// make sure we preserve any fields that the `member` doesn't contain from `m`. | ||||||
|  | 		if member.JoinedAt == "" { | ||||||
|  | 			member.JoinedAt = m.JoinedAt | ||||||
|  | 		} | ||||||
|  | 		*m = *member | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return nil | 	return nil | ||||||
| @@ -607,7 +616,7 @@ func (s *State) EmojisAdd(guildID string, emojis []*Emoji) error { | |||||||
| 
 | 
 | ||||||
| // MessageAdd adds a message to the current world state, or updates it if it exists. | // MessageAdd adds a message to the current world state, or updates it if it exists. | ||||||
| // If the channel cannot be found, the message is discarded. | // If the channel cannot be found, the message is discarded. | ||||||
| // Messages are kept in state up to s.MaxMessageCount | // Messages are kept in state up to s.MaxMessageCount per channel. | ||||||
| func (s *State) MessageAdd(message *Message) error { | func (s *State) MessageAdd(message *Message) error { | ||||||
| 	if s == nil { | 	if s == nil { | ||||||
| 		return ErrNilState | 		return ErrNilState | ||||||
| @@ -805,6 +814,14 @@ func (s *State) OnInterface(se *Session, i interface{}) (err error) { | |||||||
| 	case *GuildDelete: | 	case *GuildDelete: | ||||||
| 		err = s.GuildRemove(t.Guild) | 		err = s.GuildRemove(t.Guild) | ||||||
| 	case *GuildMemberAdd: | 	case *GuildMemberAdd: | ||||||
|  | 		// Updates the MemberCount of the guild. | ||||||
|  | 		guild, err := s.Guild(t.Member.GuildID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		guild.MemberCount++ | ||||||
|  | 
 | ||||||
|  | 		// Caches member if tracking is enabled. | ||||||
| 		if s.TrackMembers { | 		if s.TrackMembers { | ||||||
| 			err = s.MemberAdd(t.Member) | 			err = s.MemberAdd(t.Member) | ||||||
| 		} | 		} | ||||||
| @@ -813,6 +830,14 @@ func (s *State) OnInterface(se *Session, i interface{}) (err error) { | |||||||
| 			err = s.MemberAdd(t.Member) | 			err = s.MemberAdd(t.Member) | ||||||
| 		} | 		} | ||||||
| 	case *GuildMemberRemove: | 	case *GuildMemberRemove: | ||||||
|  | 		// Updates the MemberCount of the guild. | ||||||
|  | 		guild, err := s.Guild(t.Member.GuildID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		guild.MemberCount-- | ||||||
|  | 
 | ||||||
|  | 		// Removes member from the cache if tracking is enabled. | ||||||
| 		if s.TrackMembers { | 		if s.TrackMembers { | ||||||
| 			err = s.MemberRemove(t.Member) | 			err = s.MemberRemove(t.Member) | ||||||
| 		} | 		} | ||||||
| @@ -13,6 +13,7 @@ package discordgo | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -84,6 +85,9 @@ type Session struct { | |||||||
| 	// Stores the last HeartbeatAck that was recieved (in UTC) | 	// Stores the last HeartbeatAck that was recieved (in UTC) | ||||||
| 	LastHeartbeatAck time.Time | 	LastHeartbeatAck time.Time | ||||||
| 
 | 
 | ||||||
|  | 	// Stores the last Heartbeat sent (in UTC) | ||||||
|  | 	LastHeartbeatSent time.Time | ||||||
|  | 
 | ||||||
| 	// used to deal with rate limits | 	// used to deal with rate limits | ||||||
| 	Ratelimiter *RateLimiter | 	Ratelimiter *RateLimiter | ||||||
| 
 | 
 | ||||||
| @@ -111,6 +115,37 @@ type Session struct { | |||||||
| 	wsMutex sync.Mutex | 	wsMutex sync.Mutex | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // UserConnection is a Connection returned from the UserConnections endpoint | ||||||
|  | type UserConnection struct { | ||||||
|  | 	ID           string         `json:"id"` | ||||||
|  | 	Name         string         `json:"name"` | ||||||
|  | 	Type         string         `json:"type"` | ||||||
|  | 	Revoked      bool           `json:"revoked"` | ||||||
|  | 	Integrations []*Integration `json:"integrations"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Integration stores integration information | ||||||
|  | type Integration struct { | ||||||
|  | 	ID                string             `json:"id"` | ||||||
|  | 	Name              string             `json:"name"` | ||||||
|  | 	Type              string             `json:"type"` | ||||||
|  | 	Enabled           bool               `json:"enabled"` | ||||||
|  | 	Syncing           bool               `json:"syncing"` | ||||||
|  | 	RoleID            string             `json:"role_id"` | ||||||
|  | 	ExpireBehavior    int                `json:"expire_behavior"` | ||||||
|  | 	ExpireGracePeriod int                `json:"expire_grace_period"` | ||||||
|  | 	User              *User              `json:"user"` | ||||||
|  | 	Account           IntegrationAccount `json:"account"` | ||||||
|  | 	SyncedAt          Timestamp          `json:"synced_at"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // IntegrationAccount is integration account information | ||||||
|  | // sent by the UserConnections endpoint | ||||||
|  | type IntegrationAccount struct { | ||||||
|  | 	ID   string `json:"id"` | ||||||
|  | 	Name string `json:"name"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // A VoiceRegion stores data for a specific voice region server. | // A VoiceRegion stores data for a specific voice region server. | ||||||
| type VoiceRegion struct { | type VoiceRegion struct { | ||||||
| 	ID       string `json:"id"` | 	ID       string `json:"id"` | ||||||
| @@ -145,6 +180,10 @@ type Invite struct { | |||||||
| 	Revoked   bool      `json:"revoked"` | 	Revoked   bool      `json:"revoked"` | ||||||
| 	Temporary bool      `json:"temporary"` | 	Temporary bool      `json:"temporary"` | ||||||
| 	Unique    bool      `json:"unique"` | 	Unique    bool      `json:"unique"` | ||||||
|  | 
 | ||||||
|  | 	// will only be filled when using InviteWithCounts | ||||||
|  | 	ApproximatePresenceCount int `json:"approximate_presence_count"` | ||||||
|  | 	ApproximateMemberCount   int `json:"approximate_member_count"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ChannelType is the type of a Channel | // ChannelType is the type of a Channel | ||||||
| @@ -161,22 +200,61 @@ const ( | |||||||
| 
 | 
 | ||||||
| // A Channel holds all data related to an individual Discord channel. | // A Channel holds all data related to an individual Discord channel. | ||||||
| type Channel struct { | type Channel struct { | ||||||
| 	ID                   string                 `json:"id"` | 	// The ID of the channel. | ||||||
| 	GuildID              string                 `json:"guild_id"` | 	ID string `json:"id"` | ||||||
| 	Name                 string                 `json:"name"` | 
 | ||||||
| 	Topic                string                 `json:"topic"` | 	// The ID of the guild to which the channel belongs, if it is in a guild. | ||||||
| 	Type                 ChannelType            `json:"type"` | 	// Else, this ID is empty (e.g. DM channels). | ||||||
| 	LastMessageID        string                 `json:"last_message_id"` | 	GuildID string `json:"guild_id"` | ||||||
| 	NSFW                 bool                   `json:"nsfw"` | 
 | ||||||
| 	Position             int                    `json:"position"` | 	// The name of the channel. | ||||||
| 	Bitrate              int                    `json:"bitrate"` | 	Name string `json:"name"` | ||||||
| 	Recipients           []*User                `json:"recipients"` | 
 | ||||||
| 	Messages             []*Message             `json:"-"` | 	// The topic of the channel. | ||||||
|  | 	Topic string `json:"topic"` | ||||||
|  | 
 | ||||||
|  | 	// The type of the channel. | ||||||
|  | 	Type ChannelType `json:"type"` | ||||||
|  | 
 | ||||||
|  | 	// The ID of the last message sent in the channel. This is not | ||||||
|  | 	// guaranteed to be an ID of a valid message. | ||||||
|  | 	LastMessageID string `json:"last_message_id"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the channel is marked as NSFW. | ||||||
|  | 	NSFW bool `json:"nsfw"` | ||||||
|  | 
 | ||||||
|  | 	// Icon of the group DM channel. | ||||||
|  | 	Icon string `json:"icon"` | ||||||
|  | 
 | ||||||
|  | 	// The position of the channel, used for sorting in client. | ||||||
|  | 	Position int `json:"position"` | ||||||
|  | 
 | ||||||
|  | 	// The bitrate of the channel, if it is a voice channel. | ||||||
|  | 	Bitrate int `json:"bitrate"` | ||||||
|  | 
 | ||||||
|  | 	// The recipients of the channel. This is only populated in DM channels. | ||||||
|  | 	Recipients []*User `json:"recipients"` | ||||||
|  | 
 | ||||||
|  | 	// The messages in the channel. This is only present in state-cached channels, | ||||||
|  | 	// and State.MaxMessageCount must be non-zero. | ||||||
|  | 	Messages []*Message `json:"-"` | ||||||
|  | 
 | ||||||
|  | 	// A list of permission overwrites present for the channel. | ||||||
| 	PermissionOverwrites []*PermissionOverwrite `json:"permission_overwrites"` | 	PermissionOverwrites []*PermissionOverwrite `json:"permission_overwrites"` | ||||||
| 	ParentID             string                 `json:"parent_id"` | 
 | ||||||
|  | 	// The user limit of the voice channel. | ||||||
|  | 	UserLimit int `json:"user_limit"` | ||||||
|  | 
 | ||||||
|  | 	// The ID of the parent channel, if the channel is under a category | ||||||
|  | 	ParentID string `json:"parent_id"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A ChannelEdit holds Channel Feild data for a channel edit. | // Mention returns a string which mentions the channel | ||||||
|  | func (c *Channel) Mention() string { | ||||||
|  | 	return fmt.Sprintf("<#%s>", c.ID) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // A ChannelEdit holds Channel Field data for a channel edit. | ||||||
| type ChannelEdit struct { | type ChannelEdit struct { | ||||||
| 	Name                 string                 `json:"name,omitempty"` | 	Name                 string                 `json:"name,omitempty"` | ||||||
| 	Topic                string                 `json:"topic,omitempty"` | 	Topic                string                 `json:"topic,omitempty"` | ||||||
| @@ -186,6 +264,7 @@ type ChannelEdit struct { | |||||||
| 	UserLimit            int                    `json:"user_limit,omitempty"` | 	UserLimit            int                    `json:"user_limit,omitempty"` | ||||||
| 	PermissionOverwrites []*PermissionOverwrite `json:"permission_overwrites,omitempty"` | 	PermissionOverwrites []*PermissionOverwrite `json:"permission_overwrites,omitempty"` | ||||||
| 	ParentID             string                 `json:"parent_id,omitempty"` | 	ParentID             string                 `json:"parent_id,omitempty"` | ||||||
|  | 	RateLimitPerUser     int                    `json:"rate_limit_per_user,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A PermissionOverwrite holds permission overwrite data for a Channel | // A PermissionOverwrite holds permission overwrite data for a Channel | ||||||
| @@ -206,6 +285,19 @@ type Emoji struct { | |||||||
| 	Animated      bool     `json:"animated"` | 	Animated      bool     `json:"animated"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // MessageFormat returns a correctly formatted Emoji for use in Message content and embeds | ||||||
|  | func (e *Emoji) MessageFormat() string { | ||||||
|  | 	if e.ID != "" && e.Name != "" { | ||||||
|  | 		if e.Animated { | ||||||
|  | 			return "<a:" + e.APIName() + ">" | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return "<:" + e.APIName() + ">" | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return e.APIName() | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // APIName returns an correctly formatted API name for use in the MessageReactions endpoints. | // APIName returns an correctly formatted API name for use in the MessageReactions endpoints. | ||||||
| func (e *Emoji) APIName() string { | func (e *Emoji) APIName() string { | ||||||
| 	if e.ID != "" && e.Name != "" { | 	if e.ID != "" && e.Name != "" { | ||||||
| @@ -228,31 +320,129 @@ const ( | |||||||
| 	VerificationLevelHigh | 	VerificationLevelHigh | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // ExplicitContentFilterLevel type definition | ||||||
|  | type ExplicitContentFilterLevel int | ||||||
|  | 
 | ||||||
|  | // Constants for ExplicitContentFilterLevel levels from 0 to 2 inclusive | ||||||
|  | const ( | ||||||
|  | 	ExplicitContentFilterDisabled ExplicitContentFilterLevel = iota | ||||||
|  | 	ExplicitContentFilterMembersWithoutRoles | ||||||
|  | 	ExplicitContentFilterAllMembers | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // MfaLevel type definition | ||||||
|  | type MfaLevel int | ||||||
|  | 
 | ||||||
|  | // Constants for MfaLevel levels from 0 to 1 inclusive | ||||||
|  | const ( | ||||||
|  | 	MfaLevelNone MfaLevel = iota | ||||||
|  | 	MfaLevelElevated | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| // A Guild holds all data related to a specific Discord Guild.  Guilds are also | // A Guild holds all data related to a specific Discord Guild.  Guilds are also | ||||||
| // sometimes referred to as Servers in the Discord client. | // sometimes referred to as Servers in the Discord client. | ||||||
| type Guild struct { | type Guild struct { | ||||||
| 	ID                          string            `json:"id"` | 	// The ID of the guild. | ||||||
| 	Name                        string            `json:"name"` | 	ID string `json:"id"` | ||||||
| 	Icon                        string            `json:"icon"` | 
 | ||||||
| 	Region                      string            `json:"region"` | 	// The name of the guild. (2–100 characters) | ||||||
| 	AfkChannelID                string            `json:"afk_channel_id"` | 	Name string `json:"name"` | ||||||
| 	EmbedChannelID              string            `json:"embed_channel_id"` | 
 | ||||||
| 	OwnerID                     string            `json:"owner_id"` | 	// The hash of the guild's icon. Use Session.GuildIcon | ||||||
| 	JoinedAt                    Timestamp         `json:"joined_at"` | 	// to retrieve the icon itself. | ||||||
| 	Splash                      string            `json:"splash"` | 	Icon string `json:"icon"` | ||||||
| 	AfkTimeout                  int               `json:"afk_timeout"` | 
 | ||||||
| 	MemberCount                 int               `json:"member_count"` | 	// The voice region of the guild. | ||||||
| 	VerificationLevel           VerificationLevel `json:"verification_level"` | 	Region string `json:"region"` | ||||||
| 	EmbedEnabled                bool              `json:"embed_enabled"` | 
 | ||||||
| 	Large                       bool              `json:"large"` // ?? | 	// The ID of the AFK voice channel. | ||||||
| 	DefaultMessageNotifications int               `json:"default_message_notifications"` | 	AfkChannelID string `json:"afk_channel_id"` | ||||||
| 	Roles                       []*Role           `json:"roles"` | 
 | ||||||
| 	Emojis                      []*Emoji          `json:"emojis"` | 	// The ID of the embed channel ID, used for embed widgets. | ||||||
| 	Members                     []*Member         `json:"members"` | 	EmbedChannelID string `json:"embed_channel_id"` | ||||||
| 	Presences                   []*Presence       `json:"presences"` | 
 | ||||||
| 	Channels                    []*Channel        `json:"channels"` | 	// The user ID of the owner of the guild. | ||||||
| 	VoiceStates                 []*VoiceState     `json:"voice_states"` | 	OwnerID string `json:"owner_id"` | ||||||
| 	Unavailable                 bool              `json:"unavailable"` | 
 | ||||||
|  | 	// The time at which the current user joined the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	JoinedAt Timestamp `json:"joined_at"` | ||||||
|  | 
 | ||||||
|  | 	// The hash of the guild's splash. | ||||||
|  | 	Splash string `json:"splash"` | ||||||
|  | 
 | ||||||
|  | 	// The timeout, in seconds, before a user is considered AFK in voice. | ||||||
|  | 	AfkTimeout int `json:"afk_timeout"` | ||||||
|  | 
 | ||||||
|  | 	// The number of members in the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	MemberCount int `json:"member_count"` | ||||||
|  | 
 | ||||||
|  | 	// The verification level required for the guild. | ||||||
|  | 	VerificationLevel VerificationLevel `json:"verification_level"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the guild has embedding enabled. | ||||||
|  | 	EmbedEnabled bool `json:"embed_enabled"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the guild is considered large. This is | ||||||
|  | 	// determined by a member threshold in the identify packet, | ||||||
|  | 	// and is currently hard-coded at 250 members in the library. | ||||||
|  | 	Large bool `json:"large"` | ||||||
|  | 
 | ||||||
|  | 	// The default message notification setting for the guild. | ||||||
|  | 	// 0 == all messages, 1 == mentions only. | ||||||
|  | 	DefaultMessageNotifications int `json:"default_message_notifications"` | ||||||
|  | 
 | ||||||
|  | 	// A list of roles in the guild. | ||||||
|  | 	Roles []*Role `json:"roles"` | ||||||
|  | 
 | ||||||
|  | 	// A list of the custom emojis present in the guild. | ||||||
|  | 	Emojis []*Emoji `json:"emojis"` | ||||||
|  | 
 | ||||||
|  | 	// A list of the members in the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	Members []*Member `json:"members"` | ||||||
|  | 
 | ||||||
|  | 	// A list of partial presence objects for members in the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	Presences []*Presence `json:"presences"` | ||||||
|  | 
 | ||||||
|  | 	// A list of channels in the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	Channels []*Channel `json:"channels"` | ||||||
|  | 
 | ||||||
|  | 	// A list of voice states for the guild. | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	VoiceStates []*VoiceState `json:"voice_states"` | ||||||
|  | 
 | ||||||
|  | 	// Whether this guild is currently unavailable (most likely due to outage). | ||||||
|  | 	// This field is only present in GUILD_CREATE events and websocket | ||||||
|  | 	// update events, and thus is only present in state-cached guilds. | ||||||
|  | 	Unavailable bool `json:"unavailable"` | ||||||
|  | 
 | ||||||
|  | 	// The explicit content filter level | ||||||
|  | 	ExplicitContentFilter ExplicitContentFilterLevel `json:"explicit_content_filter"` | ||||||
|  | 
 | ||||||
|  | 	// The list of enabled guild features | ||||||
|  | 	Features []string `json:"features"` | ||||||
|  | 
 | ||||||
|  | 	// Required MFA level for the guild | ||||||
|  | 	MfaLevel MfaLevel `json:"mfa_level"` | ||||||
|  | 
 | ||||||
|  | 	// Whether or not the Server Widget is enabled | ||||||
|  | 	WidgetEnabled bool `json:"widget_enabled"` | ||||||
|  | 
 | ||||||
|  | 	// The Channel ID for the Server Widget | ||||||
|  | 	WidgetChannelID string `json:"widget_channel_id"` | ||||||
|  | 
 | ||||||
|  | 	// The Channel ID to which system messages are sent (eg join and leave messages) | ||||||
|  | 	SystemChannelID string `json:"system_channel_id"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A UserGuild holds a brief version of a Guild | // A UserGuild holds a brief version of a Guild | ||||||
| @@ -279,14 +469,37 @@ type GuildParams struct { | |||||||
| 
 | 
 | ||||||
| // A Role stores information about Discord guild member roles. | // A Role stores information about Discord guild member roles. | ||||||
| type Role struct { | type Role struct { | ||||||
| 	ID          string `json:"id"` | 	// The ID of the role. | ||||||
| 	Name        string `json:"name"` | 	ID string `json:"id"` | ||||||
| 	Managed     bool   `json:"managed"` | 
 | ||||||
| 	Mentionable bool   `json:"mentionable"` | 	// The name of the role. | ||||||
| 	Hoist       bool   `json:"hoist"` | 	Name string `json:"name"` | ||||||
| 	Color       int    `json:"color"` | 
 | ||||||
| 	Position    int    `json:"position"` | 	// Whether this role is managed by an integration, and | ||||||
| 	Permissions int    `json:"permissions"` | 	// thus cannot be manually added to, or taken from, members. | ||||||
|  | 	Managed bool `json:"managed"` | ||||||
|  | 
 | ||||||
|  | 	// Whether this role is mentionable. | ||||||
|  | 	Mentionable bool `json:"mentionable"` | ||||||
|  | 
 | ||||||
|  | 	// Whether this role is hoisted (shows up separately in member list). | ||||||
|  | 	Hoist bool `json:"hoist"` | ||||||
|  | 
 | ||||||
|  | 	// The hex color of this role. | ||||||
|  | 	Color int `json:"color"` | ||||||
|  | 
 | ||||||
|  | 	// The position of this role in the guild's role hierarchy. | ||||||
|  | 	Position int `json:"position"` | ||||||
|  | 
 | ||||||
|  | 	// The permissions of the role on the guild (doesn't include channel overrides). | ||||||
|  | 	// This is a combination of bit masks; the presence of a certain permission can | ||||||
|  | 	// be checked by performing a bitwise AND between this int and the permission. | ||||||
|  | 	Permissions int `json:"permissions"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Mention returns a string which mentions the role | ||||||
|  | func (r *Role) Mention() string { | ||||||
|  | 	return fmt.Sprintf("<@&%s>", r.ID) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Roles are a collection of Role | // Roles are a collection of Role | ||||||
| @@ -334,6 +547,8 @@ type GameType int | |||||||
| const ( | const ( | ||||||
| 	GameTypeGame GameType = iota | 	GameTypeGame GameType = iota | ||||||
| 	GameTypeStreaming | 	GameTypeStreaming | ||||||
|  | 	GameTypeListening | ||||||
|  | 	GameTypeWatching | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // A Game struct holds the name of the "playing .." game for a user | // A Game struct holds the name of the "playing .." game for a user | ||||||
| @@ -379,15 +594,34 @@ type Assets struct { | |||||||
| 	SmallText    string `json:"small_text,omitempty"` | 	SmallText    string `json:"small_text,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A Member stores user information for Guild members. | // A Member stores user information for Guild members. A guild | ||||||
|  | // member represents a certain user's presence in a guild. | ||||||
| type Member struct { | type Member struct { | ||||||
| 	GuildID  string   `json:"guild_id"` | 	// The guild ID on which the member exists. | ||||||
| 	JoinedAt string   `json:"joined_at"` | 	GuildID string `json:"guild_id"` | ||||||
| 	Nick     string   `json:"nick"` | 
 | ||||||
| 	Deaf     bool     `json:"deaf"` | 	// The time at which the member joined the guild, in ISO8601. | ||||||
| 	Mute     bool     `json:"mute"` | 	JoinedAt Timestamp `json:"joined_at"` | ||||||
| 	User     *User    `json:"user"` | 
 | ||||||
| 	Roles    []string `json:"roles"` | 	// The nickname of the member, if they have one. | ||||||
|  | 	Nick string `json:"nick"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the member is deafened at a guild level. | ||||||
|  | 	Deaf bool `json:"deaf"` | ||||||
|  | 
 | ||||||
|  | 	// Whether the member is muted at a guild level. | ||||||
|  | 	Mute bool `json:"mute"` | ||||||
|  | 
 | ||||||
|  | 	// The underlying user on which the member is based. | ||||||
|  | 	User *User `json:"user"` | ||||||
|  | 
 | ||||||
|  | 	// A list of IDs of the roles which are possessed by the member. | ||||||
|  | 	Roles []string `json:"roles"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Mention creates a member mention | ||||||
|  | func (m *Member) Mention() string { | ||||||
|  | 	return "<@!" + m.User.ID + ">" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A Settings stores data for a specific users Discord client settings. | // A Settings stores data for a specific users Discord client settings. | ||||||
| @@ -467,33 +701,88 @@ type GuildBan struct { | |||||||
| 	User   *User  `json:"user"` | 	User   *User  `json:"user"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // A GuildIntegration stores data for a guild integration. |  | ||||||
| type GuildIntegration struct { |  | ||||||
| 	ID                string                   `json:"id"` |  | ||||||
| 	Name              string                   `json:"name"` |  | ||||||
| 	Type              string                   `json:"type"` |  | ||||||
| 	Enabled           bool                     `json:"enabled"` |  | ||||||
| 	Syncing           bool                     `json:"syncing"` |  | ||||||
| 	RoleID            string                   `json:"role_id"` |  | ||||||
| 	ExpireBehavior    int                      `json:"expire_behavior"` |  | ||||||
| 	ExpireGracePeriod int                      `json:"expire_grace_period"` |  | ||||||
| 	User              *User                    `json:"user"` |  | ||||||
| 	Account           *GuildIntegrationAccount `json:"account"` |  | ||||||
| 	SyncedAt          int                      `json:"synced_at"` |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // A GuildIntegrationAccount stores data for a guild integration account. |  | ||||||
| type GuildIntegrationAccount struct { |  | ||||||
| 	ID   string `json:"id"` |  | ||||||
| 	Name string `json:"name"` |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // A GuildEmbed stores data for a guild embed. | // A GuildEmbed stores data for a guild embed. | ||||||
| type GuildEmbed struct { | type GuildEmbed struct { | ||||||
| 	Enabled   bool   `json:"enabled"` | 	Enabled   bool   `json:"enabled"` | ||||||
| 	ChannelID string `json:"channel_id"` | 	ChannelID string `json:"channel_id"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // A GuildAuditLog stores data for a guild audit log. | ||||||
|  | type GuildAuditLog struct { | ||||||
|  | 	Webhooks []struct { | ||||||
|  | 		ChannelID string `json:"channel_id"` | ||||||
|  | 		GuildID   string `json:"guild_id"` | ||||||
|  | 		ID        string `json:"id"` | ||||||
|  | 		Avatar    string `json:"avatar"` | ||||||
|  | 		Name      string `json:"name"` | ||||||
|  | 	} `json:"webhooks,omitempty"` | ||||||
|  | 	Users []struct { | ||||||
|  | 		Username      string `json:"username"` | ||||||
|  | 		Discriminator string `json:"discriminator"` | ||||||
|  | 		Bot           bool   `json:"bot"` | ||||||
|  | 		ID            string `json:"id"` | ||||||
|  | 		Avatar        string `json:"avatar"` | ||||||
|  | 	} `json:"users,omitempty"` | ||||||
|  | 	AuditLogEntries []struct { | ||||||
|  | 		TargetID string `json:"target_id"` | ||||||
|  | 		Changes  []struct { | ||||||
|  | 			NewValue interface{} `json:"new_value"` | ||||||
|  | 			OldValue interface{} `json:"old_value"` | ||||||
|  | 			Key      string      `json:"key"` | ||||||
|  | 		} `json:"changes,omitempty"` | ||||||
|  | 		UserID     string `json:"user_id"` | ||||||
|  | 		ID         string `json:"id"` | ||||||
|  | 		ActionType int    `json:"action_type"` | ||||||
|  | 		Options    struct { | ||||||
|  | 			DeleteMembersDay string `json:"delete_member_days"` | ||||||
|  | 			MembersRemoved   string `json:"members_removed"` | ||||||
|  | 			ChannelID        string `json:"channel_id"` | ||||||
|  | 			Count            string `json:"count"` | ||||||
|  | 			ID               string `json:"id"` | ||||||
|  | 			Type             string `json:"type"` | ||||||
|  | 			RoleName         string `json:"role_name"` | ||||||
|  | 		} `json:"options,omitempty"` | ||||||
|  | 		Reason string `json:"reason"` | ||||||
|  | 	} `json:"audit_log_entries"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Block contains Discord Audit Log Action Types | ||||||
|  | const ( | ||||||
|  | 	AuditLogActionGuildUpdate = 1 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionChannelCreate          = 10 | ||||||
|  | 	AuditLogActionChannelUpdate          = 11 | ||||||
|  | 	AuditLogActionChannelDelete          = 12 | ||||||
|  | 	AuditLogActionChannelOverwriteCreate = 13 | ||||||
|  | 	AuditLogActionChannelOverwriteUpdate = 14 | ||||||
|  | 	AuditLogActionChannelOverwriteDelete = 15 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionMemberKick       = 20 | ||||||
|  | 	AuditLogActionMemberPrune      = 21 | ||||||
|  | 	AuditLogActionMemberBanAdd     = 22 | ||||||
|  | 	AuditLogActionMemberBanRemove  = 23 | ||||||
|  | 	AuditLogActionMemberUpdate     = 24 | ||||||
|  | 	AuditLogActionMemberRoleUpdate = 25 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionRoleCreate = 30 | ||||||
|  | 	AuditLogActionRoleUpdate = 31 | ||||||
|  | 	AuditLogActionRoleDelete = 32 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionInviteCreate = 40 | ||||||
|  | 	AuditLogActionInviteUpdate = 41 | ||||||
|  | 	AuditLogActionInviteDelete = 42 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionWebhookCreate = 50 | ||||||
|  | 	AuditLogActionWebhookUpdate = 51 | ||||||
|  | 	AuditLogActionWebhookDelete = 52 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionEmojiCreate = 60 | ||||||
|  | 	AuditLogActionEmojiUpdate = 61 | ||||||
|  | 	AuditLogActionEmojiDelete = 62 | ||||||
|  | 
 | ||||||
|  | 	AuditLogActionMessageDelete = 72 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| // A UserGuildSettingsChannelOverride stores data for a channel override for a users guild settings. | // A UserGuildSettingsChannelOverride stores data for a channel override for a users guild settings. | ||||||
| type UserGuildSettingsChannelOverride struct { | type UserGuildSettingsChannelOverride struct { | ||||||
| 	Muted                bool   `json:"muted"` | 	Muted                bool   `json:"muted"` | ||||||
| @@ -553,6 +842,7 @@ type MessageReaction struct { | |||||||
| 	MessageID string `json:"message_id"` | 	MessageID string `json:"message_id"` | ||||||
| 	Emoji     Emoji  `json:"emoji"` | 	Emoji     Emoji  `json:"emoji"` | ||||||
| 	ChannelID string `json:"channel_id"` | 	ChannelID string `json:"channel_id"` | ||||||
|  | 	GuildID   string `json:"guild_id,omitempty"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GatewayBotResponse stores the data for the gateway/bot response | // GatewayBotResponse stores the data for the gateway/bot response | ||||||
| @@ -629,7 +919,9 @@ const ( | |||||||
| 		PermissionKickMembers | | 		PermissionKickMembers | | ||||||
| 		PermissionBanMembers | | 		PermissionBanMembers | | ||||||
| 		PermissionManageServer | | 		PermissionManageServer | | ||||||
| 		PermissionAdministrator | 		PermissionAdministrator | | ||||||
|  | 		PermissionManageWebhooks | | ||||||
|  | 		PermissionManageEmojis | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // Block contains Discord JSON Error Response codes | // Block contains Discord JSON Error Response codes | ||||||
| @@ -648,6 +940,7 @@ const ( | |||||||
| 	ErrCodeUnknownToken       = 10012 | 	ErrCodeUnknownToken       = 10012 | ||||||
| 	ErrCodeUnknownUser        = 10013 | 	ErrCodeUnknownUser        = 10013 | ||||||
| 	ErrCodeUnknownEmoji       = 10014 | 	ErrCodeUnknownEmoji       = 10014 | ||||||
|  | 	ErrCodeUnknownWebhook     = 10015 | ||||||
| 
 | 
 | ||||||
| 	ErrCodeBotsCannotUseEndpoint  = 20001 | 	ErrCodeBotsCannotUseEndpoint  = 20001 | ||||||
| 	ErrCodeOnlyBotsCanUseEndpoint = 20002 | 	ErrCodeOnlyBotsCanUseEndpoint = 20002 | ||||||
| @@ -11,7 +11,6 @@ package discordgo | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" |  | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -54,5 +53,5 @@ func newRestError(req *http.Request, resp *http.Response, body []byte) *RESTErro | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (r RESTError) Error() string { | func (r RESTError) Error() string { | ||||||
| 	return fmt.Sprintf("HTTP %s, %s", r.Response.Status, r.ResponseBody) | 	return "HTTP " + r.Response.Status + ", " + string(r.ResponseBody) | ||||||
| } | } | ||||||
							
								
								
									
										69
									
								
								vendor/github.com/bwmarrin/discordgo/user.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | |||||||
|  | package discordgo | ||||||
|  |  | ||||||
|  | import "strings" | ||||||
|  |  | ||||||
|  | // A User stores all data for an individual Discord user. | ||||||
|  | type User struct { | ||||||
|  | 	// The ID of the user. | ||||||
|  | 	ID string `json:"id"` | ||||||
|  |  | ||||||
|  | 	// The email of the user. This is only present when | ||||||
|  | 	// the application possesses the email scope for the user. | ||||||
|  | 	Email string `json:"email"` | ||||||
|  |  | ||||||
|  | 	// The user's username. | ||||||
|  | 	Username string `json:"username"` | ||||||
|  |  | ||||||
|  | 	// The hash of the user's avatar. Use Session.UserAvatar | ||||||
|  | 	// to retrieve the avatar itself. | ||||||
|  | 	Avatar string `json:"avatar"` | ||||||
|  |  | ||||||
|  | 	// The user's chosen language option. | ||||||
|  | 	Locale string `json:"locale"` | ||||||
|  |  | ||||||
|  | 	// The discriminator of the user (4 numbers after name). | ||||||
|  | 	Discriminator string `json:"discriminator"` | ||||||
|  |  | ||||||
|  | 	// The token of the user. This is only present for | ||||||
|  | 	// the user represented by the current session. | ||||||
|  | 	Token string `json:"token"` | ||||||
|  |  | ||||||
|  | 	// Whether the user's email is verified. | ||||||
|  | 	Verified bool `json:"verified"` | ||||||
|  |  | ||||||
|  | 	// Whether the user has multi-factor authentication enabled. | ||||||
|  | 	MFAEnabled bool `json:"mfa_enabled"` | ||||||
|  |  | ||||||
|  | 	// Whether the user is a bot. | ||||||
|  | 	Bot bool `json:"bot"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String returns a unique identifier of the form username#discriminator | ||||||
|  | func (u *User) String() string { | ||||||
|  | 	return u.Username + "#" + u.Discriminator | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Mention return a string which mentions the user | ||||||
|  | func (u *User) Mention() string { | ||||||
|  | 	return "<@" + u.ID + ">" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // AvatarURL returns a URL to the user's avatar. | ||||||
|  | //    size:    The size of the user's avatar as a power of two | ||||||
|  | //             if size is an empty string, no size parameter will | ||||||
|  | //             be added to the URL. | ||||||
|  | func (u *User) AvatarURL(size string) string { | ||||||
|  | 	var URL string | ||||||
|  | 	if u.Avatar == "" { | ||||||
|  | 		URL = EndpointDefaultUserAvatar(u.Discriminator) | ||||||
|  | 	} else if strings.HasPrefix(u.Avatar, "a_") { | ||||||
|  | 		URL = EndpointUserAvatarAnimated(u.ID, u.Avatar) | ||||||
|  | 	} else { | ||||||
|  | 		URL = EndpointUserAvatar(u.ID, u.Avatar) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if size != "" { | ||||||
|  | 		return URL + "?size=" + size | ||||||
|  | 	} | ||||||
|  | 	return URL | ||||||
|  | } | ||||||
| @@ -14,6 +14,7 @@ import ( | |||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net" | 	"net" | ||||||
|  | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -103,7 +104,7 @@ func (v *VoiceConnection) Speaking(b bool) (err error) { | |||||||
| 	defer v.Unlock() | 	defer v.Unlock() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		v.speaking = false | 		v.speaking = false | ||||||
| 		v.log(LogError, "Speaking() write json error:", err) | 		v.log(LogError, "Speaking() write json error, %s", err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @@ -135,7 +136,6 @@ func (v *VoiceConnection) ChangeChannel(channelID string, mute, deaf bool) (err | |||||||
| 
 | 
 | ||||||
| // Disconnect disconnects from this voice channel and closes the websocket | // Disconnect disconnects from this voice channel and closes the websocket | ||||||
| // and udp connections to Discord. | // and udp connections to Discord. | ||||||
| // !!! NOTE !!! this function may be removed in favour of ChannelVoiceLeave |  | ||||||
| func (v *VoiceConnection) Disconnect() (err error) { | func (v *VoiceConnection) Disconnect() (err error) { | ||||||
| 
 | 
 | ||||||
| 	// Send a OP4 with a nil channel to disconnect | 	// Send a OP4 with a nil channel to disconnect | ||||||
| @@ -180,7 +180,7 @@ func (v *VoiceConnection) Close() { | |||||||
| 		v.log(LogInformational, "closing udp") | 		v.log(LogInformational, "closing udp") | ||||||
| 		err := v.udpConn.Close() | 		err := v.udpConn.Close() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			v.log(LogError, "error closing udp connection: ", err) | 			v.log(LogError, "error closing udp connection, %s", err) | ||||||
| 		} | 		} | ||||||
| 		v.udpConn = nil | 		v.udpConn = nil | ||||||
| 	} | 	} | ||||||
| @@ -299,7 +299,7 @@ func (v *VoiceConnection) open() (err error) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Connect to VoiceConnection Websocket | 	// Connect to VoiceConnection Websocket | ||||||
| 	vg := fmt.Sprintf("wss://%s", strings.TrimSuffix(v.endpoint, ":80")) | 	vg := "wss://" + strings.TrimSuffix(v.endpoint, ":80") | ||||||
| 	v.log(LogInformational, "connecting to voice endpoint %s", vg) | 	v.log(LogInformational, "connecting to voice endpoint %s", vg) | ||||||
| 	v.wsConn, _, err = websocket.DefaultDialer.Dial(vg, nil) | 	v.wsConn, _, err = websocket.DefaultDialer.Dial(vg, nil) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -542,7 +542,7 @@ func (v *VoiceConnection) udpOpen() (err error) { | |||||||
| 		return fmt.Errorf("empty endpoint") | 		return fmt.Errorf("empty endpoint") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	host := fmt.Sprintf("%s:%d", strings.TrimSuffix(v.endpoint, ":80"), v.op2.Port) | 	host := strings.TrimSuffix(v.endpoint, ":80") + ":" + strconv.Itoa(v.op2.Port) | ||||||
| 	addr, err := net.ResolveUDPAddr("udp", host) | 	addr, err := net.ResolveUDPAddr("udp", host) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		v.log(LogWarning, "error resolving udp host %s, %s", host, err) | 		v.log(LogWarning, "error resolving udp host %s, %s", host, err) | ||||||
| @@ -86,6 +86,10 @@ func (s *Session) Open() error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	s.wsConn.SetCloseHandler(func(code int, text string) error { | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		// because of this, all code below must set err to the error | 		// because of this, all code below must set err to the error | ||||||
| 		// when exiting with an error :)  Maybe someone has a better | 		// when exiting with an error :)  Maybe someone has a better | ||||||
| @@ -263,6 +267,13 @@ type helloOp struct { | |||||||
| // FailedHeartbeatAcks is the Number of heartbeat intervals to wait until forcing a connection restart. | // FailedHeartbeatAcks is the Number of heartbeat intervals to wait until forcing a connection restart. | ||||||
| const FailedHeartbeatAcks time.Duration = 5 * time.Millisecond | const FailedHeartbeatAcks time.Duration = 5 * time.Millisecond | ||||||
| 
 | 
 | ||||||
|  | // HeartbeatLatency returns the latency between heartbeat acknowledgement and heartbeat send. | ||||||
|  | func (s *Session) HeartbeatLatency() time.Duration { | ||||||
|  | 
 | ||||||
|  | 	return s.LastHeartbeatAck.Sub(s.LastHeartbeatSent) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // heartbeat sends regular heartbeats to Discord so it knows the client | // heartbeat sends regular heartbeats to Discord so it knows the client | ||||||
| // is still connected.  If you do not send these heartbeats Discord will | // is still connected.  If you do not send these heartbeats Discord will | ||||||
| // disconnect the websocket connection after a few seconds. | // disconnect the websocket connection after a few seconds. | ||||||
| @@ -283,8 +294,9 @@ func (s *Session) heartbeat(wsConn *websocket.Conn, listening <-chan interface{} | |||||||
| 		last := s.LastHeartbeatAck | 		last := s.LastHeartbeatAck | ||||||
| 		s.RUnlock() | 		s.RUnlock() | ||||||
| 		sequence := atomic.LoadInt64(s.sequence) | 		sequence := atomic.LoadInt64(s.sequence) | ||||||
| 		s.log(LogInformational, "sending gateway websocket heartbeat seq %d", sequence) | 		s.log(LogDebug, "sending gateway websocket heartbeat seq %d", sequence) | ||||||
| 		s.wsMutex.Lock() | 		s.wsMutex.Lock() | ||||||
|  | 		s.LastHeartbeatSent = time.Now().UTC() | ||||||
| 		err = wsConn.WriteJSON(heartbeatOp{1, sequence}) | 		err = wsConn.WriteJSON(heartbeatOp{1, sequence}) | ||||||
| 		s.wsMutex.Unlock() | 		s.wsMutex.Unlock() | ||||||
| 		if err != nil || time.Now().UTC().Sub(last) > (heartbeatIntervalMsec*FailedHeartbeatAcks) { | 		if err != nil || time.Now().UTC().Sub(last) > (heartbeatIntervalMsec*FailedHeartbeatAcks) { | ||||||
| @@ -323,16 +335,8 @@ type updateStatusOp struct { | |||||||
| 	Data UpdateStatusData `json:"d"` | 	Data UpdateStatusData `json:"d"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // UpdateStreamingStatus is used to update the user's streaming status. | func newUpdateStatusData(idle int, gameType GameType, game, url string) *UpdateStatusData { | ||||||
| // If idle>0 then set status to idle. | 	usd := &UpdateStatusData{ | ||||||
| // If game!="" then set game. |  | ||||||
| // If game!="" and url!="" then set the status type to streaming with the URL set. |  | ||||||
| // if otherwise, set status to active, and no game. |  | ||||||
| func (s *Session) UpdateStreamingStatus(idle int, game string, url string) (err error) { |  | ||||||
| 
 |  | ||||||
| 	s.log(LogInformational, "called") |  | ||||||
| 
 |  | ||||||
| 	usd := UpdateStatusData{ |  | ||||||
| 		Status: "online", | 		Status: "online", | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @@ -341,10 +345,6 @@ func (s *Session) UpdateStreamingStatus(idle int, game string, url string) (err | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if game != "" { | 	if game != "" { | ||||||
| 		gameType := GameTypeGame |  | ||||||
| 		if url != "" { |  | ||||||
| 			gameType = GameTypeStreaming |  | ||||||
| 		} |  | ||||||
| 		usd.Game = &Game{ | 		usd.Game = &Game{ | ||||||
| 			Name: game, | 			Name: game, | ||||||
| 			Type: gameType, | 			Type: gameType, | ||||||
| @@ -352,7 +352,35 @@ func (s *Session) UpdateStreamingStatus(idle int, game string, url string) (err | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return s.UpdateStatusComplex(usd) | 	return usd | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // UpdateStatus is used to update the user's status. | ||||||
|  | // If idle>0 then set status to idle. | ||||||
|  | // If game!="" then set game. | ||||||
|  | // if otherwise, set status to active, and no game. | ||||||
|  | func (s *Session) UpdateStatus(idle int, game string) (err error) { | ||||||
|  | 	return s.UpdateStatusComplex(*newUpdateStatusData(idle, GameTypeGame, game, "")) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // UpdateStreamingStatus is used to update the user's streaming status. | ||||||
|  | // If idle>0 then set status to idle. | ||||||
|  | // If game!="" then set game. | ||||||
|  | // If game!="" and url!="" then set the status type to streaming with the URL set. | ||||||
|  | // if otherwise, set status to active, and no game. | ||||||
|  | func (s *Session) UpdateStreamingStatus(idle int, game string, url string) (err error) { | ||||||
|  | 	gameType := GameTypeGame | ||||||
|  | 	if url != "" { | ||||||
|  | 		gameType = GameTypeStreaming | ||||||
|  | 	} | ||||||
|  | 	return s.UpdateStatusComplex(*newUpdateStatusData(idle, gameType, game, url)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // UpdateListeningStatus is used to set the user to "Listening to..." | ||||||
|  | // If game!="" then set to what user is listening to | ||||||
|  | // Else, set user to active and no game. | ||||||
|  | func (s *Session) UpdateListeningStatus(game string) (err error) { | ||||||
|  | 	return s.UpdateStatusComplex(*newUpdateStatusData(0, GameTypeListening, game, "")) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // UpdateStatusComplex allows for sending the raw status update data untouched by discordgo. | // UpdateStatusComplex allows for sending the raw status update data untouched by discordgo. | ||||||
| @@ -371,14 +399,6 @@ func (s *Session) UpdateStatusComplex(usd UpdateStatusData) (err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // UpdateStatus is used to update the user's status. |  | ||||||
| // If idle>0 then set status to idle. |  | ||||||
| // If game!="" then set game. |  | ||||||
| // if otherwise, set status to active, and no game. |  | ||||||
| func (s *Session) UpdateStatus(idle int, game string) (err error) { |  | ||||||
| 	return s.UpdateStreamingStatus(idle, game, "") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type requestGuildMembersData struct { | type requestGuildMembersData struct { | ||||||
| 	GuildID string `json:"guild_id"` | 	GuildID string `json:"guild_id"` | ||||||
| 	Query   string `json:"query"` | 	Query   string `json:"query"` | ||||||
| @@ -508,7 +528,7 @@ func (s *Session) onEvent(messageType int, message []byte) (*Event, error) { | |||||||
| 		s.Lock() | 		s.Lock() | ||||||
| 		s.LastHeartbeatAck = time.Now().UTC() | 		s.LastHeartbeatAck = time.Now().UTC() | ||||||
| 		s.Unlock() | 		s.Unlock() | ||||||
| 		s.log(LogInformational, "got heartbeat ACK") | 		s.log(LogDebug, "got heartbeat ACK") | ||||||
| 		return e, nil | 		return e, nil | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @@ -615,6 +635,30 @@ func (s *Session) ChannelVoiceJoin(gID, cID string, mute, deaf bool) (voice *Voi | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ChannelVoiceJoinManual initiates a voice session to a voice channel, but does not complete it. | ||||||
|  | // | ||||||
|  | // This should only be used when the VoiceServerUpdate will be intercepted and used elsewhere. | ||||||
|  | // | ||||||
|  | //    gID     : Guild ID of the channel to join. | ||||||
|  | //    cID     : Channel ID of the channel to join. | ||||||
|  | //    mute    : If true, you will be set to muted upon joining. | ||||||
|  | //    deaf    : If true, you will be set to deafened upon joining. | ||||||
|  | func (s *Session) ChannelVoiceJoinManual(gID, cID string, mute, deaf bool) (err error) { | ||||||
|  | 
 | ||||||
|  | 	s.log(LogInformational, "called") | ||||||
|  | 
 | ||||||
|  | 	// Send the request to Discord that we want to join the voice channel | ||||||
|  | 	data := voiceChannelJoinOp{4, voiceChannelJoinData{&gID, &cID, mute, deaf}} | ||||||
|  | 	s.wsMutex.Lock() | ||||||
|  | 	err = s.wsConn.WriteJSON(data) | ||||||
|  | 	s.wsMutex.Unlock() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // onVoiceStateUpdate handles Voice State Update events on the data websocket. | // onVoiceStateUpdate handles Voice State Update events on the data websocket. | ||||||
| func (s *Session) onVoiceStateUpdate(st *VoiceStateUpdate) { | func (s *Session) onVoiceStateUpdate(st *VoiceStateUpdate) { | ||||||
| 
 | 
 | ||||||
| @@ -732,11 +776,8 @@ func (s *Session) identify() error { | |||||||
| 	s.wsMutex.Lock() | 	s.wsMutex.Lock() | ||||||
| 	err := s.wsConn.WriteJSON(op) | 	err := s.wsConn.WriteJSON(op) | ||||||
| 	s.wsMutex.Unlock() | 	s.wsMutex.Unlock() | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Session) reconnect() { | func (s *Session) reconnect() { | ||||||
							
								
								
									
										2
									
								
								vendor/github.com/davecgh/go-spew/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -2,7 +2,7 @@ ISC License | |||||||
|  |  | ||||||
| Copyright (c) 2012-2016 Dave Collins <dave@davec.name> | Copyright (c) 2012-2016 Dave Collins <dave@davec.name> | ||||||
|  |  | ||||||
| Permission to use, copy, modify, and distribute this software for any | Permission to use, copy, modify, and/or distribute this software for any | ||||||
| purpose with or without fee is hereby granted, provided that the above | purpose with or without fee is hereby granted, provided that the above | ||||||
| copyright notice and this permission notice appear in all copies. | copyright notice and this permission notice appear in all copies. | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										187
									
								
								vendor/github.com/davecgh/go-spew/spew/bypass.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,9 @@ | |||||||
| // when the code is not running on Google App Engine, compiled by GopherJS, and | // when the code is not running on Google App Engine, compiled by GopherJS, and | ||||||
| // "-tags safe" is not added to the go build command line.  The "disableunsafe" | // "-tags safe" is not added to the go build command line.  The "disableunsafe" | ||||||
| // tag is deprecated and thus should not be used. | // tag is deprecated and thus should not be used. | ||||||
| // +build !js,!appengine,!safe,!disableunsafe | // Go versions prior to 1.4 are disabled because they use a different layout | ||||||
|  | // for interfaces which make the implementation of unsafeReflectValue more complex. | ||||||
|  | // +build !js,!appengine,!safe,!disableunsafe,go1.4 | ||||||
|  |  | ||||||
| package spew | package spew | ||||||
|  |  | ||||||
| @@ -34,80 +36,49 @@ const ( | |||||||
| 	ptrSize = unsafe.Sizeof((*byte)(nil)) | 	ptrSize = unsafe.Sizeof((*byte)(nil)) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | type flag uintptr | ||||||
| 	// offsetPtr, offsetScalar, and offsetFlag are the offsets for the |  | ||||||
| 	// internal reflect.Value fields.  These values are valid before golang |  | ||||||
| 	// commit ecccf07e7f9d which changed the format.  The are also valid |  | ||||||
| 	// after commit 82f48826c6c7 which changed the format again to mirror |  | ||||||
| 	// the original format.  Code in the init function updates these offsets |  | ||||||
| 	// as necessary. |  | ||||||
| 	offsetPtr    = uintptr(ptrSize) |  | ||||||
| 	offsetScalar = uintptr(0) |  | ||||||
| 	offsetFlag   = uintptr(ptrSize * 2) |  | ||||||
|  |  | ||||||
| 	// flagKindWidth and flagKindShift indicate various bits that the | var ( | ||||||
| 	// reflect package uses internally to track kind information. | 	// flagRO indicates whether the value field of a reflect.Value | ||||||
| 	// | 	// is read-only. | ||||||
| 	// flagRO indicates whether or not the value field of a reflect.Value is | 	flagRO flag | ||||||
| 	// read-only. |  | ||||||
| 	// | 	// flagAddr indicates whether the address of the reflect.Value's | ||||||
| 	// flagIndir indicates whether the value field of a reflect.Value is | 	// value may be taken. | ||||||
| 	// the actual data or a pointer to the data. | 	flagAddr flag | ||||||
| 	// |  | ||||||
| 	// These values are valid before golang commit 90a7c3c86944 which |  | ||||||
| 	// changed their positions.  Code in the init function updates these |  | ||||||
| 	// flags as necessary. |  | ||||||
| 	flagKindWidth = uintptr(5) |  | ||||||
| 	flagKindShift = uintptr(flagKindWidth - 1) |  | ||||||
| 	flagRO        = uintptr(1 << 0) |  | ||||||
| 	flagIndir     = uintptr(1 << 1) |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func init() { | // flagKindMask holds the bits that make up the kind | ||||||
| 	// Older versions of reflect.Value stored small integers directly in the | // part of the flags field. In all the supported versions, | ||||||
| 	// ptr field (which is named val in the older versions).  Versions | // it is in the lower 5 bits. | ||||||
| 	// between commits ecccf07e7f9d and 82f48826c6c7 added a new field named | const flagKindMask = flag(0x1f) | ||||||
| 	// scalar for this purpose which unfortunately came before the flag |  | ||||||
| 	// field, so the offset of the flag field is different for those |  | ||||||
| 	// versions. |  | ||||||
| 	// |  | ||||||
| 	// This code constructs a new reflect.Value from a known small integer |  | ||||||
| 	// and checks if the size of the reflect.Value struct indicates it has |  | ||||||
| 	// the scalar field. When it does, the offsets are updated accordingly. |  | ||||||
| 	vv := reflect.ValueOf(0xf00) |  | ||||||
| 	if unsafe.Sizeof(vv) == (ptrSize * 4) { |  | ||||||
| 		offsetScalar = ptrSize * 2 |  | ||||||
| 		offsetFlag = ptrSize * 3 |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Commit 90a7c3c86944 changed the flag positions such that the low | // Different versions of Go have used different | ||||||
| 	// order bits are the kind.  This code extracts the kind from the flags | // bit layouts for the flags type. This table | ||||||
| 	// field and ensures it's the correct type.  When it's not, the flag | // records the known combinations. | ||||||
| 	// order has been changed to the newer format, so the flags are updated | var okFlags = []struct { | ||||||
| 	// accordingly. | 	ro, addr flag | ||||||
| 	upf := unsafe.Pointer(uintptr(unsafe.Pointer(&vv)) + offsetFlag) | }{{ | ||||||
| 	upfv := *(*uintptr)(upf) | 	// From Go 1.4 to 1.5 | ||||||
| 	flagKindMask := uintptr((1<<flagKindWidth - 1) << flagKindShift) | 	ro:   1 << 5, | ||||||
| 	if (upfv&flagKindMask)>>flagKindShift != uintptr(reflect.Int) { | 	addr: 1 << 7, | ||||||
| 		flagKindShift = 0 | }, { | ||||||
| 		flagRO = 1 << 5 | 	// Up to Go tip. | ||||||
| 		flagIndir = 1 << 6 | 	ro:   1<<5 | 1<<6, | ||||||
|  | 	addr: 1 << 8, | ||||||
|  | }} | ||||||
|  |  | ||||||
| 		// Commit adf9b30e5594 modified the flags to separate the | var flagValOffset = func() uintptr { | ||||||
| 		// flagRO flag into two bits which specifies whether or not the | 	field, ok := reflect.TypeOf(reflect.Value{}).FieldByName("flag") | ||||||
| 		// field is embedded.  This causes flagIndir to move over a bit | 	if !ok { | ||||||
| 		// and means that flagRO is the combination of either of the | 		panic("reflect.Value has no flag field") | ||||||
| 		// original flagRO bit and the new bit. |  | ||||||
| 		// |  | ||||||
| 		// This code detects the change by extracting what used to be |  | ||||||
| 		// the indirect bit to ensure it's set.  When it's not, the flag |  | ||||||
| 		// order has been changed to the newer format, so the flags are |  | ||||||
| 		// updated accordingly. |  | ||||||
| 		if upfv&flagIndir == 0 { |  | ||||||
| 			flagRO = 3 << 5 |  | ||||||
| 			flagIndir = 1 << 7 |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  | 	return field.Offset | ||||||
|  | }() | ||||||
|  |  | ||||||
|  | // flagField returns a pointer to the flag field of a reflect.Value. | ||||||
|  | func flagField(v *reflect.Value) *flag { | ||||||
|  | 	return (*flag)(unsafe.Pointer(uintptr(unsafe.Pointer(v)) + flagValOffset)) | ||||||
| } | } | ||||||
|  |  | ||||||
| // unsafeReflectValue converts the passed reflect.Value into a one that bypasses | // unsafeReflectValue converts the passed reflect.Value into a one that bypasses | ||||||
| @@ -119,34 +90,56 @@ func init() { | |||||||
| // This allows us to check for implementations of the Stringer and error | // This allows us to check for implementations of the Stringer and error | ||||||
| // interfaces to be used for pretty printing ordinarily unaddressable and | // interfaces to be used for pretty printing ordinarily unaddressable and | ||||||
| // inaccessible values such as unexported struct fields. | // inaccessible values such as unexported struct fields. | ||||||
| func unsafeReflectValue(v reflect.Value) (rv reflect.Value) { | func unsafeReflectValue(v reflect.Value) reflect.Value { | ||||||
| 	indirects := 1 | 	if !v.IsValid() || (v.CanInterface() && v.CanAddr()) { | ||||||
| 	vt := v.Type() | 		return v | ||||||
| 	upv := unsafe.Pointer(uintptr(unsafe.Pointer(&v)) + offsetPtr) | 	} | ||||||
| 	rvf := *(*uintptr)(unsafe.Pointer(uintptr(unsafe.Pointer(&v)) + offsetFlag)) | 	flagFieldPtr := flagField(&v) | ||||||
| 	if rvf&flagIndir != 0 { | 	*flagFieldPtr &^= flagRO | ||||||
| 		vt = reflect.PtrTo(v.Type()) | 	*flagFieldPtr |= flagAddr | ||||||
| 		indirects++ | 	return v | ||||||
| 	} else if offsetScalar != 0 { | } | ||||||
| 		// The value is in the scalar field when it's not one of the |  | ||||||
| 		// reference types. | // Sanity checks against future reflect package changes | ||||||
| 		switch vt.Kind() { | // to the type or semantics of the Value.flag field. | ||||||
| 		case reflect.Uintptr: | func init() { | ||||||
| 		case reflect.Chan: | 	field, ok := reflect.TypeOf(reflect.Value{}).FieldByName("flag") | ||||||
| 		case reflect.Func: | 	if !ok { | ||||||
| 		case reflect.Map: | 		panic("reflect.Value has no flag field") | ||||||
| 		case reflect.Ptr: | 	} | ||||||
| 		case reflect.UnsafePointer: | 	if field.Type.Kind() != reflect.TypeOf(flag(0)).Kind() { | ||||||
| 		default: | 		panic("reflect.Value flag field has changed kind") | ||||||
| 			upv = unsafe.Pointer(uintptr(unsafe.Pointer(&v)) + | 	} | ||||||
| 				offsetScalar) | 	type t0 int | ||||||
|  | 	var t struct { | ||||||
|  | 		A t0 | ||||||
|  | 		// t0 will have flagEmbedRO set. | ||||||
|  | 		t0 | ||||||
|  | 		// a will have flagStickyRO set | ||||||
|  | 		a t0 | ||||||
|  | 	} | ||||||
|  | 	vA := reflect.ValueOf(t).FieldByName("A") | ||||||
|  | 	va := reflect.ValueOf(t).FieldByName("a") | ||||||
|  | 	vt0 := reflect.ValueOf(t).FieldByName("t0") | ||||||
|  |  | ||||||
|  | 	// Infer flagRO from the difference between the flags | ||||||
|  | 	// for the (otherwise identical) fields in t. | ||||||
|  | 	flagPublic := *flagField(&vA) | ||||||
|  | 	flagWithRO := *flagField(&va) | *flagField(&vt0) | ||||||
|  | 	flagRO = flagPublic ^ flagWithRO | ||||||
|  |  | ||||||
|  | 	// Infer flagAddr from the difference between a value | ||||||
|  | 	// taken from a pointer and not. | ||||||
|  | 	vPtrA := reflect.ValueOf(&t).Elem().FieldByName("A") | ||||||
|  | 	flagNoPtr := *flagField(&vA) | ||||||
|  | 	flagPtr := *flagField(&vPtrA) | ||||||
|  | 	flagAddr = flagNoPtr ^ flagPtr | ||||||
|  |  | ||||||
|  | 	// Check that the inferred flags tally with one of the known versions. | ||||||
|  | 	for _, f := range okFlags { | ||||||
|  | 		if flagRO == f.ro && flagAddr == f.addr { | ||||||
|  | 			return | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	panic("reflect.Value read-only flag has changed semantics") | ||||||
| 	pv := reflect.NewAt(vt, upv) |  | ||||||
| 	rv = pv |  | ||||||
| 	for i := 0; i < indirects; i++ { |  | ||||||
| 		rv = rv.Elem() |  | ||||||
| 	} |  | ||||||
| 	return rv |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								vendor/github.com/davecgh/go-spew/spew/bypasssafe.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,7 +16,7 @@ | |||||||
| // when the code is running on Google App Engine, compiled by GopherJS, or | // when the code is running on Google App Engine, compiled by GopherJS, or | ||||||
| // "-tags safe" is added to the go build command line.  The "disableunsafe" | // "-tags safe" is added to the go build command line.  The "disableunsafe" | ||||||
| // tag is deprecated and thus should not be used. | // tag is deprecated and thus should not be used. | ||||||
| // +build js appengine safe disableunsafe | // +build js appengine safe disableunsafe !go1.4 | ||||||
|  |  | ||||||
| package spew | package spew | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								vendor/github.com/davecgh/go-spew/spew/common.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -180,7 +180,7 @@ func printComplex(w io.Writer, c complex128, floatPrecision int) { | |||||||
| 	w.Write(closeParenBytes) | 	w.Write(closeParenBytes) | ||||||
| } | } | ||||||
|  |  | ||||||
| // printHexPtr outputs a uintptr formatted as hexidecimal with a leading '0x' | // printHexPtr outputs a uintptr formatted as hexadecimal with a leading '0x' | ||||||
| // prefix to Writer w. | // prefix to Writer w. | ||||||
| func printHexPtr(w io.Writer, p uintptr) { | func printHexPtr(w io.Writer, p uintptr) { | ||||||
| 	// Null pointer. | 	// Null pointer. | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								vendor/github.com/davecgh/go-spew/spew/dump.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -35,16 +35,16 @@ var ( | |||||||
|  |  | ||||||
| 	// cCharRE is a regular expression that matches a cgo char. | 	// cCharRE is a regular expression that matches a cgo char. | ||||||
| 	// It is used to detect character arrays to hexdump them. | 	// It is used to detect character arrays to hexdump them. | ||||||
| 	cCharRE = regexp.MustCompile("^.*\\._Ctype_char$") | 	cCharRE = regexp.MustCompile(`^.*\._Ctype_char$`) | ||||||
|  |  | ||||||
| 	// cUnsignedCharRE is a regular expression that matches a cgo unsigned | 	// cUnsignedCharRE is a regular expression that matches a cgo unsigned | ||||||
| 	// char.  It is used to detect unsigned character arrays to hexdump | 	// char.  It is used to detect unsigned character arrays to hexdump | ||||||
| 	// them. | 	// them. | ||||||
| 	cUnsignedCharRE = regexp.MustCompile("^.*\\._Ctype_unsignedchar$") | 	cUnsignedCharRE = regexp.MustCompile(`^.*\._Ctype_unsignedchar$`) | ||||||
|  |  | ||||||
| 	// cUint8tCharRE is a regular expression that matches a cgo uint8_t. | 	// cUint8tCharRE is a regular expression that matches a cgo uint8_t. | ||||||
| 	// It is used to detect uint8_t arrays to hexdump them. | 	// It is used to detect uint8_t arrays to hexdump them. | ||||||
| 	cUint8tCharRE = regexp.MustCompile("^.*\\._Ctype_uint8_t$") | 	cUint8tCharRE = regexp.MustCompile(`^.*\._Ctype_uint8_t$`) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // dumpState contains information about the state of a dump operation. | // dumpState contains information about the state of a dump operation. | ||||||
| @@ -143,10 +143,10 @@ func (d *dumpState) dumpPtr(v reflect.Value) { | |||||||
| 	// Display dereferenced value. | 	// Display dereferenced value. | ||||||
| 	d.w.Write(openParenBytes) | 	d.w.Write(openParenBytes) | ||||||
| 	switch { | 	switch { | ||||||
| 	case nilFound == true: | 	case nilFound: | ||||||
| 		d.w.Write(nilAngleBytes) | 		d.w.Write(nilAngleBytes) | ||||||
|  |  | ||||||
| 	case cycleFound == true: | 	case cycleFound: | ||||||
| 		d.w.Write(circularBytes) | 		d.w.Write(circularBytes) | ||||||
|  |  | ||||||
| 	default: | 	default: | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								vendor/github.com/davecgh/go-spew/spew/format.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -182,10 +182,10 @@ func (f *formatState) formatPtr(v reflect.Value) { | |||||||
|  |  | ||||||
| 	// Display dereferenced value. | 	// Display dereferenced value. | ||||||
| 	switch { | 	switch { | ||||||
| 	case nilFound == true: | 	case nilFound: | ||||||
| 		f.fs.Write(nilAngleBytes) | 		f.fs.Write(nilAngleBytes) | ||||||
|  |  | ||||||
| 	case cycleFound == true: | 	case cycleFound: | ||||||
| 		f.fs.Write(circularShortBytes) | 		f.fs.Write(circularShortBytes) | ||||||
|  |  | ||||||
| 	default: | 	default: | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								vendor/github.com/gorilla/websocket/.gitignore
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -22,4 +22,4 @@ _testmain.go | |||||||
| *.exe | *.exe | ||||||
|  |  | ||||||
| .idea/ | .idea/ | ||||||
| *.iml | *.iml | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								vendor/github.com/gorilla/websocket/.travis.yml
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -3,11 +3,11 @@ sudo: false | |||||||
|  |  | ||||||
| matrix: | matrix: | ||||||
|   include: |   include: | ||||||
|     - go: 1.4 |     - go: 1.7.x | ||||||
|     - go: 1.5 |     - go: 1.8.x | ||||||
|     - go: 1.6 |     - go: 1.9.x | ||||||
|     - go: 1.7 |     - go: 1.10.x | ||||||
|     - go: 1.8 |     - go: 1.11.x | ||||||
|     - go: tip |     - go: tip | ||||||
|   allow_failures: |   allow_failures: | ||||||
|     - go: tip |     - go: tip | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								vendor/github.com/gorilla/websocket/AUTHORS
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -4,5 +4,6 @@ | |||||||
| # Please keep the list sorted. | # Please keep the list sorted. | ||||||
|  |  | ||||||
| Gary Burd <gary@beagledreams.com> | Gary Burd <gary@beagledreams.com> | ||||||
|  | Google LLC (https://opensource.google.com/) | ||||||
| Joachim Bauch <mail@joachim-bauch.de> | Joachim Bauch <mail@joachim-bauch.de> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								vendor/github.com/gorilla/websocket/README.md
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -51,7 +51,7 @@ subdirectory](https://github.com/gorilla/websocket/tree/master/examples/autobahn | |||||||
| <tr><td>Write message using io.WriteCloser</td><td><a href="http://godoc.org/github.com/gorilla/websocket#Conn.NextWriter">Yes</a></td><td>No, see note 3</td></tr> | <tr><td>Write message using io.WriteCloser</td><td><a href="http://godoc.org/github.com/gorilla/websocket#Conn.NextWriter">Yes</a></td><td>No, see note 3</td></tr> | ||||||
| </table> | </table> | ||||||
|  |  | ||||||
| Notes:  | Notes: | ||||||
|  |  | ||||||
| 1. Large messages are fragmented in [Chrome's new WebSocket implementation](http://www.ietf.org/mail-archive/web/hybi/current/msg10503.html). | 1. Large messages are fragmented in [Chrome's new WebSocket implementation](http://www.ietf.org/mail-archive/web/hybi/current/msg10503.html). | ||||||
| 2. The application can get the type of a received data message by implementing | 2. The application can get the type of a received data message by implementing | ||||||
|   | |||||||
							
								
								
									
										253
									
								
								vendor/github.com/gorilla/websocket/client.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -5,15 +5,15 @@ | |||||||
| package websocket | package websocket | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bufio" |  | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"context" | ||||||
| 	"crypto/tls" | 	"crypto/tls" | ||||||
| 	"encoding/base64" |  | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"io" | 	"io" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net" | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"net/http/httptrace" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -53,6 +53,10 @@ type Dialer struct { | |||||||
| 	// NetDial is nil, net.Dial is used. | 	// NetDial is nil, net.Dial is used. | ||||||
| 	NetDial func(network, addr string) (net.Conn, error) | 	NetDial func(network, addr string) (net.Conn, error) | ||||||
|  |  | ||||||
|  | 	// NetDialContext specifies the dial function for creating TCP connections. If | ||||||
|  | 	// NetDialContext is nil, net.DialContext is used. | ||||||
|  | 	NetDialContext func(ctx context.Context, network, addr string) (net.Conn, error) | ||||||
|  |  | ||||||
| 	// Proxy specifies a function to return a proxy for a given | 	// Proxy specifies a function to return a proxy for a given | ||||||
| 	// Request. If the function returns a non-nil error, the | 	// Request. If the function returns a non-nil error, the | ||||||
| 	// request is aborted with the provided error. | 	// request is aborted with the provided error. | ||||||
| @@ -71,6 +75,17 @@ type Dialer struct { | |||||||
| 	// do not limit the size of the messages that can be sent or received. | 	// do not limit the size of the messages that can be sent or received. | ||||||
| 	ReadBufferSize, WriteBufferSize int | 	ReadBufferSize, WriteBufferSize int | ||||||
|  |  | ||||||
|  | 	// WriteBufferPool is a pool of buffers for write operations. If the value | ||||||
|  | 	// is not set, then write buffers are allocated to the connection for the | ||||||
|  | 	// lifetime of the connection. | ||||||
|  | 	// | ||||||
|  | 	// A pool is most useful when the application has a modest volume of writes | ||||||
|  | 	// across a large number of connections. | ||||||
|  | 	// | ||||||
|  | 	// Applications should use a single pool for each unique value of | ||||||
|  | 	// WriteBufferSize. | ||||||
|  | 	WriteBufferPool BufferPool | ||||||
|  |  | ||||||
| 	// Subprotocols specifies the client's requested subprotocols. | 	// Subprotocols specifies the client's requested subprotocols. | ||||||
| 	Subprotocols []string | 	Subprotocols []string | ||||||
|  |  | ||||||
| @@ -86,52 +101,13 @@ type Dialer struct { | |||||||
| 	Jar http.CookieJar | 	Jar http.CookieJar | ||||||
| } | } | ||||||
|  |  | ||||||
| var errMalformedURL = errors.New("malformed ws or wss URL") | // Dial creates a new client connection by calling DialContext with a background context. | ||||||
|  | func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Response, error) { | ||||||
| // parseURL parses the URL. | 	return d.DialContext(context.Background(), urlStr, requestHeader) | ||||||
| // |  | ||||||
| // This function is a replacement for the standard library url.Parse function. |  | ||||||
| // In Go 1.4 and earlier, url.Parse loses information from the path. |  | ||||||
| func parseURL(s string) (*url.URL, error) { |  | ||||||
| 	// From the RFC: |  | ||||||
| 	// |  | ||||||
| 	// ws-URI = "ws:" "//" host [ ":" port ] path [ "?" query ] |  | ||||||
| 	// wss-URI = "wss:" "//" host [ ":" port ] path [ "?" query ] |  | ||||||
| 	var u url.URL |  | ||||||
| 	switch { |  | ||||||
| 	case strings.HasPrefix(s, "ws://"): |  | ||||||
| 		u.Scheme = "ws" |  | ||||||
| 		s = s[len("ws://"):] |  | ||||||
| 	case strings.HasPrefix(s, "wss://"): |  | ||||||
| 		u.Scheme = "wss" |  | ||||||
| 		s = s[len("wss://"):] |  | ||||||
| 	default: |  | ||||||
| 		return nil, errMalformedURL |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if i := strings.Index(s, "?"); i >= 0 { |  | ||||||
| 		u.RawQuery = s[i+1:] |  | ||||||
| 		s = s[:i] |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if i := strings.Index(s, "/"); i >= 0 { |  | ||||||
| 		u.Opaque = s[i:] |  | ||||||
| 		s = s[:i] |  | ||||||
| 	} else { |  | ||||||
| 		u.Opaque = "/" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	u.Host = s |  | ||||||
|  |  | ||||||
| 	if strings.Contains(u.Host, "@") { |  | ||||||
| 		// Don't bother parsing user information because user information is |  | ||||||
| 		// not allowed in websocket URIs. |  | ||||||
| 		return nil, errMalformedURL |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return &u, nil |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | var errMalformedURL = errors.New("malformed ws or wss URL") | ||||||
|  |  | ||||||
| func hostPortNoPort(u *url.URL) (hostPort, hostNoPort string) { | func hostPortNoPort(u *url.URL) (hostPort, hostNoPort string) { | ||||||
| 	hostPort = u.Host | 	hostPort = u.Host | ||||||
| 	hostNoPort = u.Host | 	hostNoPort = u.Host | ||||||
| @@ -150,26 +126,29 @@ func hostPortNoPort(u *url.URL) (hostPort, hostNoPort string) { | |||||||
| 	return hostPort, hostNoPort | 	return hostPort, hostNoPort | ||||||
| } | } | ||||||
|  |  | ||||||
| // DefaultDialer is a dialer with all fields set to the default zero values. | // DefaultDialer is a dialer with all fields set to the default values. | ||||||
| var DefaultDialer = &Dialer{ | var DefaultDialer = &Dialer{ | ||||||
| 	Proxy: http.ProxyFromEnvironment, | 	Proxy:            http.ProxyFromEnvironment, | ||||||
|  | 	HandshakeTimeout: 45 * time.Second, | ||||||
| } | } | ||||||
|  |  | ||||||
| // Dial creates a new client connection. Use requestHeader to specify the | // nilDialer is dialer to use when receiver is nil. | ||||||
|  | var nilDialer = *DefaultDialer | ||||||
|  |  | ||||||
|  | // DialContext creates a new client connection. Use requestHeader to specify the | ||||||
| // origin (Origin), subprotocols (Sec-WebSocket-Protocol) and cookies (Cookie). | // origin (Origin), subprotocols (Sec-WebSocket-Protocol) and cookies (Cookie). | ||||||
| // Use the response.Header to get the selected subprotocol | // Use the response.Header to get the selected subprotocol | ||||||
| // (Sec-WebSocket-Protocol) and cookies (Set-Cookie). | // (Sec-WebSocket-Protocol) and cookies (Set-Cookie). | ||||||
| // | // | ||||||
|  | // The context will be used in the request and in the Dialer | ||||||
|  | // | ||||||
| // If the WebSocket handshake fails, ErrBadHandshake is returned along with a | // If the WebSocket handshake fails, ErrBadHandshake is returned along with a | ||||||
| // non-nil *http.Response so that callers can handle redirects, authentication, | // non-nil *http.Response so that callers can handle redirects, authentication, | ||||||
| // etcetera. The response body may not contain the entire response and does not | // etcetera. The response body may not contain the entire response and does not | ||||||
| // need to be closed by the application. | // need to be closed by the application. | ||||||
| func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Response, error) { | func (d *Dialer) DialContext(ctx context.Context, urlStr string, requestHeader http.Header) (*Conn, *http.Response, error) { | ||||||
|  |  | ||||||
| 	if d == nil { | 	if d == nil { | ||||||
| 		d = &Dialer{ | 		d = &nilDialer | ||||||
| 			Proxy: http.ProxyFromEnvironment, |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	challengeKey, err := generateChallengeKey() | 	challengeKey, err := generateChallengeKey() | ||||||
| @@ -177,7 +156,7 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	u, err := parseURL(urlStr) | 	u, err := url.Parse(urlStr) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| 	} | 	} | ||||||
| @@ -205,6 +184,7 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 		Header:     make(http.Header), | 		Header:     make(http.Header), | ||||||
| 		Host:       u.Host, | 		Host:       u.Host, | ||||||
| 	} | 	} | ||||||
|  | 	req = req.WithContext(ctx) | ||||||
|  |  | ||||||
| 	// Set the cookies present in the cookie jar of the dialer | 	// Set the cookies present in the cookie jar of the dialer | ||||||
| 	if d.Jar != nil { | 	if d.Jar != nil { | ||||||
| @@ -237,45 +217,83 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 			k == "Sec-Websocket-Extensions" || | 			k == "Sec-Websocket-Extensions" || | ||||||
| 			(k == "Sec-Websocket-Protocol" && len(d.Subprotocols) > 0): | 			(k == "Sec-Websocket-Protocol" && len(d.Subprotocols) > 0): | ||||||
| 			return nil, nil, errors.New("websocket: duplicate header not allowed: " + k) | 			return nil, nil, errors.New("websocket: duplicate header not allowed: " + k) | ||||||
|  | 		case k == "Sec-Websocket-Protocol": | ||||||
|  | 			req.Header["Sec-WebSocket-Protocol"] = vs | ||||||
| 		default: | 		default: | ||||||
| 			req.Header[k] = vs | 			req.Header[k] = vs | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if d.EnableCompression { | 	if d.EnableCompression { | ||||||
| 		req.Header.Set("Sec-Websocket-Extensions", "permessage-deflate; server_no_context_takeover; client_no_context_takeover") | 		req.Header["Sec-WebSocket-Extensions"] = []string{"permessage-deflate; server_no_context_takeover; client_no_context_takeover"} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if d.HandshakeTimeout != 0 { | ||||||
|  | 		var cancel func() | ||||||
|  | 		ctx, cancel = context.WithTimeout(ctx, d.HandshakeTimeout) | ||||||
|  | 		defer cancel() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Get network dial function. | ||||||
|  | 	var netDial func(network, add string) (net.Conn, error) | ||||||
|  |  | ||||||
|  | 	if d.NetDialContext != nil { | ||||||
|  | 		netDial = func(network, addr string) (net.Conn, error) { | ||||||
|  | 			return d.NetDialContext(ctx, network, addr) | ||||||
|  | 		} | ||||||
|  | 	} else if d.NetDial != nil { | ||||||
|  | 		netDial = d.NetDial | ||||||
|  | 	} else { | ||||||
|  | 		netDialer := &net.Dialer{} | ||||||
|  | 		netDial = func(network, addr string) (net.Conn, error) { | ||||||
|  | 			return netDialer.DialContext(ctx, network, addr) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If needed, wrap the dial function to set the connection deadline. | ||||||
|  | 	if deadline, ok := ctx.Deadline(); ok { | ||||||
|  | 		forwardDial := netDial | ||||||
|  | 		netDial = func(network, addr string) (net.Conn, error) { | ||||||
|  | 			c, err := forwardDial(network, addr) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, err | ||||||
|  | 			} | ||||||
|  | 			err = c.SetDeadline(deadline) | ||||||
|  | 			if err != nil { | ||||||
|  | 				c.Close() | ||||||
|  | 				return nil, err | ||||||
|  | 			} | ||||||
|  | 			return c, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If needed, wrap the dial function to connect through a proxy. | ||||||
|  | 	if d.Proxy != nil { | ||||||
|  | 		proxyURL, err := d.Proxy(req) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, nil, err | ||||||
|  | 		} | ||||||
|  | 		if proxyURL != nil { | ||||||
|  | 			dialer, err := proxy_FromURL(proxyURL, netDialerFunc(netDial)) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, nil, err | ||||||
|  | 			} | ||||||
|  | 			netDial = dialer.Dial | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	hostPort, hostNoPort := hostPortNoPort(u) | 	hostPort, hostNoPort := hostPortNoPort(u) | ||||||
|  | 	trace := httptrace.ContextClientTrace(ctx) | ||||||
| 	var proxyURL *url.URL | 	if trace != nil && trace.GetConn != nil { | ||||||
| 	// Check wether the proxy method has been configured | 		trace.GetConn(hostPort) | ||||||
| 	if d.Proxy != nil { |  | ||||||
| 		proxyURL, err = d.Proxy(req) |  | ||||||
| 	} |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, nil, err |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var targetHostPort string | 	netConn, err := netDial("tcp", hostPort) | ||||||
| 	if proxyURL != nil { | 	if trace != nil && trace.GotConn != nil { | ||||||
| 		targetHostPort, _ = hostPortNoPort(proxyURL) | 		trace.GotConn(httptrace.GotConnInfo{ | ||||||
| 	} else { | 			Conn: netConn, | ||||||
| 		targetHostPort = hostPort | 		}) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var deadline time.Time |  | ||||||
| 	if d.HandshakeTimeout != 0 { |  | ||||||
| 		deadline = time.Now().Add(d.HandshakeTimeout) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	netDial := d.NetDial |  | ||||||
| 	if netDial == nil { |  | ||||||
| 		netDialer := &net.Dialer{Deadline: deadline} |  | ||||||
| 		netDial = netDialer.Dial |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	netConn, err := netDial("tcp", targetHostPort) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| 	} | 	} | ||||||
| @@ -286,42 +304,6 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  |  | ||||||
| 	if err := netConn.SetDeadline(deadline); err != nil { |  | ||||||
| 		return nil, nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if proxyURL != nil { |  | ||||||
| 		connectHeader := make(http.Header) |  | ||||||
| 		if user := proxyURL.User; user != nil { |  | ||||||
| 			proxyUser := user.Username() |  | ||||||
| 			if proxyPassword, passwordSet := user.Password(); passwordSet { |  | ||||||
| 				credential := base64.StdEncoding.EncodeToString([]byte(proxyUser + ":" + proxyPassword)) |  | ||||||
| 				connectHeader.Set("Proxy-Authorization", "Basic "+credential) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		connectReq := &http.Request{ |  | ||||||
| 			Method: "CONNECT", |  | ||||||
| 			URL:    &url.URL{Opaque: hostPort}, |  | ||||||
| 			Host:   hostPort, |  | ||||||
| 			Header: connectHeader, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		connectReq.Write(netConn) |  | ||||||
|  |  | ||||||
| 		// Read response. |  | ||||||
| 		// Okay to use and discard buffered reader here, because |  | ||||||
| 		// TLS server will not speak until spoken to. |  | ||||||
| 		br := bufio.NewReader(netConn) |  | ||||||
| 		resp, err := http.ReadResponse(br, connectReq) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, nil, err |  | ||||||
| 		} |  | ||||||
| 		if resp.StatusCode != 200 { |  | ||||||
| 			f := strings.SplitN(resp.Status, " ", 2) |  | ||||||
| 			return nil, nil, errors.New(f[1]) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if u.Scheme == "https" { | 	if u.Scheme == "https" { | ||||||
| 		cfg := cloneTLSConfig(d.TLSClientConfig) | 		cfg := cloneTLSConfig(d.TLSClientConfig) | ||||||
| 		if cfg.ServerName == "" { | 		if cfg.ServerName == "" { | ||||||
| @@ -329,22 +311,31 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 		} | 		} | ||||||
| 		tlsConn := tls.Client(netConn, cfg) | 		tlsConn := tls.Client(netConn, cfg) | ||||||
| 		netConn = tlsConn | 		netConn = tlsConn | ||||||
| 		if err := tlsConn.Handshake(); err != nil { |  | ||||||
| 			return nil, nil, err | 		var err error | ||||||
|  | 		if trace != nil { | ||||||
|  | 			err = doHandshakeWithTrace(trace, tlsConn, cfg) | ||||||
|  | 		} else { | ||||||
|  | 			err = doHandshake(tlsConn, cfg) | ||||||
| 		} | 		} | ||||||
| 		if !cfg.InsecureSkipVerify { |  | ||||||
| 			if err := tlsConn.VerifyHostname(cfg.ServerName); err != nil { | 		if err != nil { | ||||||
| 				return nil, nil, err | 			return nil, nil, err | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	conn := newConn(netConn, false, d.ReadBufferSize, d.WriteBufferSize) | 	conn := newConn(netConn, false, d.ReadBufferSize, d.WriteBufferSize, d.WriteBufferPool, nil, nil) | ||||||
|  |  | ||||||
| 	if err := req.Write(netConn); err != nil { | 	if err := req.Write(netConn); err != nil { | ||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if trace != nil && trace.GotFirstResponseByte != nil { | ||||||
|  | 		if peek, err := conn.br.Peek(1); err == nil && len(peek) == 1 { | ||||||
|  | 			trace.GotFirstResponseByte() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	resp, err := http.ReadResponse(conn.br, req) | 	resp, err := http.ReadResponse(conn.br, req) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, nil, err | 		return nil, nil, err | ||||||
| @@ -390,3 +381,15 @@ func (d *Dialer) Dial(urlStr string, requestHeader http.Header) (*Conn, *http.Re | |||||||
| 	netConn = nil // to avoid close in defer. | 	netConn = nil // to avoid close in defer. | ||||||
| 	return conn, resp, nil | 	return conn, resp, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func doHandshake(tlsConn *tls.Conn, cfg *tls.Config) error { | ||||||
|  | 	if err := tlsConn.Handshake(); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if !cfg.InsecureSkipVerify { | ||||||
|  | 		if err := tlsConn.VerifyHostname(cfg.ServerName); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										174
									
								
								vendor/github.com/gorilla/websocket/conn.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -76,7 +76,7 @@ const ( | |||||||
| 	// is UTF-8 encoded text. | 	// is UTF-8 encoded text. | ||||||
| 	PingMessage = 9 | 	PingMessage = 9 | ||||||
|  |  | ||||||
| 	// PongMessage denotes a ping control message. The optional message payload | 	// PongMessage denotes a pong control message. The optional message payload | ||||||
| 	// is UTF-8 encoded text. | 	// is UTF-8 encoded text. | ||||||
| 	PongMessage = 10 | 	PongMessage = 10 | ||||||
| ) | ) | ||||||
| @@ -100,9 +100,8 @@ func (e *netError) Error() string   { return e.msg } | |||||||
| func (e *netError) Temporary() bool { return e.temporary } | func (e *netError) Temporary() bool { return e.temporary } | ||||||
| func (e *netError) Timeout() bool   { return e.timeout } | func (e *netError) Timeout() bool   { return e.timeout } | ||||||
|  |  | ||||||
| // CloseError represents close frame. | // CloseError represents a close message. | ||||||
| type CloseError struct { | type CloseError struct { | ||||||
|  |  | ||||||
| 	// Code is defined in RFC 6455, section 11.7. | 	// Code is defined in RFC 6455, section 11.7. | ||||||
| 	Code int | 	Code int | ||||||
|  |  | ||||||
| @@ -224,6 +223,20 @@ func isValidReceivedCloseCode(code int) bool { | |||||||
| 	return validReceivedCloseCodes[code] || (code >= 3000 && code <= 4999) | 	return validReceivedCloseCodes[code] || (code >= 3000 && code <= 4999) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // BufferPool represents a pool of buffers. The *sync.Pool type satisfies this | ||||||
|  | // interface.  The type of the value stored in a pool is not specified. | ||||||
|  | type BufferPool interface { | ||||||
|  | 	// Get gets a value from the pool or returns nil if the pool is empty. | ||||||
|  | 	Get() interface{} | ||||||
|  | 	// Put adds a value to the pool. | ||||||
|  | 	Put(interface{}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // writePoolData is the type added to the write buffer pool. This wrapper is | ||||||
|  | // used to prevent applications from peeking at and depending on the values | ||||||
|  | // added to the pool. | ||||||
|  | type writePoolData struct{ buf []byte } | ||||||
|  |  | ||||||
| // The Conn type represents a WebSocket connection. | // The Conn type represents a WebSocket connection. | ||||||
| type Conn struct { | type Conn struct { | ||||||
| 	conn        net.Conn | 	conn        net.Conn | ||||||
| @@ -233,6 +246,8 @@ type Conn struct { | |||||||
| 	// Write fields | 	// Write fields | ||||||
| 	mu            chan bool // used as mutex to protect write to conn | 	mu            chan bool // used as mutex to protect write to conn | ||||||
| 	writeBuf      []byte    // frame is constructed in this buffer. | 	writeBuf      []byte    // frame is constructed in this buffer. | ||||||
|  | 	writePool     BufferPool | ||||||
|  | 	writeBufSize  int | ||||||
| 	writeDeadline time.Time | 	writeDeadline time.Time | ||||||
| 	writer        io.WriteCloser // the current writer returned to the application | 	writer        io.WriteCloser // the current writer returned to the application | ||||||
| 	isWriting     bool           // for best-effort concurrent write detection | 	isWriting     bool           // for best-effort concurrent write detection | ||||||
| @@ -264,64 +279,29 @@ type Conn struct { | |||||||
| 	newDecompressionReader func(io.Reader) io.ReadCloser | 	newDecompressionReader func(io.Reader) io.ReadCloser | ||||||
| } | } | ||||||
|  |  | ||||||
| func newConn(conn net.Conn, isServer bool, readBufferSize, writeBufferSize int) *Conn { | func newConn(conn net.Conn, isServer bool, readBufferSize, writeBufferSize int, writeBufferPool BufferPool, br *bufio.Reader, writeBuf []byte) *Conn { | ||||||
| 	return newConnBRW(conn, isServer, readBufferSize, writeBufferSize, nil) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type writeHook struct { |  | ||||||
| 	p []byte |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (wh *writeHook) Write(p []byte) (int, error) { |  | ||||||
| 	wh.p = p |  | ||||||
| 	return len(p), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func newConnBRW(conn net.Conn, isServer bool, readBufferSize, writeBufferSize int, brw *bufio.ReadWriter) *Conn { |  | ||||||
| 	mu := make(chan bool, 1) |  | ||||||
| 	mu <- true |  | ||||||
|  |  | ||||||
| 	var br *bufio.Reader |  | ||||||
| 	if readBufferSize == 0 && brw != nil && brw.Reader != nil { |  | ||||||
| 		// Reuse the supplied bufio.Reader if the buffer has a useful size. |  | ||||||
| 		// This code assumes that peek on a reader returns |  | ||||||
| 		// bufio.Reader.buf[:0]. |  | ||||||
| 		brw.Reader.Reset(conn) |  | ||||||
| 		if p, err := brw.Reader.Peek(0); err == nil && cap(p) >= 256 { |  | ||||||
| 			br = brw.Reader |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if br == nil { | 	if br == nil { | ||||||
| 		if readBufferSize == 0 { | 		if readBufferSize == 0 { | ||||||
| 			readBufferSize = defaultReadBufferSize | 			readBufferSize = defaultReadBufferSize | ||||||
| 		} | 		} else if readBufferSize < maxControlFramePayloadSize { | ||||||
| 		if readBufferSize < maxControlFramePayloadSize { | 			// must be large enough for control frame | ||||||
| 			readBufferSize = maxControlFramePayloadSize | 			readBufferSize = maxControlFramePayloadSize | ||||||
| 		} | 		} | ||||||
| 		br = bufio.NewReaderSize(conn, readBufferSize) | 		br = bufio.NewReaderSize(conn, readBufferSize) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var writeBuf []byte | 	if writeBufferSize <= 0 { | ||||||
| 	if writeBufferSize == 0 && brw != nil && brw.Writer != nil { | 		writeBufferSize = defaultWriteBufferSize | ||||||
| 		// Use the bufio.Writer's buffer if the buffer has a useful size. This | 	} | ||||||
| 		// code assumes that bufio.Writer.buf[:1] is passed to the | 	writeBufferSize += maxFrameHeaderSize | ||||||
| 		// bufio.Writer's underlying writer. |  | ||||||
| 		var wh writeHook | 	if writeBuf == nil && writeBufferPool == nil { | ||||||
| 		brw.Writer.Reset(&wh) | 		writeBuf = make([]byte, writeBufferSize) | ||||||
| 		brw.Writer.WriteByte(0) |  | ||||||
| 		brw.Flush() |  | ||||||
| 		if cap(wh.p) >= maxFrameHeaderSize+256 { |  | ||||||
| 			writeBuf = wh.p[:cap(wh.p)] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if writeBuf == nil { |  | ||||||
| 		if writeBufferSize == 0 { |  | ||||||
| 			writeBufferSize = defaultWriteBufferSize |  | ||||||
| 		} |  | ||||||
| 		writeBuf = make([]byte, writeBufferSize+maxFrameHeaderSize) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	mu := make(chan bool, 1) | ||||||
|  | 	mu <- true | ||||||
| 	c := &Conn{ | 	c := &Conn{ | ||||||
| 		isServer:               isServer, | 		isServer:               isServer, | ||||||
| 		br:                     br, | 		br:                     br, | ||||||
| @@ -329,6 +309,8 @@ func newConnBRW(conn net.Conn, isServer bool, readBufferSize, writeBufferSize in | |||||||
| 		mu:                     mu, | 		mu:                     mu, | ||||||
| 		readFinal:              true, | 		readFinal:              true, | ||||||
| 		writeBuf:               writeBuf, | 		writeBuf:               writeBuf, | ||||||
|  | 		writePool:              writeBufferPool, | ||||||
|  | 		writeBufSize:           writeBufferSize, | ||||||
| 		enableWriteCompression: true, | 		enableWriteCompression: true, | ||||||
| 		compressionLevel:       defaultCompressionLevel, | 		compressionLevel:       defaultCompressionLevel, | ||||||
| 	} | 	} | ||||||
| @@ -343,7 +325,8 @@ func (c *Conn) Subprotocol() string { | |||||||
| 	return c.subprotocol | 	return c.subprotocol | ||||||
| } | } | ||||||
|  |  | ||||||
| // Close closes the underlying network connection without sending or waiting for a close frame. | // Close closes the underlying network connection without sending or waiting | ||||||
|  | // for a close message. | ||||||
| func (c *Conn) Close() error { | func (c *Conn) Close() error { | ||||||
| 	return c.conn.Close() | 	return c.conn.Close() | ||||||
| } | } | ||||||
| @@ -370,7 +353,16 @@ func (c *Conn) writeFatal(err error) error { | |||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *Conn) write(frameType int, deadline time.Time, bufs ...[]byte) error { | func (c *Conn) read(n int) ([]byte, error) { | ||||||
|  | 	p, err := c.br.Peek(n) | ||||||
|  | 	if err == io.EOF { | ||||||
|  | 		err = errUnexpectedEOF | ||||||
|  | 	} | ||||||
|  | 	c.br.Discard(len(p)) | ||||||
|  | 	return p, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *Conn) write(frameType int, deadline time.Time, buf0, buf1 []byte) error { | ||||||
| 	<-c.mu | 	<-c.mu | ||||||
| 	defer func() { c.mu <- true }() | 	defer func() { c.mu <- true }() | ||||||
|  |  | ||||||
| @@ -382,15 +374,14 @@ func (c *Conn) write(frameType int, deadline time.Time, bufs ...[]byte) error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	c.conn.SetWriteDeadline(deadline) | 	c.conn.SetWriteDeadline(deadline) | ||||||
| 	for _, buf := range bufs { | 	if len(buf1) == 0 { | ||||||
| 		if len(buf) > 0 { | 		_, err = c.conn.Write(buf0) | ||||||
| 			_, err := c.conn.Write(buf) | 	} else { | ||||||
| 			if err != nil { | 		err = c.writeBufs(buf0, buf1) | ||||||
| 				return c.writeFatal(err) | 	} | ||||||
| 			} | 	if err != nil { | ||||||
| 		} | 		return c.writeFatal(err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if frameType == CloseMessage { | 	if frameType == CloseMessage { | ||||||
| 		c.writeFatal(ErrCloseSent) | 		c.writeFatal(ErrCloseSent) | ||||||
| 	} | 	} | ||||||
| @@ -476,7 +467,19 @@ func (c *Conn) prepWrite(messageType int) error { | |||||||
| 	c.writeErrMu.Lock() | 	c.writeErrMu.Lock() | ||||||
| 	err := c.writeErr | 	err := c.writeErr | ||||||
| 	c.writeErrMu.Unlock() | 	c.writeErrMu.Unlock() | ||||||
| 	return err | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if c.writeBuf == nil { | ||||||
|  | 		wpd, ok := c.writePool.Get().(writePoolData) | ||||||
|  | 		if ok { | ||||||
|  | 			c.writeBuf = wpd.buf | ||||||
|  | 		} else { | ||||||
|  | 			c.writeBuf = make([]byte, c.writeBufSize) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // NextWriter returns a writer for the next message to send. The writer's Close | // NextWriter returns a writer for the next message to send. The writer's Close | ||||||
| @@ -484,6 +487,9 @@ func (c *Conn) prepWrite(messageType int) error { | |||||||
| // | // | ||||||
| // There can be at most one open writer on a connection. NextWriter closes the | // There can be at most one open writer on a connection. NextWriter closes the | ||||||
| // previous writer if the application has not already done so. | // previous writer if the application has not already done so. | ||||||
|  | // | ||||||
|  | // All message types (TextMessage, BinaryMessage, CloseMessage, PingMessage and | ||||||
|  | // PongMessage) are supported. | ||||||
| func (c *Conn) NextWriter(messageType int) (io.WriteCloser, error) { | func (c *Conn) NextWriter(messageType int) (io.WriteCloser, error) { | ||||||
| 	if err := c.prepWrite(messageType); err != nil { | 	if err := c.prepWrite(messageType); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @@ -599,6 +605,10 @@ func (w *messageWriter) flushFrame(final bool, extra []byte) error { | |||||||
|  |  | ||||||
| 	if final { | 	if final { | ||||||
| 		c.writer = nil | 		c.writer = nil | ||||||
|  | 		if c.writePool != nil { | ||||||
|  | 			c.writePool.Put(writePoolData{buf: c.writeBuf}) | ||||||
|  | 			c.writeBuf = nil | ||||||
|  | 		} | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -764,7 +774,6 @@ func (c *Conn) SetWriteDeadline(t time.Time) error { | |||||||
| // Read methods | // Read methods | ||||||
|  |  | ||||||
| func (c *Conn) advanceFrame() (int, error) { | func (c *Conn) advanceFrame() (int, error) { | ||||||
|  |  | ||||||
| 	// 1. Skip remainder of previous frame. | 	// 1. Skip remainder of previous frame. | ||||||
|  |  | ||||||
| 	if c.readRemaining > 0 { | 	if c.readRemaining > 0 { | ||||||
| @@ -1033,7 +1042,7 @@ func (c *Conn) SetReadDeadline(t time.Time) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // SetReadLimit sets the maximum size for a message read from the peer. If a | // SetReadLimit sets the maximum size for a message read from the peer. If a | ||||||
| // message exceeds the limit, the connection sends a close frame to the peer | // message exceeds the limit, the connection sends a close message to the peer | ||||||
| // and returns ErrReadLimit to the application. | // and returns ErrReadLimit to the application. | ||||||
| func (c *Conn) SetReadLimit(limit int64) { | func (c *Conn) SetReadLimit(limit int64) { | ||||||
| 	c.readLimit = limit | 	c.readLimit = limit | ||||||
| @@ -1046,24 +1055,22 @@ func (c *Conn) CloseHandler() func(code int, text string) error { | |||||||
|  |  | ||||||
| // SetCloseHandler sets the handler for close messages received from the peer. | // SetCloseHandler sets the handler for close messages received from the peer. | ||||||
| // The code argument to h is the received close code or CloseNoStatusReceived | // The code argument to h is the received close code or CloseNoStatusReceived | ||||||
| // if the close message is empty. The default close handler sends a close frame | // if the close message is empty. The default close handler sends a close | ||||||
| // back to the peer. | // message back to the peer. | ||||||
| // | // | ||||||
| // The application must read the connection to process close messages as | // The handler function is called from the NextReader, ReadMessage and message | ||||||
| // described in the section on Control Frames above. | // reader Read methods. The application must read the connection to process | ||||||
|  | // close messages as described in the section on Control Messages above. | ||||||
| // | // | ||||||
| // The connection read methods return a CloseError when a close frame is | // The connection read methods return a CloseError when a close message is | ||||||
| // received. Most applications should handle close messages as part of their | // received. Most applications should handle close messages as part of their | ||||||
| // normal error handling. Applications should only set a close handler when the | // normal error handling. Applications should only set a close handler when the | ||||||
| // application must perform some action before sending a close frame back to | // application must perform some action before sending a close message back to | ||||||
| // the peer. | // the peer. | ||||||
| func (c *Conn) SetCloseHandler(h func(code int, text string) error) { | func (c *Conn) SetCloseHandler(h func(code int, text string) error) { | ||||||
| 	if h == nil { | 	if h == nil { | ||||||
| 		h = func(code int, text string) error { | 		h = func(code int, text string) error { | ||||||
| 			message := []byte{} | 			message := FormatCloseMessage(code, "") | ||||||
| 			if code != CloseNoStatusReceived { |  | ||||||
| 				message = FormatCloseMessage(code, "") |  | ||||||
| 			} |  | ||||||
| 			c.WriteControl(CloseMessage, message, time.Now().Add(writeWait)) | 			c.WriteControl(CloseMessage, message, time.Now().Add(writeWait)) | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| @@ -1077,11 +1084,12 @@ func (c *Conn) PingHandler() func(appData string) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // SetPingHandler sets the handler for ping messages received from the peer. | // SetPingHandler sets the handler for ping messages received from the peer. | ||||||
| // The appData argument to h is the PING frame application data. The default | // The appData argument to h is the PING message application data. The default | ||||||
| // ping handler sends a pong to the peer. | // ping handler sends a pong to the peer. | ||||||
| // | // | ||||||
| // The application must read the connection to process ping messages as | // The handler function is called from the NextReader, ReadMessage and message | ||||||
| // described in the section on Control Frames above. | // reader Read methods. The application must read the connection to process | ||||||
|  | // ping messages as described in the section on Control Messages above. | ||||||
| func (c *Conn) SetPingHandler(h func(appData string) error) { | func (c *Conn) SetPingHandler(h func(appData string) error) { | ||||||
| 	if h == nil { | 	if h == nil { | ||||||
| 		h = func(message string) error { | 		h = func(message string) error { | ||||||
| @@ -1103,11 +1111,12 @@ func (c *Conn) PongHandler() func(appData string) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // SetPongHandler sets the handler for pong messages received from the peer. | // SetPongHandler sets the handler for pong messages received from the peer. | ||||||
| // The appData argument to h is the PONG frame application data. The default | // The appData argument to h is the PONG message application data. The default | ||||||
| // pong handler does nothing. | // pong handler does nothing. | ||||||
| // | // | ||||||
| // The application must read the connection to process ping messages as | // The handler function is called from the NextReader, ReadMessage and message | ||||||
| // described in the section on Control Frames above. | // reader Read methods. The application must read the connection to process | ||||||
|  | // pong messages as described in the section on Control Messages above. | ||||||
| func (c *Conn) SetPongHandler(h func(appData string) error) { | func (c *Conn) SetPongHandler(h func(appData string) error) { | ||||||
| 	if h == nil { | 	if h == nil { | ||||||
| 		h = func(string) error { return nil } | 		h = func(string) error { return nil } | ||||||
| @@ -1141,7 +1150,14 @@ func (c *Conn) SetCompressionLevel(level int) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // FormatCloseMessage formats closeCode and text as a WebSocket close message. | // FormatCloseMessage formats closeCode and text as a WebSocket close message. | ||||||
|  | // An empty message is returned for code CloseNoStatusReceived. | ||||||
| func FormatCloseMessage(closeCode int, text string) []byte { | func FormatCloseMessage(closeCode int, text string) []byte { | ||||||
|  | 	if closeCode == CloseNoStatusReceived { | ||||||
|  | 		// Return empty message because it's illegal to send | ||||||
|  | 		// CloseNoStatusReceived. Return non-nil value in case application | ||||||
|  | 		// checks for nil. | ||||||
|  | 		return []byte{} | ||||||
|  | 	} | ||||||
| 	buf := make([]byte, 2+len(text)) | 	buf := make([]byte, 2+len(text)) | ||||||
| 	binary.BigEndian.PutUint16(buf, uint16(closeCode)) | 	binary.BigEndian.PutUint16(buf, uint16(closeCode)) | ||||||
| 	copy(buf[2:], text) | 	copy(buf[2:], text) | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								vendor/github.com/gorilla/websocket/conn_read_legacy.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,21 +0,0 @@ | |||||||
| // Copyright 2016 The Gorilla WebSocket Authors. All rights reserved. |  | ||||||
| // Use of this source code is governed by a BSD-style |  | ||||||
| // license that can be found in the LICENSE file. |  | ||||||
|  |  | ||||||
| // +build !go1.5 |  | ||||||
|  |  | ||||||
| package websocket |  | ||||||
|  |  | ||||||
| import "io" |  | ||||||
|  |  | ||||||
| func (c *Conn) read(n int) ([]byte, error) { |  | ||||||
| 	p, err := c.br.Peek(n) |  | ||||||
| 	if err == io.EOF { |  | ||||||
| 		err = errUnexpectedEOF |  | ||||||
| 	} |  | ||||||
| 	if len(p) > 0 { |  | ||||||
| 		// advance over the bytes just read |  | ||||||
| 		io.ReadFull(c.br, p) |  | ||||||
| 	} |  | ||||||
| 	return p, err |  | ||||||
| } |  | ||||||
| @@ -2,17 +2,14 @@ | |||||||
| // Use of this source code is governed by a BSD-style | // Use of this source code is governed by a BSD-style | ||||||
| // license that can be found in the LICENSE file. | // license that can be found in the LICENSE file. | ||||||
| 
 | 
 | ||||||
| // +build go1.5 | // +build go1.8 | ||||||
| 
 | 
 | ||||||
| package websocket | package websocket | ||||||
| 
 | 
 | ||||||
| import "io" | import "net" | ||||||
| 
 | 
 | ||||||
| func (c *Conn) read(n int) ([]byte, error) { | func (c *Conn) writeBufs(bufs ...[]byte) error { | ||||||
| 	p, err := c.br.Peek(n) | 	b := net.Buffers(bufs) | ||||||
| 	if err == io.EOF { | 	_, err := b.WriteTo(c.conn) | ||||||
| 		err = errUnexpectedEOF | 	return err | ||||||
| 	} |  | ||||||
| 	c.br.Discard(len(p)) |  | ||||||
| 	return p, err |  | ||||||
| } | } | ||||||
							
								
								
									
										18
									
								
								vendor/github.com/gorilla/websocket/conn_write_legacy.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | // Copyright 2016 The Gorilla WebSocket Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a BSD-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  |  | ||||||
|  | // +build !go1.8 | ||||||
|  |  | ||||||
|  | package websocket | ||||||
|  |  | ||||||
|  | func (c *Conn) writeBufs(bufs ...[]byte) error { | ||||||
|  | 	for _, buf := range bufs { | ||||||
|  | 		if len(buf) > 0 { | ||||||
|  | 			if _, err := c.conn.Write(buf); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										56
									
								
								vendor/github.com/gorilla/websocket/doc.go
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						| @@ -6,9 +6,8 @@ | |||||||
| // | // | ||||||
| // Overview | // Overview | ||||||
| // | // | ||||||
| // The Conn type represents a WebSocket connection. A server application uses | // The Conn type represents a WebSocket connection. A server application calls | ||||||
| // the Upgrade function from an Upgrader object with a HTTP request handler | // the Upgrader.Upgrade method from an HTTP request handler to get a *Conn: | ||||||
| // to get a pointer to a Conn: |  | ||||||
| // | // | ||||||
| //  var upgrader = websocket.Upgrader{ | //  var upgrader = websocket.Upgrader{ | ||||||
| //      ReadBufferSize:  1024, | //      ReadBufferSize:  1024, | ||||||
| @@ -31,10 +30,12 @@ | |||||||
| //  for { | //  for { | ||||||
| //      messageType, p, err := conn.ReadMessage() | //      messageType, p, err := conn.ReadMessage() | ||||||
| //      if err != nil { | //      if err != nil { | ||||||
|  | //          log.Println(err) | ||||||
| //          return | //          return | ||||||
| //      } | //      } | ||||||
| //      if err = conn.WriteMessage(messageType, p); err != nil { | //      if err := conn.WriteMessage(messageType, p); err != nil { | ||||||
| //          return err | //          log.Println(err) | ||||||
|  | //          return | ||||||
| //      } | //      } | ||||||
| //  } | //  } | ||||||
| // | // | ||||||
| @@ -85,20 +86,26 @@ | |||||||
| // and pong. Call the connection WriteControl, WriteMessage or NextWriter | // and pong. Call the connection WriteControl, WriteMessage or NextWriter | ||||||
| // methods to send a control message to the peer. | // methods to send a control message to the peer. | ||||||
| // | // | ||||||
| // Connections handle received close messages by sending a close message to the | // Connections handle received close messages by calling the handler function | ||||||
| // peer and returning a *CloseError from the the NextReader, ReadMessage or the | // set with the SetCloseHandler method and by returning a *CloseError from the | ||||||
| // message Read method. | // NextReader, ReadMessage or the message Read method. The default close | ||||||
|  | // handler sends a close message to the peer. | ||||||
| // | // | ||||||
| // Connections handle received ping and pong messages by invoking callback | // Connections handle received ping messages by calling the handler function | ||||||
| // functions set with SetPingHandler and SetPongHandler methods. The callback | // set with the SetPingHandler method. The default ping handler sends a pong | ||||||
| // functions are called from the NextReader, ReadMessage and the message Read | // message to the peer. | ||||||
| // methods. |  | ||||||
| // | // | ||||||
| // The default ping handler sends a pong to the peer. The application's reading | // Connections handle received pong messages by calling the handler function | ||||||
| // goroutine can block for a short time while the handler writes the pong data | // set with the SetPongHandler method. The default pong handler does nothing. | ||||||
| // to the connection. | // If an application sends ping messages, then the application should set a | ||||||
|  | // pong handler to receive the corresponding pong. | ||||||
| // | // | ||||||
| // The application must read the connection to process ping, pong and close | // The control message handler functions are called from the NextReader, | ||||||
|  | // ReadMessage and message reader Read methods. The default close and ping | ||||||
|  | // handlers can block these methods for a short time when the handler writes to | ||||||
|  | // the connection. | ||||||
|  | // | ||||||
|  | // The application must read the connection to process close, ping and pong | ||||||
| // messages sent from the peer. If the application is not otherwise interested | // messages sent from the peer. If the application is not otherwise interested | ||||||
| // in messages from the peer, then the application should start a goroutine to | // in messages from the peer, then the application should start a goroutine to | ||||||
| // read and discard messages from the peer. A simple example is: | // read and discard messages from the peer. A simple example is: | ||||||
| @@ -137,19 +144,12 @@ | |||||||
| // method fails the WebSocket handshake with HTTP status 403. | // method fails the WebSocket handshake with HTTP status 403. | ||||||
| // | // | ||||||
| // If the CheckOrigin field is nil, then the Upgrader uses a safe default: fail | // If the CheckOrigin field is nil, then the Upgrader uses a safe default: fail | ||||||
| // the handshake if the Origin request header is present and not equal to the | // the handshake if the Origin request header is present and the Origin host is | ||||||
| // Host request header. | // not equal to the Host request header. | ||||||
| // | // | ||||||
| // An application can allow connections from any origin by specifying a | // The deprecated package-level Upgrade function does not perform origin | ||||||
| // function that always returns true: | // checking. The application is responsible for checking the Origin header | ||||||
| // | // before calling the Upgrade function. | ||||||
| //  var upgrader = websocket.Upgrader{ |  | ||||||
| //      CheckOrigin: func(r *http.Request) bool { return true }, |  | ||||||
| //  } |  | ||||||
| // |  | ||||||
| // The deprecated Upgrade function does not enforce an origin policy. It's the |  | ||||||
| // application's responsibility to check the Origin header before calling |  | ||||||
| // Upgrade. |  | ||||||
| // | // | ||||||
| // Compression EXPERIMENTAL | // Compression EXPERIMENTAL | ||||||
| // | // | ||||||
|   | |||||||