Merge of r85:88 from branch polarssl into trunk.
[umurmur.git] / src / channel.c
1 /* Copyright (C) 2009-2010, Martin Johansson <martin@fatbob.nu>
2    Copyright (C) 2005-2010, Thorvald Natvig <thorvald@natvig.com>
3
4    All rights reserved.
5
6    Redistribution and use in source and binary forms, with or without
7    modification, are permitted provided that the following conditions
8    are met:
9
10    - Redistributions of source code must retain the above copyright notice,
11      this list of conditions and the following disclaimer.
12    - Redistributions in binary form must reproduce the above copyright notice,
13      this list of conditions and the following disclaimer in the documentation
14      and/or other materials provided with the distribution.
15    - Neither the name of the Developers nor the names of its contributors may
16      be used to endorse or promote products derived from this software without
17      specific prior written permission.
18
19    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20    ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22    A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR
23    CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
24    EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
25    PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
26    PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
27    LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
28    NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
29    SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 */
31 #include <limits.h>
32 #include <stdlib.h>
33 #include <string.h>
34 #include "log.h"
35 #include "list.h"
36 #include "client.h"
37 #include "channel.h"
38 #include "conf.h"
39
40
41 static channel_t *rootChan;
42 channel_t *defaultChan;
43 declare_list(channels); /* A flat list of the channels */
44
45 static channel_t *createChannel(int id, const char *name, const char *desc)
46 {
47         channel_t *ch;
48
49         ch = malloc(sizeof(channel_t));
50         if (ch == NULL)
51                 Log_fatal("out of memory");
52         memset(ch, 0, sizeof(channel_t));
53         ch->id = id;
54         ch->name = strdup(name);
55         if (desc)
56                 ch->desc = strdup(desc);
57         init_list_entry(&ch->subs);
58         init_list_entry(&ch->node);
59         init_list_entry(&ch->clients);
60         init_list_entry(&ch->flatlist_node);
61         init_list_entry(&ch->channel_links);
62         return ch;
63 }
64
65 static int findFreeId()
66 {
67         int id = 0;
68         channel_t *ch_itr = NULL;
69         for (id = 0; id < INT_MAX; id++) {
70                 ch_itr = NULL;
71                 while ((ch_itr = Chan_iterate(&ch_itr)) != NULL) {
72                         if (ch_itr->id == id)
73                                 break;
74                 }
75                 if (ch_itr == NULL) /* Found free id */
76                         return id;
77         }
78         return -1;
79 }
80
81 #if 0
82 /* Might be used when tree traversal becomes neccessary */
83 static channel_t *first_subchannel(channel_t *ch)
84 {
85         if (list_empty(&ch->subs))
86                 return NULL;
87         else
88                 return list_get_entry(list_get_first(&ch->subs), channel_t, node);
89 }
90
91 static channel_t *next_channel(channel_t *ch)
92 {
93         if (list_get_next(&ch->node) == &list_get_entry(&ch->node, channel_t, node)->parent->subs)
94                 return NULL;
95         else
96                 return list_get_entry(list_get_next(&ch->node), channel_t, node);       
97 }
98 #endif
99
100 channel_t *Chan_iterate(channel_t **channelpptr)
101 {
102         channel_t *ch = *channelpptr;
103
104         if (!list_empty(&channels)) {
105                 if (ch == NULL)
106                         ch = list_get_entry(list_get_first(&channels), channel_t, flatlist_node);
107                 else {
108                         if (list_get_next(&ch->flatlist_node) == &channels)
109                                 ch = NULL;
110                         else
111                                 ch = list_get_entry(list_get_next(&ch->flatlist_node), channel_t, flatlist_node);
112                 }
113         }
114
115         *channelpptr = ch;
116         return ch;
117 }
118
119 channel_t *Chan_iterate_siblings(channel_t *parent, channel_t **channelpptr)
120 {
121         channel_t *ch = *channelpptr;
122
123         if (!list_empty(&parent->subs)) {
124                 if (ch == NULL)
125                         ch = list_get_entry(list_get_first(&parent->subs), channel_t, node);
126                 else {
127                         if (list_get_next(&ch->node) == &parent->subs)
128                                 ch = NULL;
129                         else
130                                 ch = list_get_entry(list_get_next(&ch->node), channel_t, node);
131                 }
132         }
133
134         *channelpptr = ch;
135         return ch;
136 }
137                         
138 void Chan_init()
139 {
140         int i;
141         conf_channel_t chdesc;
142         conf_channel_link_t chlink;
143         const char *defaultChannelName;
144
145         defaultChannelName = getStrConf(DEFAULT_CHANNEL);
146         
147         for (i = 0; ; i++) {
148                 if (Conf_getNextChannel(&chdesc, i) < 0) {
149                         if (i == 0)
150                                 Log_fatal("No valid channels found in configuration file. Exiting.");
151                         break;
152                 }
153                 if (i == 0) {
154                         rootChan = createChannel(0, chdesc.name, chdesc.description);
155                         rootChan->noenter = chdesc.noenter;
156                         list_add_tail(&rootChan->flatlist_node, &channels);
157                         if (strcmp(defaultChannelName, chdesc.name) == 0)
158                                 defaultChan = rootChan;
159                 }
160                 else {
161                         channel_t *ch, *ch_itr = NULL;
162                         ch = Chan_createChannel(chdesc.name, chdesc.description);
163                         ch->noenter = chdesc.noenter;
164                         
165                         if (strcmp(defaultChannelName, chdesc.name) == 0) {
166                                 Log_info("Setting default channel %s", ch->name); 
167                                 defaultChan = ch;
168                         }
169                         
170                         do {
171                                 Chan_iterate(&ch_itr);
172                         } while (ch_itr != NULL && strcmp(ch_itr->name, chdesc.parent) != 0);
173                         
174                         if (ch_itr == NULL)
175                                 Log_fatal("Error in channel configuration: parent not found");
176                         else {
177                                 Chan_addChannel(ch_itr, ch);
178                                 Log_info("Adding channel '%s' parent '%s'", ch->name, chdesc.parent);
179                         }
180                 }
181         }
182         if (defaultChan == NULL)
183                 defaultChan = rootChan;
184         
185         if (defaultChan->noenter)
186                 Log_fatal("Error in channel configuration: default channel is marked as noenter");
187
188         /* Channel links */
189         for (i = 0; ; i++) {
190                 channel_t *ch_src, *ch_dst, *ch_itr = NULL;
191                 if (Conf_getNextChannelLink(&chlink, i) < 0) {
192                         if (i == 0)
193                                 Log_info("No channel links found in configuration file.");
194                         break;
195                 }
196                 ch_itr = NULL;
197                 do {
198                         Chan_iterate(&ch_itr);
199                 } while (ch_itr != NULL && strcmp(ch_itr->name, chlink.source) != 0);
200                 if (ch_itr == NULL)
201                         Log_fatal("Error in channel link configuration: source channel '%s' not found.",
202                                           chlink.source);
203                 else
204                         ch_src = ch_itr;
205                 
206                 ch_itr = NULL;          
207                 do {
208                         Chan_iterate(&ch_itr);
209                 } while (ch_itr != NULL && strcmp(ch_itr->name, chlink.destination) != 0);
210                 if (ch_itr == NULL)
211                         Log_fatal("Error in channel link configuration: destination channel '%s' not found",
212                                           chlink.destination);
213                 else
214                         ch_dst = ch_itr;
215                 
216                 list_add_tail(&ch_dst->link_node, &ch_src->channel_links);
217                 ch_src->linkcount++;
218                 Log_info("Adding channel link '%s' -> '%s'", ch_src->name, ch_dst->name);
219         }
220 }
221
222 void Chan_free()
223 {
224         struct dlist *itr, *save;
225         channel_t *ch;
226         
227         list_iterate_safe(itr, save, &channels) {
228                 ch = list_get_entry(itr, channel_t, flatlist_node);
229                 Log_debug("Free channel '%s'", ch->name);
230                 free(ch->name);
231                 if (ch->desc)
232                         free(ch->desc);
233                 free(ch);
234         }
235 }
236
237 channel_t *Chan_createChannel(const char *name, const char *desc)
238 {
239         int id = findFreeId();
240         if (id < 0)
241                 Log_fatal("No free channel ID found");
242         return createChannel(id, name, desc);
243 }
244
245 void Chan_freeChannel(channel_t *ch)
246 {
247         list_del(&ch->node);
248         list_del(&ch->flatlist_node);
249         free(ch);
250 }
251
252 void Chan_addChannel(channel_t *parent, channel_t *ch)
253 {
254         list_add_tail(&ch->node, &parent->subs);
255         ch->parent = parent;
256         list_add_tail(&ch->flatlist_node, &channels);
257 }
258
259
260 int Chan_userLeave(client_t *client)
261 {
262         channel_t *leaving = NULL;
263         int leaving_id = -1;
264         
265         if (client->channel) {
266                 list_del(&client->chan_node);
267                 leaving = (channel_t *)client->channel;
268                 if (leaving->temporary && list_empty(&leaving->clients)) {
269                         leaving_id = leaving->id;
270                         Chan_freeChannel(leaving);
271                 }
272         }
273         return leaving_id;
274 }
275
276 int Chan_userJoin(channel_t *ch, client_t *client)
277 {
278         int leaving_id;
279         
280         Log_debug("Add user %s to channel %s", client->username, ch->name); 
281
282         /* Only allowed in one channel at a time */
283         leaving_id = Chan_userLeave(client);
284         list_add_tail(&client->chan_node, &ch->clients);
285         client->channel = (void *)ch;
286         return leaving_id;
287 }
288
289 int Chan_userJoin_id(int channelid, client_t *client)
290 {
291         channel_t *ch_itr = NULL;
292         do {
293                 Chan_iterate(&ch_itr);
294         } while (ch_itr != NULL && ch_itr->id != channelid);
295         if (ch_itr == NULL) {
296                 Log_warn("Channel id %d not found - ignoring.", channelid);
297                 return -1;
298         }
299         else
300                 return Chan_userJoin(ch_itr, client);   
301 }
302
303 bool_t Chan_userJoin_id_test(int channelid)
304 {
305         channel_t *ch_itr = NULL;
306         do {
307                 Chan_iterate(&ch_itr);
308         } while (ch_itr != NULL && ch_itr->id != channelid);
309         if (ch_itr == NULL) {
310                 Log_warn("Channel id %d not found - ignoring.", channelid);
311                 return false;
312         }
313         if (ch_itr->noenter)
314                 return false;
315         else
316                 return true;
317 }
318
319 #if 0
320 void Chan_addChannel_id(int parentId, channel_t *ch)
321 {
322         channel_t *ch_itr = NULL;
323         do {
324                 Chan_iterate(&ch_itr);
325         } while (ch_itr != NULL && ch_itr->id != parentId);
326         if (ch_itr == NULL)
327                 Log_warn("Chan_addChannel_id: Channel id %d not found - ignoring.", parentId);
328         else
329                 list_add_tail(&ch->node, &ch_itr->subs);
330 }
331 #endif
332
333 channel_t *Chan_fromId(int channelid)
334 {
335         channel_t *ch_itr = NULL;
336         do {
337                 Chan_iterate(&ch_itr);
338         } while (ch_itr != NULL && ch_itr->id != channelid);
339         if (ch_itr == NULL)
340                 Log_warn("Chan_fromId: Channel id %d not found.", channelid);
341         return ch_itr;
342 }
343
344 void Chan_removeChannel(channel_t *ch)
345 {
346         list_del(&ch->node);
347 }
348
349 void Chan_buildTreeList(channel_t *ch, struct dlist *head)
350 {
351         channellist_t *chl;
352         struct dlist *itr;
353         channel_t *sub;
354         
355         chl = malloc(sizeof(channellist_t));
356         chl->chan = ch;
357         init_list_entry(&chl->node);
358         list_add_tail(&chl->node, head);
359
360         list_iterate(itr, &ch->subs) {
361                 sub = list_get_entry(itr, channel_t, node);
362                 Chan_buildTreeList(sub, head);
363         }
364 }
365
366 void Chan_freeTreeList(struct dlist *head)
367 {
368         struct dlist *itr, *save;
369         list_iterate_safe(itr, save, head) {
370                 free(list_get_entry(itr, channellist_t, node));
371         }
372 }