Merge lp:~widelands-dev/widelands-metaserver/ircbridge into lp:widelands-metaserver
- ircbridge
- Merge into trunk
Status: | Merged | ||||
---|---|---|---|---|---|
Merged at revision: | 47 | ||||
Proposed branch: | lp:~widelands-dev/widelands-metaserver/ircbridge | ||||
Merge into: | lp:widelands-metaserver | ||||
Diff against target: |
291 lines (+127/-15) 6 files modified
.bzrignore (+1/-0) wlms/client.go (+1/-0) wlms/ircbridge.go (+71/-0) wlms/main.go (+14/-3) wlms/server.go (+34/-11) wlms/server_test.go (+6/-1) |
||||
To merge this branch: | bzr merge lp:~widelands-dev/widelands-metaserver/ircbridge | ||||
Related bugs: |
|
Reviewer | Review Type | Date Requested | Status |
---|---|---|---|
Tino | Needs Resubmitting | ||
SirVer | Needs Fixing | ||
Review via email: mp+203277@code.launchpad.net |
Commit message
Description of the change
A first draft of a irce bridge. Lets the metaserver connect to a irc channel an mirrors messages between the channel and the metaserver lobby.
SirVer (sirver) wrote : | # |
SirVer (sirver) : | # |
Tino (tino79) wrote : | # |
Thanks for the comments, of course i only wanted a review, not a merge in the current state.
This are my first steps with go, so i am just glad that it works at all ;).
- i will add tests
- the irc library handles disconnects itself, and because of the use of a callback function for irc->lobby it is already decoupled. I disconnected from the internet to test and it automatically reconnected and the lobby was still usable. But i will look into using a routine and buffers...
- pointer recievers:
Here i am really unsure, but i've found:
A method is a function with an implicit first argument, called a receiver.
So when you call a method:
- if the receiver is a pointer to int, it copies the pointer to int.
- if the receiver is a struct, it copies the struct.
- if the receiver is a pointer to struct, it copies the pointer to struct.
So even if i don't want to modify the struct i would think avoiding a copy operation by passing a pointer to the struct should be faster? But here speaks only 1 day go experience... ;)
SirVer (sirver) wrote : | # |
> But i will look into using a routine and buffers...
I understand your comment, but I still think it is a good idea. First, because it is more idiomatic go (as far as I see it right now :)) and secondly goroutines and channels are essential building blocks that somehow set the language apart - you'll find them really interesting I think.
> So even if i don't want to modify the struct i would think avoiding a copy operation by passing a pointer to the struct should be faster?
While this is correct, go optimizes these very strongly. Sofar not using a pointer has not been a performance bottleneck in go code for me. And not using a pointer signifies to the programmer that the method will not change the receiver - which is nice documentation. So only use a pointer receiver when you need to (i.e. when you need to modify the receiver) or when you measured that it impacts performance by benchmarking.
- 49. By Tino
-
revert to non pointer recievers when the struct does not get manipulated
- 50. By Tino
-
read irc configuration from file
- 51. By Tino
-
IRCBridge "interfaced", does compile and run again
- 52. By Tino
-
pushing messages to irc via channel
- 53. By Tino
-
go channels 2way irc<->lobby
- 54. By Tino
-
channel refactoring
tests are ok again (no testing of irc functionality right now) - 55. By Tino
-
non blocking sending
Tino (tino79) wrote : | # |
Ok, channels are in place.
Next up are tests, but first i need to understand go tests. The current tests starts dozens of server instances in a few seconds, i need to use only one irc bridge instance for all tests or have to implement my own tests with active irc bridge.
SirVer (sirver) wrote : | # |
I do not fully understand your question.
If you only want to test the behavior of the server you can just hand in two channels and fake messages on those in your server tests (i.e. no IRC Bridge is created at all). I think this is the most reasonable approach to take since you cannot rely on IRC being up when you want to test your scripts.
You can also choose not to test your new implementations, but you must at least change the setup procedure so that not necessarily an IRC connection is made when a server is created.
SirVer (sirver) wrote : | # |
And progress on this? Anything I can help you with? If you are stuck we can also remote pair on the code for a bit, maybe we come up with something together.
Tino (tino79) wrote : | # |
The current state is:
- it is working, communication between ircbridge and metaserver via nonblocking channels
- tests are working, result ok
- no tests regarding the ircbridge itself or communication implemented
I am a bit busy atm, i think i won't have time to implement further test before next week.
SirVer (sirver) wrote : | # |
no pressure - just wondered if you still want to work on this or if it should be merged around now.
Tino (tino79) wrote : | # |
Ok, perhaps we can merge now?
I does work(tm), only the testcase i've drafted seems to do nothing?
SirVer (sirver) wrote : | # |
looks good to me. I merged and deployed on the server - thanks for your work on this!
Preview Diff
1 | === added file '.bzrignore' | |||
2 | --- .bzrignore 1970-01-01 00:00:00 +0000 | |||
3 | +++ .bzrignore 2014-03-17 19:05:27 +0000 | |||
4 | @@ -0,0 +1,1 @@ | |||
5 | 1 | *.exe | ||
6 | 0 | 2 | ||
7 | === modified file 'wlms/client.go' | |||
8 | --- wlms/client.go 2014-02-03 21:14:19 +0000 | |||
9 | +++ wlms/client.go 2014-03-17 19:05:27 +0000 | |||
10 | @@ -277,6 +277,7 @@ | |||
11 | 277 | 277 | ||
12 | 278 | if len(receiver) == 0 { | 278 | if len(receiver) == 0 { |
13 | 279 | server.BroadcastToConnectedClients("CHAT", client.Name(), message, "public") | 279 | server.BroadcastToConnectedClients("CHAT", client.Name(), message, "public") |
14 | 280 | server.BroadcastToIrc(client.Name() + ": " + message) | ||
15 | 280 | } else { | 281 | } else { |
16 | 281 | recv_client := server.HasClient(receiver) | 282 | recv_client := server.HasClient(receiver) |
17 | 282 | if recv_client != nil { | 283 | if recv_client != nil { |
18 | 283 | 284 | ||
19 | === added file 'wlms/ircbridge.go' | |||
20 | --- wlms/ircbridge.go 1970-01-01 00:00:00 +0000 | |||
21 | +++ wlms/ircbridge.go 2014-03-17 19:05:27 +0000 | |||
22 | @@ -0,0 +1,71 @@ | |||
23 | 1 | package main | ||
24 | 2 | |||
25 | 3 | import ( | ||
26 | 4 | "log" | ||
27 | 5 | |||
28 | 6 | "github.com/thoj/go-ircevent" | ||
29 | 7 | ) | ||
30 | 8 | |||
31 | 9 | type IRCBridger interface { | ||
32 | 10 | Connect(chan Message, chan Message) bool | ||
33 | 11 | Quit() | ||
34 | 12 | } | ||
35 | 13 | |||
36 | 14 | type IRCBridge struct { | ||
37 | 15 | connection *irc.Connection | ||
38 | 16 | nick, user, channel, server string | ||
39 | 17 | useTLS bool | ||
40 | 18 | } | ||
41 | 19 | |||
42 | 20 | type Message struct { | ||
43 | 21 | message, nick string | ||
44 | 22 | } | ||
45 | 23 | |||
46 | 24 | func NewIRCBridge(server, realname, nickname, channel string, tls bool) *IRCBridge { | ||
47 | 25 | return &IRCBridge{ | ||
48 | 26 | server: server, | ||
49 | 27 | user: realname, | ||
50 | 28 | nick: nickname, | ||
51 | 29 | channel: channel, | ||
52 | 30 | useTLS: tls, | ||
53 | 31 | } | ||
54 | 32 | } | ||
55 | 33 | |||
56 | 34 | func (bridge *IRCBridge) Connect(messagesIn chan Message, messagesOut chan Message) bool { | ||
57 | 35 | //Create new connection | ||
58 | 36 | bridge.connection = irc.IRC(bridge.nick, bridge.user) | ||
59 | 37 | //Set options | ||
60 | 38 | bridge.connection.UseTLS = bridge.useTLS | ||
61 | 39 | //connection.TLSOptions //set ssl options | ||
62 | 40 | //connection.Password = "[server password]" | ||
63 | 41 | //Commands | ||
64 | 42 | err := bridge.connection.Connect(bridge.server) //Connect to server | ||
65 | 43 | if err != nil { | ||
66 | 44 | log.Fatal("Can't connect %s", bridge.server) | ||
67 | 45 | return false | ||
68 | 46 | } | ||
69 | 47 | bridge.connection.Join(bridge.channel) | ||
70 | 48 | bridge.connection.AddCallback("PRIVMSG", func(event *irc.Event) { | ||
71 | 49 | //e.Message contains the message | ||
72 | 50 | //e.Nick Contains the sender | ||
73 | 51 | //e.Arguments[0] Contains the channel | ||
74 | 52 | select { | ||
75 | 53 | case messagesOut <- Message{nick: event.Nick, | ||
76 | 54 | message: event.Message(), | ||
77 | 55 | }: | ||
78 | 56 | default: | ||
79 | 57 | log.Println("Message Queue full.") | ||
80 | 58 | } | ||
81 | 59 | |||
82 | 60 | }) | ||
83 | 61 | go func() { | ||
84 | 62 | for m := range messagesIn { | ||
85 | 63 | bridge.connection.Privmsg(bridge.channel, m.message) | ||
86 | 64 | } | ||
87 | 65 | }() | ||
88 | 66 | return true | ||
89 | 67 | } | ||
90 | 68 | |||
91 | 69 | func (bridge *IRCBridge) Quit() { | ||
92 | 70 | bridge.connection.Quit() | ||
93 | 71 | } | ||
94 | 0 | 72 | ||
95 | === modified file 'wlms/main.go' | |||
96 | --- wlms/main.go 2013-12-30 12:20:31 +0000 | |||
97 | +++ wlms/main.go 2014-03-17 19:05:27 +0000 | |||
98 | @@ -8,7 +8,8 @@ | |||
99 | 8 | ) | 8 | ) |
100 | 9 | 9 | ||
101 | 10 | type Config struct { | 10 | type Config struct { |
103 | 11 | Database, User, Password, Table string | 11 | Database, User, Password, Table, Backend, IRCServer, Nickname, Realname, Channel string |
104 | 12 | UseTLS bool | ||
105 | 12 | } | 13 | } |
106 | 13 | 14 | ||
107 | 14 | func (l *Config) ConfigFrom(path string) error { | 15 | func (l *Config) ConfigFrom(path string) error { |
108 | @@ -25,16 +26,26 @@ | |||
109 | 25 | flag.Parse() | 26 | flag.Parse() |
110 | 26 | 27 | ||
111 | 27 | var db UserDb | 28 | var db UserDb |
112 | 29 | var ircbridge IRCBridger | ||
113 | 28 | if config != "" { | 30 | if config != "" { |
114 | 29 | var cfg Config | 31 | var cfg Config |
115 | 30 | if err := cfg.ConfigFrom(config); err != nil { | 32 | if err := cfg.ConfigFrom(config); err != nil { |
116 | 31 | log.Fatalf("Could not parse config file: %v", err) | 33 | log.Fatalf("Could not parse config file: %v", err) |
117 | 32 | } | 34 | } |
119 | 33 | db = NewMySqlDatabase(cfg.Database, cfg.User, cfg.Password, cfg.Table) | 35 | if cfg.Backend == "mysql" { |
120 | 36 | db = NewMySqlDatabase(cfg.Database, cfg.User, cfg.Password, cfg.Table) | ||
121 | 37 | } else { | ||
122 | 38 | db = NewInMemoryDb() | ||
123 | 39 | } | ||
124 | 40 | ircbridge = NewIRCBridge(cfg.IRCServer, cfg.Realname, cfg.Nickname, cfg.Channel, cfg.UseTLS) | ||
125 | 34 | } else { | 41 | } else { |
126 | 35 | db = NewInMemoryDb() | 42 | db = NewInMemoryDb() |
127 | 36 | } | 43 | } |
128 | 37 | defer db.Close() | 44 | defer db.Close() |
129 | 38 | 45 | ||
131 | 39 | RunServer(db) | 46 | messagesToIrc := make(chan Message, 50) |
132 | 47 | messagesToLobby := make(chan Message, 50) | ||
133 | 48 | ircbridge.Connect(messagesToIrc, messagesToLobby) | ||
134 | 49 | RunServer(db, messagesToLobby, messagesToIrc) | ||
135 | 50 | |||
136 | 40 | } | 51 | } |
137 | 41 | 52 | ||
138 | === modified file 'wlms/server.go' | |||
139 | --- wlms/server.go 2014-02-08 20:48:04 +0000 | |||
140 | +++ wlms/server.go 2014-03-17 19:05:27 +0000 | |||
141 | @@ -33,10 +33,11 @@ | |||
142 | 33 | gamePingTimeout time.Duration | 33 | gamePingTimeout time.Duration |
143 | 34 | 34 | ||
144 | 35 | clientForgetTimeout time.Duration | 35 | clientForgetTimeout time.Duration |
146 | 36 | gamePingerFactory GampPingerFactory | 36 | gamePingerFactory GamePingerFactory |
147 | 37 | messagesOut chan Message | ||
148 | 37 | } | 38 | } |
149 | 38 | 39 | ||
151 | 39 | type GampPingerFactory interface { | 40 | type GamePingerFactory interface { |
152 | 40 | New(client *Client, timeout time.Duration) *GamePinger | 41 | New(client *Client, timeout time.Duration) *GamePinger |
153 | 41 | } | 42 | } |
154 | 42 | 43 | ||
155 | @@ -100,6 +101,7 @@ | |||
156 | 100 | } | 101 | } |
157 | 101 | 102 | ||
158 | 102 | func (s *Server) AddClient(client *Client) { | 103 | func (s *Server) AddClient(client *Client) { |
159 | 104 | s.BroadcastToIrc(client.Name() + " has joined the lobby.") | ||
160 | 103 | s.clients.PushBack(client) | 105 | s.clients.PushBack(client) |
161 | 104 | } | 106 | } |
162 | 105 | 107 | ||
163 | @@ -121,11 +123,15 @@ | |||
164 | 121 | if e.Value.(*Client) == client { | 123 | if e.Value.(*Client) == client { |
165 | 122 | log.Printf("Removing client %s.", client.Name()) | 124 | log.Printf("Removing client %s.", client.Name()) |
166 | 123 | s.clients.Remove(e) | 125 | s.clients.Remove(e) |
167 | 126 | s.messagesOut <- Message{ | ||
168 | 127 | message: client.Name() + " has left the lobby.", | ||
169 | 128 | nick: client.Name(), | ||
170 | 129 | } | ||
171 | 124 | } | 130 | } |
172 | 125 | } | 131 | } |
173 | 126 | } | 132 | } |
174 | 127 | 133 | ||
176 | 128 | func (s *Server) HasClient(name string) *Client { | 134 | func (s Server) HasClient(name string) *Client { |
177 | 129 | for e := s.clients.Front(); e != nil; e = e.Next() { | 135 | for e := s.clients.Front(); e != nil; e = e.Next() { |
178 | 130 | client := e.Value.(*Client) | 136 | client := e.Value.(*Client) |
179 | 131 | if client.Name() == name { | 137 | if client.Name() == name { |
180 | @@ -135,7 +141,7 @@ | |||
181 | 135 | return nil | 141 | return nil |
182 | 136 | } | 142 | } |
183 | 137 | 143 | ||
185 | 138 | func (s *Server) NrActiveClients() int { | 144 | func (s Server) NrActiveClients() int { |
186 | 139 | count := 0 | 145 | count := 0 |
187 | 140 | for e := s.clients.Front(); e != nil; e = e.Next() { | 146 | for e := s.clients.Front(); e != nil; e = e.Next() { |
188 | 141 | if e.Value.(*Client).State() == CONNECTED { | 147 | if e.Value.(*Client).State() == CONNECTED { |
189 | @@ -158,6 +164,7 @@ | |||
190 | 158 | func (s *Server) AddGame(game *Game) { | 164 | func (s *Server) AddGame(game *Game) { |
191 | 159 | s.games.PushBack(game) | 165 | s.games.PushBack(game) |
192 | 160 | s.BroadcastToConnectedClients("GAMES_UPDATE") | 166 | s.BroadcastToConnectedClients("GAMES_UPDATE") |
193 | 167 | s.BroadcastToIrc("A new game " + game.Name() + " was opened by " + game.Host()) | ||
194 | 161 | } | 168 | } |
195 | 162 | 169 | ||
196 | 163 | func (s *Server) RemoveGame(game *Game) { | 170 | func (s *Server) RemoveGame(game *Game) { |
197 | @@ -170,7 +177,7 @@ | |||
198 | 170 | } | 177 | } |
199 | 171 | } | 178 | } |
200 | 172 | 179 | ||
202 | 173 | func (s *Server) HasGame(name string) *Game { | 180 | func (s Server) HasGame(name string) *Game { |
203 | 174 | for e := s.games.Front(); e != nil; e = e.Next() { | 181 | for e := s.games.Front(); e != nil; e = e.Next() { |
204 | 175 | game := e.Value.(*Game) | 182 | game := e.Value.(*Game) |
205 | 176 | if game.Name() == name { | 183 | if game.Name() == name { |
206 | @@ -180,7 +187,7 @@ | |||
207 | 180 | return nil | 187 | return nil |
208 | 181 | } | 188 | } |
209 | 182 | 189 | ||
211 | 183 | func (s *Server) NrGames() int { | 190 | func (s Server) NrGames() int { |
212 | 184 | return s.games.Len() | 191 | return s.games.Len() |
213 | 185 | } | 192 | } |
214 | 186 | 193 | ||
215 | @@ -199,7 +206,18 @@ | |||
216 | 199 | } | 206 | } |
217 | 200 | } | 207 | } |
218 | 201 | 208 | ||
220 | 202 | func RunServer(db UserDb) { | 209 | func (s Server) BroadcastToIrc(message string) { |
221 | 210 | select { | ||
222 | 211 | case s.messagesOut <- Message{ | ||
223 | 212 | message: message, | ||
224 | 213 | }: | ||
225 | 214 | default: | ||
226 | 215 | log.Println("Message Queue full.") | ||
227 | 216 | } | ||
228 | 217 | |||
229 | 218 | } | ||
230 | 219 | |||
231 | 220 | func RunServer(db UserDb, messagesIn chan Message, messagesOut chan Message) { | ||
232 | 203 | ln, err := net.Listen("tcp", ":7395") | 221 | ln, err := net.Listen("tcp", ":7395") |
233 | 204 | if err != nil { | 222 | if err != nil { |
234 | 205 | log.Fatal(err) | 223 | log.Fatal(err) |
235 | @@ -216,8 +234,7 @@ | |||
236 | 216 | C <- conn | 234 | C <- conn |
237 | 217 | } | 235 | } |
238 | 218 | }() | 236 | }() |
241 | 219 | 237 | CreateServerUsing(C, db, messagesIn, messagesOut).WaitTillShutdown() | |
240 | 220 | CreateServerUsing(C, db).WaitTillShutdown() | ||
242 | 221 | } | 238 | } |
243 | 222 | 239 | ||
244 | 223 | type RealGamePingerFactory struct { | 240 | type RealGamePingerFactory struct { |
245 | @@ -255,11 +272,11 @@ | |||
246 | 255 | return pinger | 272 | return pinger |
247 | 256 | } | 273 | } |
248 | 257 | 274 | ||
250 | 258 | func (server *Server) InjectGamePingerFactory(gpf GampPingerFactory) { | 275 | func (server *Server) InjectGamePingerFactory(gpf GamePingerFactory) { |
251 | 259 | server.gamePingerFactory = gpf | 276 | server.gamePingerFactory = gpf |
252 | 260 | } | 277 | } |
253 | 261 | 278 | ||
255 | 262 | func CreateServerUsing(acceptedConnections chan ReadWriteCloserWithIp, db UserDb) *Server { | 279 | func CreateServerUsing(acceptedConnections chan ReadWriteCloserWithIp, db UserDb, messagesIn chan Message, messagesOut chan Message) *Server { |
256 | 263 | server := &Server{ | 280 | server := &Server{ |
257 | 264 | acceptedConnections: acceptedConnections, | 281 | acceptedConnections: acceptedConnections, |
258 | 265 | shutdownServer: make(chan bool), | 282 | shutdownServer: make(chan bool), |
259 | @@ -272,8 +289,14 @@ | |||
260 | 272 | pingCycleTime: time.Second * 15, | 289 | pingCycleTime: time.Second * 15, |
261 | 273 | clientSendingTimeout: time.Minute * 2, | 290 | clientSendingTimeout: time.Minute * 2, |
262 | 274 | clientForgetTimeout: time.Minute * 5, | 291 | clientForgetTimeout: time.Minute * 5, |
263 | 292 | messagesOut: messagesOut, | ||
264 | 275 | } | 293 | } |
265 | 276 | server.gamePingerFactory = RealGamePingerFactory{server} | 294 | server.gamePingerFactory = RealGamePingerFactory{server} |
266 | 295 | go func() { | ||
267 | 296 | for m := range messagesIn { | ||
268 | 297 | server.BroadcastToConnectedClients("CHAT", m.nick+"(IRC)", m.message, "public") | ||
269 | 298 | } | ||
270 | 299 | }() | ||
271 | 277 | go server.mainLoop() | 300 | go server.mainLoop() |
272 | 278 | return server | 301 | return server |
273 | 279 | } | 302 | } |
274 | 280 | 303 | ||
275 | === modified file 'wlms/server_test.go' | |||
276 | --- wlms/server_test.go 2014-02-08 20:48:04 +0000 | |||
277 | +++ wlms/server_test.go 2014-03-17 19:05:27 +0000 | |||
278 | @@ -40,7 +40,12 @@ | |||
279 | 40 | cons[i] = NewFakeConn(c) | 40 | cons[i] = NewFakeConn(c) |
280 | 41 | acceptingConnections <- cons[i] | 41 | acceptingConnections <- cons[i] |
281 | 42 | } | 42 | } |
283 | 43 | return CreateServerUsing(acceptingConnections, db), cons | 43 | |
284 | 44 | //irc := NewIRCBridge("chat.freenode.net:7000", "wltest", "wltest", "#widelands-test", true) | ||
285 | 45 | toIrc := make(chan Message, 100) | ||
286 | 46 | fromIrc := make(chan Message, 100) | ||
287 | 47 | //irc.Connect(toIrc, fromIrc) | ||
288 | 48 | return CreateServerUsing(acceptingConnections, db, fromIrc, toIrc), cons | ||
289 | 44 | } | 49 | } |
290 | 45 | 50 | ||
291 | 46 | type Matching string | 51 | type Matching string |
Nice. That is a really good idea and I think the implementation looks reasonable too. It is a bit drafty still, though (see comments below). Was this the first time you did something with go? How did you like it?
Some comments:
- func NewIRCBridge() IRCBridge should read its configuration from the same json file where we read the data from right now too (see main.go).
- I think the design is a little fragile - if something goes wrong with the IRC connection the whole metaserver comes down. How about running the IRC bridge in a go routine that gets a (buffered) channel of stuff to send and also sends stuff into a channel that the metaserver then displays? That way if the bridge is disconnected it can just discard the messages and the whole design is a bit decoupled. Also you can fake some IRC messages in a test and test the whole thing.
- +func (s *Server) Motd() string - I do not agree with this change. When a method is not changing a struct, it should not be a pointer receiver. There are only a few exceptions to this rule as far as I understood go so far (not an expert by any means).
- Please add some tests. Feel free to add end to end tests, there are already a few in the metaserver.