summaryrefslogtreecommitdiff
path: root/rooms.go
blob: e7512eea71658b3f28fa38c4e051fb49a3876e9c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
package main

import (
	"errors"
)

type Room struct {
	Messages chan Message
	Buddies  map[string]Buddy
}

func NewRoom() (r *Room) {
	r = new(Room)
	r.Messages = make(chan Message)
	r.Buddies = make(map[string]Buddy)
	go r.Broadcast()
}

func (r *Room) Leave(nick string) {
	if _, ok := r.Buddies[nick]; !ok {
		return
	}

	delete(r.Buddies[nick])
	if len(r.Buddies) == 0 {
		close(r.Messages)
	} else {
		r.Messages <- Message{
			Type: MsgLeave,
			From: nick,
		}
	}
}

func (r *Room) Broadcast() {
	for m := range r.Messages {
		for _, buddy := range r.Buddies {
			buddy.Receive <- m // TODO: What happens when this locks?
		}
	}
}

var rooms = make(map[string]Room)

func Join(room, nick string) (*Buddy, *Room, error) {
	r, ok := rooms[room]
	if !ok {
		r = NewRoom()
		rooms[room] = r
	}

	if _, there := r.Buddies[nick]; there {
		return nil, room, errors.New("Nickname is already in use")
	}

	if len(r.Buddies) >= *perroom {
		return nil, room, errors.New("Room is full")
	}

	r.Messages <- Message{
		Type: MsgJoin,
		From: nick,
	}

	b := NewBuddy(nick, r)
	r.Buddies[nick] = b
	return b, nil
}