]> jfr.im git - irc/quakenet/snircd-patchqueue.git/blame - split.patch
welcome.patch redone how messages are saved - split.patch small changes
[irc/quakenet/snircd-patchqueue.git] / split.patch
CommitLineData
12e82c05 1work in progress
2Add split functionality into ircd
3Add /split command
4Add /stats S/split (make /STATS s/S case sensitive, s spoofhosts)
5Add feature SPLIT
6Add split.c split.h m_split.c
7
7efbed3b 8NOTE: feature SPLIT is default enabled as that is just easier when working on this patch.
9
cb0b5df1 10diff -r 5ec670983aff doc/api/log.txt
11--- a/doc/api/log.txt Wed Jan 28 15:15:55 2009 +0100
12+++ b/doc/api/log.txt Wed Jan 28 15:46:43 2009 +0100
0c63b280 13@@ -41,7 +41,7 @@
14
15 <enum>
16 enum LogSys {
17- LS_SYSTEM, LS_CONFIG, LS_OPERMODE, LS_GLINE, LS_JUPE, LS_WHO, LS_NETWORK,
18+ LS_SYSTEM, LS_CONFIG, LS_OPERMODE, LS_GLINE, LS_JUPE, LS_SPLIT, LS_WHO, LS_NETWORK,
19 LS_OPERKILL, LS_SERVKILL, LS_USER, LS_OPER, LS_RESOLVER, LS_SOCKET,
20 LS_DEBUG, LS_OLDLOG,
21 LS_LAST_SYSTEM
cb0b5df1 22diff -r 5ec670983aff include/handlers.h
23--- a/include/handlers.h Wed Jan 28 15:15:55 2009 +0100
24+++ b/include/handlers.h Wed Jan 28 15:46:43 2009 +0100
12e82c05 25@@ -139,6 +139,7 @@
26 extern int m_registered(struct Client*, struct Client*, int, char*[]);
27 extern int m_sethost(struct Client*, struct Client*, int, char*[]);
28 extern int m_silence(struct Client*, struct Client*, int, char*[]);
29+extern int m_split(struct Client*, struct Client*, int, char*[]);
30 extern int m_stats(struct Client*, struct Client*, int, char*[]);
31 extern int m_time(struct Client*, struct Client*, int, char*[]);
32 extern int m_topic(struct Client*, struct Client*, int, char*[]);
d61bc4d5 33@@ -179,6 +180,7 @@
12e82c05 34 extern int mo_rping(struct Client*, struct Client*, int, char*[]);
35 extern int mo_set(struct Client*, struct Client*, int, char*[]);
36 extern int mo_settime(struct Client*, struct Client*, int, char*[]);
37+extern int mo_split(struct Client*, struct Client*, int, char*[]);
38 extern int mo_squit(struct Client*, struct Client*, int, char*[]);
39 extern int mo_stats(struct Client*, struct Client*, int, char*[]);
40 extern int mo_trace(struct Client*, struct Client*, int, char*[]);
d61bc4d5 41@@ -234,6 +236,7 @@
12e82c05 42 extern int ms_sethost(struct Client*, struct Client*, int, char*[]);
43 extern int ms_settime(struct Client*, struct Client*, int, char*[]);
44 extern int ms_silence(struct Client*, struct Client*, int, char*[]);
45+extern int ms_split(struct Client*, struct Client*, int, char*[]);
46 extern int ms_squit(struct Client*, struct Client*, int, char*[]);
47 extern int ms_stats(struct Client*, struct Client*, int, char*[]);
48 extern int ms_topic(struct Client*, struct Client*, int, char*[]);
cb0b5df1 49diff -r 5ec670983aff include/ircd_features.h
50--- a/include/ircd_features.h Wed Jan 28 15:15:55 2009 +0100
51+++ b/include/ircd_features.h Wed Jan 28 15:46:43 2009 +0100
d61bc4d5 52@@ -112,6 +112,10 @@
12e82c05 53 FEAT_SETHOST_USER,
54 FEAT_SETHOST_AUTO,
12e82c05 55
65b70298 56+ /* SPLIT */
57+ FEAT_SPLIT,
58+ FEAT_SPLIT_AUTO_EXPIRE,
59+
12e82c05 60 /* HEAD_IN_SAND Features */
61 FEAT_HIS_SNOTICES,
65b70298 62 FEAT_HIS_SNOTICES_OPER_ONLY,
d61bc4d5 63@@ -139,6 +143,7 @@
12e82c05 64 FEAT_HIS_STATS_q,
65 FEAT_HIS_STATS_R,
66 FEAT_HIS_STATS_r,
67+ FEAT_HIS_STATS_S,
68 FEAT_HIS_STATS_s,
69 FEAT_HIS_STATS_t,
70 FEAT_HIS_STATS_T,
cb0b5df1 71diff -r 5ec670983aff include/ircd_log.h
72--- a/include/ircd_log.h Wed Jan 28 15:15:55 2009 +0100
73+++ b/include/ircd_log.h Wed Jan 28 15:46:43 2009 +0100
0c63b280 74@@ -55,6 +55,7 @@
75 LS_OPERMODE, /**< Uses of OPMODE, CLEARMODE< etc. */
76 LS_GLINE, /**< Adding, (de-)activating or removing GLINEs. */
77 LS_JUPE, /**< Adding, (de-)activating or removing JUPEs. */
78+ LS_SPLIT, /**< Adding, (de-)activating or removing SPLITs. */
79 LS_WHO, /**< Use of extended WHO privileges. */
80 LS_NETWORK, /**< New server connections. */
81 LS_OPERKILL, /**< Kills by IRC operators. */
cb0b5df1 82diff -r 5ec670983aff include/msg.h
83--- a/include/msg.h Wed Jan 28 15:15:55 2009 +0100
84+++ b/include/msg.h Wed Jan 28 15:46:43 2009 +0100
d61bc4d5 85@@ -336,6 +336,10 @@
12e82c05 86 #define TOK_JUPE "JU"
87 #define CMD_JUPE MSG_JUPE, TOK_JUPE
88
89+#define MSG_SPLIT "SPLIT" /* SPLIT */
90+#define TOK_SPLIT "SP"
91+#define CMD_SPLIT MSG_SPLIT, TOK_SPLIT
92+
93 #define MSG_OPMODE "OPMODE" /* OPMO */
94 #define TOK_OPMODE "OM"
95 #define CMD_OPMODE MSG_OPMODE, TOK_OPMODE
cb0b5df1 96diff -r 5ec670983aff include/numeric.h
97--- a/include/numeric.h Wed Jan 28 15:15:55 2009 +0100
98+++ b/include/numeric.h Wed Jan 28 15:46:43 2009 +0100
d61bc4d5 99@@ -118,6 +118,7 @@
12e82c05 100 #define RPL_STATSALINE 226 /* Hybrid, Undernet */
d61bc4d5 101 #define RPL_STATSWELCOME 227 /* QuakeNet extension */
12e82c05 102 #define RPL_STATSQLINE 228 /* Undernet extension */
103+#define RPL_STATSSPLIT 229 /* QuakeNet extension */
65b70298 104 #define RPL_STATSHEADER 230 /* QuakeNet extension */
12e82c05 105
106 /* RPL_SERVICEINFO 231 unused */
d61bc4d5 107@@ -179,6 +180,8 @@
12e82c05 108 #define RPL_STATSDLINE 275 /* Undernet extension */
109 #define RPL_STATSRLINE 276 /* Undernet extension */
110
111+#define RPL_SPLITLIST 278 /* QuakeNet extension */
112+#define RPL_ENDOFSPLITLIST 279 /* QuakeNet extension */
113 #define RPL_GLIST 280 /* Undernet extension */
114 #define RPL_ENDOFGLIST 281 /* Undernet extension */
115 #define RPL_JUPELIST 282 /* Undernet extension - jupe -Kev */
6b502871 116@@ -449,6 +452,7 @@
d61bc4d5 117
118 #define ERR_NOSUCHWELCOME 509 /* QuakeNet extension */
12e82c05 119
120+#define ERR_NOSUCHSPLIT 510 /* QuakeNet extension */
121 #define ERR_SILELISTFULL 511 /* Undernet extension */
122 /* ERR_NOTIFYFULL 512 aircd */
123 /* ERR_TOOMANYWATCH 512 Numeric List: Dalnet */
cb0b5df1 124diff -r 5ec670983aff include/split.h
12e82c05 125--- /dev/null Thu Jan 01 00:00:00 1970 +0000
cb0b5df1 126+++ b/include/split.h Wed Jan 28 15:46:43 2009 +0100
0c63b280 127@@ -0,0 +1,106 @@
128+#ifndef INCLUDED_split_h
129+#define INCLUDED_split_h
12e82c05 130+/*
131+ * IRC - Internet Relay Chat, include/split.h
132+ * Copyright (C) 1990 Jarkko Oikarinen and
133+ * University of Oulu, Computing Center
134+ * Copyright (C) 2000 Kevin L. Mitchell <klmitch@mit.edu>
135+ *
136+ * This program is free software; you can redistribute it and/or modify
137+ * it under the terms of the GNU General Public License as published by
138+ * the Free Software Foundation; either version 2, or (at your option)
139+ * any later version.
140+ *
141+ * This program is distributed in the hope that it will be useful,
142+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
143+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
144+ * GNU General Public License for more details.
145+ *
146+ * You should have received a copy of the GNU General Public License
147+ * along with this program; if not, write to the Free Software
148+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
149+ */
150+/** @file
151+ * @brief Interface and declarations for split server handling.
152+ * @version $Id: jupe.h 1208 2004-10-03 14:12:35Z entrope $
153+ */
12e82c05 154+#ifndef INCLUDED_sys_types_h
155+#include <sys/types.h>
156+#define INCLUDED_sys_types_h
157+#endif
158+
159+
160+struct Client;
161+struct StatDesc;
162+
65b70298 163+#define SPLIT_MAX_EXPIRE 2419200 /**< Maximum split expiration time (4 weeks). */
12e82c05 164+
165+/* Describes a SPLIT server entry. */
166+struct Split {
167+ struct Split* sp_next; /**< Pointer to next Split. */
168+ struct Split** sp_prev_p; /**< Pointer to previous next pointer. */
169+ char* sp_server; /**< Name of server. */
170+ char* sp_reason; /**< Reason. */
171+ time_t sp_creation; /**< TODO: Creation time. What are we using this for then? */
172+ time_t sp_expire; /**< Expiration time. */
173+ time_t sp_lastmod; /**< Last modification time. */
174+ time_t sp_lifetime; /**< Life time. */
175+ unsigned int sp_flags; /**< Status flags. */
176+};
177+
178+/** Split state flags. */
0c63b280 179+#define SPLIT_ACTIVE 0x01 /**< Split is active. */
180+#define SPLIT_REMOVING 0x02 /**< Split is being force removed (instead of deactivated). */
181+#define SPLIT_BURST 0x04 /**< Split is for a server that is in progress of linking,
182+ and will be destroyed at the end of burst */
12e82c05 183+
12e82c05 184+/* Actions to perform on a SPLIT. */
185+enum SplitAction {
186+ SPLIT_ACTIVATE, /**< SPLIT should be activated. */
187+ SPLIT_DEACTIVATE, /**< SPLIT should be deactivated. */
188+ SPLIT_MODIFY, /**< SPLIT should be modified. */
189+ SPLIT_REMOVE /**< SPLIT should be removed. */
190+};
191+
12e82c05 192+/* Split update flags. */
55404221 193+#define SPLIT_EXPIRE 0x01 /**< Expiration time update. */
194+#define SPLIT_LIFETIME 0x02 /**< Record lifetime update. */
195+#define SPLIT_REASON 0x04 /**< Reason update. */
196+#define SPLIT_CREATION 0x08 /**< TODO: Creation time update - unused. */
197+#define SPLIT_MODIFY 0x10 /**< No state change. */
12e82c05 198+
199+/* mask for Split update flags. */
55404221 200+#define SPLIT_UPDATE (SPLIT_EXPIRE | SPLIT_CREATION | SPLIT_LIFETIME | SPLIT_REASON)
12e82c05 201+
202+/* test whether a split entry is active. */
203+#define SplitIsActive(s) ((s)->sp_flags & SPLIT_ACTIVE)
0c63b280 204+/* test whether a split entry is marked for forced removal. */
205+#define SplitIsRemoving(s) ((s)->sp_flags & SPLIT_REMOVING)
206+/* test whether a split entry is part of a netmerge. */
207+#define SplitIsBurst(s) ((s)->sp_flags & SPLIT_BURST)
12e82c05 208+
209+extern int split_add(struct Client *cptr, struct Client *sptr,
210+ char *server, const char *reason,
211+ time_t creation, time_t expire, time_t lastmod, time_t lifetime,
212+ unsigned int flags);
213+extern int split_modify(struct Client *cptr, struct Client *sptr,
214+ struct Split *split, enum SplitAction action, const char *reason,
215+ time_t creation, time_t expire, time_t lastmod, time_t lifetime,
216+ unsigned int flags);
217+extern int split_remove(struct Client *cptr, struct Client *sptr,
218+ struct Split *split, const char *reason);
219+
220+extern struct Split* split_find(char *server);
221+extern void split_free(struct Split *split);
222+extern int split_expire(struct Split* split);
223+extern void split_burst(struct Client *cptr);
224+extern int split_resend(struct Client *cptr, struct Split *split);
225+extern int split_list(struct Client *sptr, char *server);
0c63b280 226+extern void split_newserver(struct Client *acptr);
227+extern int split_netmerge(struct Client *sptr);
228+extern int split_netbreak(struct Client *victim, const char *reason);
12e82c05 229+extern void split_conf();
230+extern void split_stats(struct Client *sptr, const struct StatDesc *sd, char *param);
231+extern int split_memory_count(size_t *sp_size);
232+
0c63b280 233+#endif /* INCLUDED_split_h */
cb0b5df1 234diff -r 5ec670983aff ircd/Makefile.in
235--- a/ircd/Makefile.in Wed Jan 28 15:15:55 2009 +0100
236+++ b/ircd/Makefile.in Wed Jan 28 15:46:43 2009 +0100
12e82c05 237@@ -173,6 +173,7 @@
238 m_sethost.c \
239 m_settime.c \
240 m_silence.c \
241+ m_split.c \
242 m_squit.c \
243 m_stats.c \
244 m_time.c \
d61bc4d5 245@@ -213,6 +214,7 @@
12e82c05 246 s_stats.c \
247 s_user.c \
248 send.c \
249+ split.c \
250 uping.c \
251 userload.c \
d61bc4d5 252 welcome.c \
253@@ -1054,6 +1056,15 @@
12e82c05 254 ../include/ircd_chattr.h ../include/list.h ../include/msg.h \
255 ../include/numeric.h ../include/numnicks.h ../include/s_user.h \
256 ../include/send.h ../include/struct.h
257+m_split.o: m_split.c ../config.h ../include/client.h ../include/ircd_defs.h \
258+ ../include/dbuf.h ../include/msgq.h ../include/ircd_events.h \
259+ ../config.h ../include/ircd_handler.h ../include/res.h \
260+ ../include/capab.h ../include/split.h ../include/hash.h \
261+ ../include/ircd.h ../include/struct.h ../include/ircd_features.h \
262+ ../include/ircd_log.h ../include/ircd_reply.h ../include/ircd_string.h \
263+ ../include/ircd_chattr.h ../include/match.h ../include/msg.h \
264+ ../include/numeric.h ../include/numnicks.h ../include/s_conf.h \
265+ ../include/client.h ../include/s_misc.h ../include/send.h
266 m_squit.o: m_squit.c ../config.h ../include/client.h \
267 ../include/ircd_defs.h ../include/dbuf.h ../include/msgq.h \
268 ../include/ircd_events.h ../config.h ../include/ircd_handler.h \
d61bc4d5 269@@ -1332,6 +1343,7 @@
12e82c05 270 ../include/msgq.h ../include/numeric.h ../include/numnicks.h \
271 ../include/res.h ../include/s_bsd.h ../include/s_conf.h \
272 ../include/s_user.h ../include/s_stats.h ../include/send.h \
273+ ../include/split.h \
274 ../include/struct.h ../include/sys.h ../include/whowas.h
275 s_err.o: s_err.c ../config.h ../include/numeric.h ../include/ircd_log.h \
276 ../include/s_debug.h ../config.h ../include/ircd_defs.h
d61bc4d5 277@@ -1417,6 +1429,16 @@
12e82c05 278 ../include/msg.h ../include/numnicks.h ../include/parse.h \
279 ../include/s_bsd.h ../include/s_debug.h ../include/s_misc.h \
280 ../include/s_user.h ../include/struct.h ../include/sys.h
281+split.o: split.c ../config.h ../include/split.h ../include/client.h \
282+ ../include/ircd_defs.h ../include/dbuf.h ../include/msgq.h \
283+ ../include/ircd_events.h ../config.h ../include/ircd_handler.h \
284+ ../include/res.h ../include/capab.h ../include/hash.h ../include/ircd.h \
285+ ../include/struct.h ../include/ircd_alloc.h ../include/ircd_features.h \
286+ ../include/ircd_log.h ../include/ircd_reply.h ../include/ircd_string.h \
287+ ../include/ircd_chattr.h ../include/match.h ../include/msg.h \
288+ ../include/numeric.h ../include/numnicks.h ../include/s_bsd.h \
289+ ../include/s_misc.h ../include/send.h ../include/struct.h \
290+ ../include/sys.h
291 uping.o: uping.c ../config.h ../include/uping.h ../include/ircd_defs.h \
292 ../include/ircd_events.h ../config.h ../include/res.h \
293 ../include/client.h ../include/dbuf.h ../include/msgq.h \
cb0b5df1 294diff -r 5ec670983aff ircd/ircd.c
295--- a/ircd/ircd.c Wed Jan 28 15:15:55 2009 +0100
296+++ b/ircd/ircd.c Wed Jan 28 15:46:43 2009 +0100
12e82c05 297@@ -55,6 +55,7 @@
298 #include "s_misc.h"
299 #include "s_stats.h"
300 #include "send.h"
301+#include "split.h"
302 #include "sys.h"
303 #include "uping.h"
304 #include "userload.h"
305@@ -763,6 +764,9 @@
306 Debug((DEBUG_NOTICE, "Server ready..."));
307 log_write(LS_SYSTEM, L_NOTICE, 0, "Server Ready");
308
309+ /* create SPLITs */
310+ split_conf();
311+
312 event_loop();
313
314 return 0;
cb0b5df1 315diff -r 5ec670983aff ircd/ircd_features.c
316--- a/ircd/ircd_features.c Wed Jan 28 15:15:55 2009 +0100
317+++ b/ircd/ircd_features.c Wed Jan 28 15:46:43 2009 +0100
d61bc4d5 318@@ -366,6 +366,10 @@
12e82c05 319 F_B(SETHOST_USER, 0, 0, 0),
320 F_B(SETHOST_AUTO, 0, 0, 0),
12e82c05 321
65b70298 322+ /* SPLIT */
323+ F_B(SPLIT, 0, 1, 0),
324+ F_I(SPLIT_AUTO_EXPIRE, 0, 604800, 0),
325+
12e82c05 326 /* HEAD_IN_SAND Features */
327 F_B(HIS_SNOTICES, 0, 1, 0),
65b70298 328 F_B(HIS_SNOTICES_OPER_ONLY, 0, 1, 0),
d61bc4d5 329@@ -393,6 +397,7 @@
12e82c05 330 F_B(HIS_STATS_q, 0, 1, 0),
331 F_B(HIS_STATS_R, 0, 1, 0),
332 F_B(HIS_STATS_r, 0, 1, 0),
333+ F_B(HIS_STATS_S, 0, 1, 0),
334 F_B(HIS_STATS_s, 0, 1, 0),
335 F_B(HIS_STATS_t, 0, 1, 0),
336 F_B(HIS_STATS_T, 0, 1, 0),
cb0b5df1 337diff -r 5ec670983aff ircd/ircd_log.c
338--- a/ircd/ircd_log.c Wed Jan 28 15:15:55 2009 +0100
339+++ b/ircd/ircd_log.c Wed Jan 28 15:46:43 2009 +0100
0c63b280 340@@ -154,6 +154,7 @@
341 S(OPERMODE, -1, SNO_HACK4),
342 S(GLINE, -1, SNO_GLINE),
343 S(JUPE, -1, SNO_NETWORK),
344+ S(SPLIT, -1, SNO_NETWORK),
345 S(WHO, -1, 0),
346 S(NETWORK, -1, SNO_NETWORK),
347 S(OPERKILL, -1, 0),
cb0b5df1 348diff -r 5ec670983aff ircd/m_endburst.c
349--- a/ircd/m_endburst.c Wed Jan 28 15:15:55 2009 +0100
350+++ b/ircd/m_endburst.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 351@@ -85,6 +85,7 @@
352 #include "client.h"
353 #include "hash.h"
354 #include "ircd.h"
355+#include "ircd_features.h"
356 #include "ircd_log.h"
357 #include "ircd_reply.h"
358 #include "ircd_string.h"
359@@ -92,6 +93,7 @@
360 #include "numeric.h"
361 #include "numnicks.h"
362 #include "send.h"
363+#include "split.h"
364
365 /* #include <assert.h> -- Now using assert in ircd_log.h */
366
367@@ -125,6 +127,7 @@
368 int ms_end_of_burst(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
369 {
370 struct Channel *chan, *next_chan;
371+ int split;
372
373 assert(0 != cptr);
374 assert(0 != sptr);
375@@ -135,6 +138,10 @@
376 dump_map(sptr, "*", 0, report_new_links, 0);
377 sendto_opmask_butone(0, SNO_NETWORK, "Completed net.burst from %C.",
378 sptr);
379+ /* cleanup SPLITs, but only bother ops when FEAT_SPLIT is enabled. */
0c63b280 380+ split = split_netmerge(sptr);
12e82c05 381+ if (feature_bool(FEAT_SPLIT))
382+ sendto_opmask_butone(0, SNO_NETWORK, "Removed %d SPLIT entries.", split);
383 if (MyConnect(sptr))
384 sendcmdto_one(&me, CMD_END_OF_BURST_ACK, sptr, "");
385
cb0b5df1 386diff -r 5ec670983aff ircd/m_reburst.c
387--- a/ircd/m_reburst.c Wed Jan 28 15:15:55 2009 +0100
388+++ b/ircd/m_reburst.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 389@@ -102,6 +102,7 @@
390 #include "ircd_snprintf.h"
391 #include "gline.h"
392 #include "jupe.h"
393+#include "split.h"
394
395 /* #include <assert.h> -- Now using assert in ircd_log.h */
396 #include <stdlib.h>
397@@ -130,6 +131,10 @@
398 case 'J':
399 jupe_burst(sptr);
400 break;
401+ case 's':
402+ case 'S':
403+ split_burst(sptr);
404+ break;
405 default:
406 break;
407 }
cb0b5df1 408diff -r 5ec670983aff ircd/m_server.c
409--- a/ircd/m_server.c Wed Jan 28 15:15:55 2009 +0100
410+++ b/ircd/m_server.c Wed Jan 28 15:46:44 2009 +0100
0c63b280 411@@ -47,6 +47,7 @@
412 #include "s_misc.h"
413 #include "s_serv.h"
414 #include "send.h"
415+#include "split.h"
416 #include "userload.h"
417
418 /* #include <assert.h> -- Now using assert in ircd_log.h */
419@@ -764,6 +765,9 @@
420 sendto_opmask_butone(0, SNO_NETWORK, "Net junction: %s %s",
421 cli_name(sptr), cli_name(acptr));
422 }
423+ /* new server - mark SPLIT for this server */
424+ split_newserver(acptr);
425+
426 /*
427 * Old sendto_serv_but_one() call removed because we now need to send
428 * different names to different servers (domain name matching).
cb0b5df1 429diff -r 5ec670983aff ircd/m_split.c
12e82c05 430--- /dev/null Thu Jan 01 00:00:00 1970 +0000
cb0b5df1 431+++ b/ircd/m_split.c Wed Jan 28 15:46:44 2009 +0100
6b502871 432@@ -0,0 +1,378 @@
12e82c05 433+/*
434+ * IRC - Internet Relay Chat, ircd/m_split.c
435+ * Copyright (C) 1990 Jarkko Oikarinen and
436+ * University of Oulu, Computing Center
437+ * Copyright (C) 2000 Kevin L. Mitchell <klmitch@mit.edu>
438+ *
439+ * See file AUTHORS in IRC package for additional names of
440+ * the programmers.
441+ *
442+ * This program is free software; you can redistribute it and/or modify
443+ * it under the terms of the GNU General Public License as published by
444+ * the Free Software Foundation; either version 1, or (at your option)
445+ * any later version.
446+ *
447+ * This program is distributed in the hope that it will be useful,
448+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
449+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
450+ * GNU General Public License for more details.
451+ *
452+ * You should have received a copy of the GNU General Public License
453+ * along with this program; if not, write to the Free Software
454+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
455+ *
456+ * $Id: m_split.c 1737 2006-12-19 05:20:48Z entrope $
457+ */
458+
459+/*
460+ * m_functions execute protocol messages on this server:
461+ *
462+ * cptr is always NON-NULL, pointing to a *LOCAL* client
463+ * structure (with an open socket connected!). This
464+ * identifies the physical socket where the message
465+ * originated (or which caused the m_function to be
466+ * executed--some m_functions may call others...).
467+ *
468+ * sptr is the source of the message, defined by the
469+ * prefix part of the message if present. If not
470+ * or prefix not found, then sptr==cptr.
471+ *
472+ * (!IsServer(cptr)) => (cptr == sptr), because
473+ * prefixes are taken *only* from servers...
474+ *
475+ * (IsServer(cptr))
476+ * (sptr == cptr) => the message didn't
477+ * have the prefix.
478+ *
479+ * (sptr != cptr && IsServer(sptr) means
480+ * the prefix specified servername. (?)
481+ *
482+ * (sptr != cptr && !IsServer(sptr) means
483+ * that message originated from a remote
484+ * user (not local).
485+ *
486+ * combining
487+ *
488+ * (!IsServer(sptr)) means that, sptr can safely
489+ * taken as defining the target structure of the
490+ * message in this server.
491+ *
492+ * *Always* true (if 'parse' and others are working correct):
493+ *
494+ * 1) sptr->from == cptr (note: cptr->from == cptr)
495+ *
496+ * 2) MyConnect(sptr) <=> sptr == cptr (e.g. sptr
497+ * *cannot* be a local connection, unless it's
498+ * actually cptr!). [MyConnect(x) should probably
499+ * be defined as (x == x->from) --msa ]
500+ *
501+ * parc number of variable parameter strings (if zero,
502+ * parv is allowed to be NULL)
503+ *
504+ * parv a NULL terminated list of parameter pointers,
505+ *
506+ * parv[0], sender (prefix string), if not present
507+ * this points to an empty string.
508+ * parv[1]...parv[parc-1]
509+ * pointers to additional parameters
510+ * parv[parc] == NULL, *always*
511+ *
512+ * note: it is guaranteed that parv[0]..parv[parc-1] are all
513+ * non-NULL pointers.
514+ */
515+#include "config.h"
516+
517+#include "client.h"
518+#include "split.h"
519+#include "hash.h"
520+#include "ircd.h"
521+#include "ircd_features.h"
522+#include "ircd_log.h"
523+#include "ircd_reply.h"
524+#include "ircd_string.h"
525+#include "match.h"
526+#include "msg.h"
527+#include "numeric.h"
528+#include "numnicks.h"
529+#include "s_conf.h"
530+#include "s_debug.h"
531+#include "s_misc.h"
532+#include "send.h"
533+
534+/* #include <assert.h> -- Now using assert in ircd_log.h */
535+#include <stdlib.h>
536+#include <string.h>
537+
538+/*
539+ * ms_split - server message handler
540+ *
541+ * parv[0] = Send prefix
542+ * parv[1] = (+|-)<server name>
543+ * parv[2] = Creation time
544+ * parv[3] = Expiration time
545+ * parv[4] = Last modification time
546+ * parv[5] = Life time
547+ * parv[6] = Comment
548+ *
549+ */
550+int ms_split(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
551+{
552+ struct Client *acptr = 0;
553+ struct Split *asplit = 0;
554+ unsigned int flags = 0;
555+ enum SplitAction action = SPLIT_MODIFY;
556+ time_t creation = 0, expire = 0, lastmod = 0, lifetime = 0;
6b502871 557+ char *server;
12e82c05 558+ const char *reason = "No reason";
559+
55404221 560+ /* reason is optional - for lazy services, we always send it */
561+ if (parc < 5)
12e82c05 562+ return need_more_params(sptr, "SPLIT");
563+
6b502871 564+ /* server param */
565+ server = parv[1];
566+
12e82c05 567+ if (*server == '!') {
568+ server++;
569+ action = SPLIT_REMOVE; /* removing entry */
570+ }
571+
572+ switch (*server) { /* handle + and - */
573+ case '+': /* activate the split entry */
574+ if (action != SPLIT_REMOVE)
575+ action = SPLIT_ACTIVATE;
576+ server++;
577+ break;
578+
579+ case '-': /* deactivate the entry */
580+ if (action != SPLIT_REMOVE)
581+ action = SPLIT_DEACTIVATE;
582+ server++;
583+ break;
584+ }
585+
586+ /* Next, try to find the split entry... */
587+ asplit = split_find(server);
588+
589+ /* We now have all the pieces to tell us what we've got; let's put
590+ * it all together and convert the rest of the arguments.
591+ */
592+
593+ /* can't modify a split entry that doesn't exist, so remap to activate */
594+ if (!asplit && action == SPLIT_MODIFY)
595+ action = SPLIT_ACTIVATE;
596+
55404221 597+ /* set all flags for modification */
598+ if (asplit && action != SPLIT_REMOVE)
599+ flags = SPLIT_UPDATE;
600+
601+ /* set the things we need */
12e82c05 602+ creation = atoi(parv[2]);
603+ expire = atoi(parv[3]);
604+ lastmod = atoi(parv[4]);
605+ lifetime = atoi(parv[5]);
55404221 606+ /* got a reason too */
cb0b5df1 607+ if (parc > 5)
55404221 608+ reason = parv[parc - 1];
12e82c05 609+
610+ Debug((DEBUG_DEBUG, "I have a SPLIT I am acting upon now; "
611+ "server %s, action %s, creation %Tu, expire %Tu, "
612+ "lastmod %Tu, lifetime %Tu, reason: %s; split %s!", server,
613+ action == SPLIT_REMOVE ? "!-" :
614+ (action == SPLIT_ACTIVATE ? "+" :
615+ (action == SPLIT_DEACTIVATE ? "-" : "(MODIFY)")),
616+ creation, expire, lastmod, lifetime, reason,
617+ asplit ? "EXISTS" : "does not exist"));
618+
619+ /* OK, at this point, we have converted all available parameters.
620+ * Let's actually do the action!
621+ */
622+ if (asplit) {
623+ if (action == SPLIT_REMOVE)
624+ return split_remove(cptr, sptr, asplit, reason);
625+ return split_modify(cptr, sptr, asplit, action, reason, creation, expire,
626+ lastmod, lifetime, flags);
627+ }
628+
55404221 629+ /* desynch or cross - we do not have it so we cannot remove it */
630+ /* TODO: manually propagate the message ?
631+ * ..but if we do not have it, can our downlinks have it?
632+ * they could if my oper force removed it earlier, and then my downlinks linked
633+ * no, then they would have bursted it again to us.. argh
634+ */
635+ if (!asplit && action == SPLIT_REMOVE)
636+ return 0;
637+
12e82c05 638+ assert(action != SPLIT_MODIFY);
55404221 639+ assert(action != SPLIT_REMOVE);
12e82c05 640+
55404221 641+ /* only give one flag when needed - SPLIT_ACTIVE */
12e82c05 642+ return split_add(cptr, sptr, server, reason, creation, expire, lastmod, lifetime,
55404221 643+ (flags & SPLIT_ACTIVATE) ? SPLIT_ACTIVE : 0);
12e82c05 644+}
645+
646+
647+/*
648+ * mo_split - oper message handler
649+ *
650+ * Local listing: 1 or 2 params
651+ * parv[0] = Send prefix
652+ * parv[1] = [Server or mask to match]
653+ *
654+ * Add or modify entry: 3, 4 or 5 params (3 is * then)
655+ * parv[0] = Send prefix
656+ * parv[1] = [+|-]<server name>
657+ * parv[2] = [Expiration offset] (required for new)
658+ * parv[3] = [Comment] (required for new)
659+ *
660+ */
661+int mo_split(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
662+{
663+ struct Client *acptr = 0;
664+ struct Split *asplit = 0;
665+ unsigned int flags = 0;
666+ enum SplitAction action = SPLIT_MODIFY;
0c63b280 667+ time_t expire = 0, lastmod = TStime(), creation = TStime();
12e82c05 668+ char *server = parv[1], *end;
669+ const char *reason = NULL;
670+
55404221 671+ /* listing */
12e82c05 672+ if (parc < 2) {
673+ if (!HasPriv(sptr, PRIV_ROUTEINFO))
674+ return send_reply(sptr, ERR_NOPRIVILEGES);
675+ else
676+ return split_list(sptr, 0);
677+ }
678+
55404221 679+ /* force removal */
12e82c05 680+ if (*server == '!') {
681+ server++;
55404221 682+ action = SPLIT_REMOVE;
12e82c05 683+ }
684+
685+ switch (*server) { /* handle + and - */
686+ case '+': /* activate the split entry */
687+ if (action != SPLIT_REMOVE)
688+ action = SPLIT_ACTIVATE;
689+ server++;
690+ break;
691+
692+ case '-': /* deactivate the split entry */
693+ if (action != SPLIT_REMOVE)
694+ action = SPLIT_DEACTIVATE;
695+ server++;
696+ break;
697+ }
12e82c05 698+
55404221 699+ /* user wants listing */
700+ if (action == SPLIT_MODIFY && parc == 2)
701+ return split_list(sptr, server);
702+
703+ /* if we have an expire, this must be it */
704+ if (parc > 2)
705+ expire = strtol(parv[2], &end, 10); /* and the expiration */
706+
707+ /* got reason and expire */
708+ if (parc > 3) {
709+ flags |= SPLIT_EXPIRE;
710+ flags |= SPLIT_REASON;
711+ }
712+ /* got reason or expire */
713+ else if (parc > 2) {
714+ if (*end != '\0')
12e82c05 715+ flags |= SPLIT_REASON;
55404221 716+ else
717+ flags |= SPLIT_EXPIRE;
718+ }
719+
720+ /* set reason */
721+ if (flags & SPLIT_REASON) {
12e82c05 722+ reason = parv[parc - 1];
55404221 723+ if (reason[0] == '\0') /* reason is empty */
724+ flags &= ~SPLIT_REASON;
12e82c05 725+ }
12e82c05 726+
55404221 727+ /* bad expire */
728+ if (flags & SPLIT_EXPIRE && *end != '\0')
729+ return send_reply(sptr, SND_EXPLICIT | ERR_BADEXPIRE, "%s :Bad expire time", parv[2]);
730+
731+ /* require a reason for forced removal */
732+ if (!(flags & SPLIT_REASON) && action == SPLIT_REMOVE)
733+ return need_more_params(sptr, "SPLIT");
734+
735+ /* we got expire */
736+ if (flags & SPLIT_EXPIRE) {
737+ if (expire == 0) /* dummy value 0 given, no change to expire */
738+ flags &= ~SPLIT_EXPIRE;
739+ else /* turn expire into timestamp */
740+ expire += TStime();
741+ }
742+
743+ /* check feature and privs */
12e82c05 744+ if (!feature_bool(FEAT_SPLIT)) /* TODO: but allow force removal? */
745+ return send_reply(sptr, ERR_DISABLED, "SPLIT");
746+ else if (!HasPriv(sptr, PRIV_SERVERINFO)) /* TODO: create PRIV_SPLIT - need help there */
747+ return send_reply(sptr, ERR_NOPRIVILEGES);
748+
55404221 749+ /* find the split */
12e82c05 750+ asplit = split_find(server);
751+
752+ /* We now have all the pieces to tell us what we've got; let's put
753+ * it all together and convert the rest of the arguments.
754+ */
755+
55404221 756+ /* SPLIT not found and thus we cannot:
757+ * remove SPLIT we do not have
758+ * add new SPLIT without expire and reason
12e82c05 759+ */
760+ if (!asplit &&
761+ ((action == SPLIT_REMOVE) ||
762+ (action == SPLIT_ACTIVATE && !reason) ||
763+ (action == SPLIT_DEACTIVATE && !reason) ||
764+ (action == SPLIT_MODIFY && !reason)))
765+ return send_reply(sptr, ERR_NOSUCHSPLIT, server);
766+
767+ /* can't modify a split entry that doesn't exist, so remap to activate */
768+ if (!asplit && action == SPLIT_MODIFY)
769+ action = SPLIT_ACTIVATE;
770+
771+ Debug((DEBUG_DEBUG, "I have a SPLIT I am acting upon now; "
772+ "server %s, action %s, expire %Tu, "
773+ "reason: %s; split %s! (fields present: %s %s)", server,
774+ action == SPLIT_REMOVE ? "!-" :
775+ (action == SPLIT_ACTIVATE ? "+" :
776+ (action == SPLIT_DEACTIVATE ? "-" : "(MODIFY)")),
777+ expire, reason, asplit ? "EXISTS" : "does not exist",
778+ flags & SPLIT_EXPIRE ? "expire" : "",
779+ flags & SPLIT_REASON ? "reason" : ""));
780+
781+ if (asplit) { /* modifying an existing SPLIT */
782+ if (action == SPLIT_REMOVE)
783+ return split_remove(cptr, sptr, asplit, reason);
784+ return split_modify(cptr, sptr, asplit, action, reason,
785+ asplit->sp_creation, expire, lastmod, asplit->sp_lifetime, flags);
786+ }
787+
788+ assert(action != SPLIT_MODIFY);
789+ assert(action != SPLIT_REMOVE);
790+
791+ /* create a new SPLIT */
792+ return split_add(cptr, sptr, server, reason,
793+ creation, expire, lastmod, expire,
794+ flags | ((action == SPLIT_ACTIVATE) ? SPLIT_ACTIVE : 0));
795+}
796+
797+/*
798+ * m_split - user message handler
799+ *
800+ * parv[0] = Send prefix
12e82c05 801+ * parv[1] = [<server name>]
802+ *
803+ */
804+int m_split(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
805+{
806+ if (parc < 2)
807+ return split_list(sptr, 0);
808+
809+ return split_list(sptr, parv[1]);
810+}
cb0b5df1 811diff -r 5ec670983aff ircd/parse.c
812--- a/ircd/parse.c Wed Jan 28 15:15:55 2009 +0100
813+++ b/ircd/parse.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 814@@ -505,6 +505,13 @@
a87bc2c2 815 /* UNREG, CLIENT, SERVER, OPER, SERVICE, HELP */
816 { m_unregistered, m_not_oper, ms_jupe, mo_jupe, m_ignore, mh_jupe }
817 },
12e82c05 818+ {
819+ MSG_SPLIT,
820+ TOK_SPLIT,
821+ 0, MAXPARA, MFLG_SLOW, 0, NULL,
a87bc2c2 822+ /* UNREG, CLIENT, SERVER, OPER, SERVICE, HELP */
823+ { m_unregistered, m_not_oper, ms_split, mo_split, m_ignore, mh_nohelp }
824+ },
12e82c05 825 {
826 MSG_OPMODE,
a87bc2c2 827 TOK_OPMODE,
cb0b5df1 828diff -r 5ec670983aff ircd/s_conf.c
829--- a/ircd/s_conf.c Wed Jan 28 15:15:55 2009 +0100
830+++ b/ircd/s_conf.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 831@@ -53,6 +53,7 @@
832 #include "s_debug.h"
833 #include "s_misc.h"
834 #include "send.h"
835+#include "split.h"
836 #include "struct.h"
837 #include "sys.h"
838
839@@ -1005,6 +1006,7 @@
840 }
841
842 attach_conf_uworld(&me);
843+ split_conf();
844
845 return ret;
846 }
cb0b5df1 847diff -r 5ec670983aff ircd/s_debug.c
848--- a/ircd/s_debug.c Wed Jan 28 15:15:55 2009 +0100
849+++ b/ircd/s_debug.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 850@@ -48,6 +48,7 @@
851 #include "s_user.h"
852 #include "s_stats.h"
853 #include "send.h"
854+#include "split.h"
855 #include "struct.h"
856 #include "sys.h"
cb0b5df1 857 #include "whowas.h"
858@@ -231,7 +232,8 @@
859 aw = 0, /* aways set */
12e82c05 860 wwa = 0, /* whowas aways */
861 gl = 0, /* glines */
cb0b5df1 862- ju = 0; /* jupes */
863+ ju = 0, /* jupes */
864+ sp = 0; /* splits */
12e82c05 865
866 size_t chm = 0, /* memory used by channels */
cb0b5df1 867 chbm = 0, /* memory used by channel bans */
868@@ -244,6 +246,7 @@
12e82c05 869 wwm = 0, /* whowas array memory used */
870 glm = 0, /* memory used by glines */
871 jum = 0, /* memory used by jupes */
cb0b5df1 872+ spm = 0, /* memory used by splits */
12e82c05 873 com = 0, /* memory used by conf lines */
874 dbufs_allocated = 0, /* memory used by dbufs */
cb0b5df1 875 dbufs_used = 0, /* memory used by dbufs */
876@@ -348,8 +351,9 @@
12e82c05 877
878 gl = gline_memory_count(&glm);
879 ju = jupe_memory_count(&jum);
880+ sp = split_memory_count(&spm);
881 send_reply(cptr, SND_EXPLICIT | RPL_STATSDEBUG,
cb0b5df1 882- ":Glines %d(%zu) Jupes %d(%zu)", gl, glm, ju, jum);
883+ ":Glines %d(%zu) Jupes %d(%zu) Splits %d(%zu)", gl, glm, ju, jum, sp, spm);
12e82c05 884
885 send_reply(cptr, SND_EXPLICIT | RPL_STATSDEBUG,
886 ":Hash: client %d(%zu), chan is the same", HASHSIZE,
cb0b5df1 887diff -r 5ec670983aff ircd/s_err.c
888--- a/ircd/s_err.c Wed Jan 28 15:15:55 2009 +0100
889+++ b/ircd/s_err.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 890@@ -490,7 +490,7 @@
891 /* 228 */
892 { RPL_STATSQLINE, "Q %s :%s", "228" },
893 /* 229 */
894- { 0 },
895+ { RPL_STATSSPLIT, "S %s %d %d %d %d %c :%s", "229" },
896 /* 230 */
65b70298 897 { RPL_STATSHEADER, 0, "230" },
12e82c05 898 /* 231 */
899@@ -588,9 +588,9 @@
900 /* 277 */
901 { 0 },
902 /* 278 */
903- { 0 },
904+ { RPL_SPLITLIST, "%s %Tu %Tu %Tu %Tu %c :%s", "278" },
905 /* 279 */
906- { 0 },
907+ { RPL_ENDOFSPLITLIST, ":End of Split List", "279" },
908 /* 280 */
65b70298 909 { RPL_GLIST, "%s%s%s%s%s %Tu %Tu %Tu %s %s%s%c :%s", "280" },
12e82c05 910 /* 281 */
911@@ -1052,7 +1052,7 @@
912 /* 509 */
d61bc4d5 913 { ERR_NOSUCHWELCOME, "%s :No such welcome", "509" },
12e82c05 914 /* 510 */
915- { 0 },
916+ { ERR_NOSUCHSPLIT, "%s :No such split", "510" },
917 /* 511 */
918 { ERR_SILELISTFULL, "%s :Your silence list is full", "511" },
919 /* 512 */
cb0b5df1 920diff -r 5ec670983aff ircd/s_misc.c
921--- a/ircd/s_misc.c Wed Jan 28 15:15:55 2009 +0100
922+++ b/ircd/s_misc.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 923@@ -53,6 +53,7 @@
924 #include "s_stats.h"
925 #include "s_user.h"
926 #include "send.h"
927+#include "split.h"
928 #include "struct.h"
929 #include "sys.h"
930 #include "uping.h"
931@@ -391,6 +392,7 @@
932 time_t on_for;
933
934 char comment1[HOSTLEN + HOSTLEN + 2];
935+ char splitreason[BUFSIZE];
936 assert(killer);
937 if (MyConnect(victim))
938 {
7efbed3b 939@@ -497,6 +499,15 @@
12e82c05 940 sendto_opmask_butone(0, SNO_NETWORK, "Net break: %C %C (%s)",
941 cli_serv(victim)->up, victim, comment);
942 dump_map(victim, "*", 0, report_lost_links, 0);
943+ if (feature_bool(FEAT_SPLIT)) {
7efbed3b 944+ /* TODO: remote & local SQUIT ? servername oper is on? */
12e82c05 945+ ircd_snprintf(0, splitreason, sizeof(splitreason),
946+ "Net break: %C %C (%s%s%s%s)", cli_serv(victim)->up, victim,
947+ IsUser(killer) ? "SQUIT by " : "", IsUser(killer) ? cli_name(killer) : "",
948+ IsUser(killer) ? ": " : "", comment);
949+ sendto_opmask_butone(0, SNO_NETWORK, "Created %d SPLIT entries.",
0c63b280 950+ split_netbreak(victim, splitreason));
12e82c05 951+ }
952 }
953
954 /*
cb0b5df1 955diff -r 5ec670983aff ircd/s_serv.c
956--- a/ircd/s_serv.c Wed Jan 28 15:15:55 2009 +0100
957+++ b/ircd/s_serv.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 958@@ -54,6 +54,7 @@
959 #include "s_misc.h"
960 #include "s_user.h"
961 #include "send.h"
962+#include "split.h"
963 #include "struct.h"
964 #include "sys.h"
965 #include "userload.h"
d61bc4d5 966@@ -197,6 +198,7 @@
12e82c05 967 */
968 gline_burst(cptr);
969 jupe_burst(cptr);
970+ split_burst(cptr);
d61bc4d5 971 welcome_burst(cptr);
12e82c05 972
973 /*
cb0b5df1 974diff -r 5ec670983aff ircd/s_stats.c
975--- a/ircd/s_stats.c Wed Jan 28 15:15:55 2009 +0100
976+++ b/ircd/s_stats.c Wed Jan 28 15:46:44 2009 +0100
12e82c05 977@@ -52,6 +52,7 @@
978 #include "s_stats.h"
979 #include "s_user.h"
980 #include "send.h"
981+#include "split.h"
982 #include "struct.h"
983 #include "userload.h"
d61bc4d5 984 #include "welcome.h"
985@@ -669,7 +670,10 @@
12e82c05 986 send_usage, 0,
987 "System resource usage (Debug only)." },
988 #endif
989- { 's', "spoofhosts", (STAT_FLAG_OPERFEAT | STAT_FLAG_VARPARAM), FEAT_HIS_STATS_s,
990+ { 'S', "splits", (STAT_FLAG_OPERFEAT | STAT_FLAG_VARPARAM | STAT_FLAG_CASESENS), FEAT_HIS_STATS_S,
991+ split_stats, 0,
992+ "Server SPLITs information."},
993+ { 's', "spoofhosts", (STAT_FLAG_OPERFEAT | STAT_FLAG_VARPARAM | STAT_FLAG_CASESENS), FEAT_HIS_STATS_s,
994 stats_sline, 0,
995 "Spoofed hosts information." },
996 { 'T', "motds", (STAT_FLAG_OPERFEAT | STAT_FLAG_CASESENS), FEAT_HIS_STATS_T,
cb0b5df1 997diff -r 5ec670983aff ircd/split.c
12e82c05 998--- /dev/null Thu Jan 01 00:00:00 1970 +0000
cb0b5df1 999+++ b/ircd/split.c Wed Jan 28 15:46:44 2009 +0100
1000@@ -0,0 +1,788 @@
12e82c05 1001+/*
1002+ * IRC - Internet Relay Chat, ircd/split.c
1003+ * Copyright (C) 1990 Jarkko Oikarinen and
1004+ * University of Oulu, Finland
1005+ * Copyright (C) 2000 Kevin L. Mitchell <klmitch@mit.edu>
1006+ *
1007+ * This program is free software; you can redistribute it and/or modify
1008+ * it under the terms of the GNU General Public License as published by
1009+ * the Free Software Foundation; either version 1, or (at your option)
1010+ * any later version.
1011+ *
1012+ * This program is distributed in the hope that it will be useful,
1013+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
1014+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1015+ * GNU General Public License for more details.
1016+ *
1017+ * You should have received a copy of the GNU General Public License
1018+ * along with this program; if not, write to the Free Software
1019+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
1020+ */
1021+/** @file
1022+ * @brief Implementation of split server handling functions.
1023+ * @version $Id: split.c 1633 2006-03-25 03:46:56Z entrope $
1024+ */
1025+#include "config.h"
1026+
1027+#include "split.h"
1028+#include "client.h"
1029+#include "hash.h"
1030+#include "ircd.h"
1031+#include "ircd_alloc.h"
1032+#include "ircd_features.h"
1033+#include "ircd_log.h"
1034+#include "ircd_reply.h"
1035+#include "ircd_string.h"
1036+#include "list.h"
1037+#include "match.h"
1038+#include "msg.h"
1039+#include "numeric.h"
1040+#include "numnicks.h"
1041+#include "s_conf.h"
1042+#include "s_bsd.h"
1043+#include "s_debug.h"
1044+#include "s_misc.h"
1045+#include "send.h"
1046+#include "struct.h"
1047+#include "sys.h" /* FALSE bleah */
1048+
1049+/* #include <assert.h> -- Now using assert in ircd_log.h */
1050+#include <string.h>
1051+
1052+/** List of split entries. */
1053+static struct Split *GlobalSplitList = 0;
1054+
1055+/** Allocate a new Split entry with the given parameters.
1056+ * @param[in] server Server name for split entry.
1057+ * @param[in] reason Reason for the split entry.
1058+ * @param[in] creation Creation time for split entry.
1059+ * @param[in] expire Expiration time for split entry.
1060+ * @param[in] lastmod Last modification time for split entry.
1061+ * @param[in] lifetime Life time for split entry.
1062+ * @param[in] flags Flags to set for the split entry.
1063+ */
1064+static struct Split *
1065+split_make(char *server, const char *reason,
1066+ time_t creation, time_t expire, time_t lastmod, time_t lifetime,
1067+ unsigned int flags)
1068+{
1069+ struct Split *asplit;
1070+
1071+ asplit = (struct Split*) MyMalloc(sizeof(struct Split)); /* alloc memory */
1072+ assert(0 != asplit);
1073+
1074+ memset(asplit, 0, sizeof(*asplit));
1075+ /* TODO: some limit for servername length? HOSTLEN? */
1076+ DupString(asplit->sp_server, server); /* copy vital information */
0c63b280 1077+ /* TODO: some limit for reason length? QUITLEN TOPICLEN? */
65b70298 1078+ /* TODO: prefix the reason here with the opername/ID? */
12e82c05 1079+ DupString(asplit->sp_reason, reason);
1080+ /* TODO: what we use creation for and do we need it? */
1081+ asplit->sp_creation = creation;
1082+ asplit->sp_expire = expire;
1083+ /* TODO: are we using nettime etc.? CurrentTime is used. */
65b70298 1084+ /* TODO: TStime() is CurrentTime + TSoffset */
12e82c05 1085+ asplit->sp_lastmod = lastmod;
1086+ asplit->sp_lifetime = lifetime;
12e82c05 1087+ asplit->sp_flags = flags & SPLIT_ACTIVE; /* set split flags */
1088+
1089+ asplit->sp_next = GlobalSplitList; /* link it into the list */
1090+ asplit->sp_prev_p = &GlobalSplitList;
1091+ if (GlobalSplitList)
1092+ GlobalSplitList->sp_prev_p = &asplit->sp_next;
1093+ GlobalSplitList = asplit;
1094+
1095+ return asplit;
1096+}
1097+
1098+/** Forward a split entry to another server.
1099+ * @param[in] cptr Local client that sent us the split entry.
1100+ * @param[in] sptr Originator of the split entry.
1101+ * @param[in] split Split entry to forward.
1102+ * @param[in] reason Reason to send upstream (used by split_remove)
1103+ */
1104+static void
1105+split_propagate(struct Client *cptr, struct Client *sptr,
1106+ struct Split *split, const char *reason)
1107+{
1108+ sendcmdto_serv_butone(sptr, CMD_SPLIT, cptr, "%s%c%s %Tu %Tu %Tu %Tu :%s",
1109+ SplitIsRemoving(split) ? "!" : "",
1110+ SplitIsActive(split) && !SplitIsRemoving(split) ? '+' : '-', /* always !- not !+ */
1111+ split->sp_server, split->sp_creation, split->sp_expire,
1112+ split->sp_lastmod, split->sp_lifetime,
65b70298 1113+ /* TODO: does reason ? reason : split->sp_reason work too? */
12e82c05 1114+ reason != NULL ? reason : split->sp_reason);
1115+}
1116+
1117+/** Add a new server split entry.
1118+ * @param[in] cptr Local client that sent us the split entry.
1119+ * @param[in] sptr Originator of the split entry.
1120+ * @param[in] server Server name to split entry.
1121+ * @param[in] reason Reason for the split entry.
1122+ * @param[in] expire Expiration timestamp.
1123+ * @param[in] lastmod Last modification timestamp.
1124+ * @param[in] flags Flags to set on the split entry.
1125+ * @return Zero
1126+ */
1127+int
1128+split_add(struct Client *cptr, struct Client *sptr, char *server, const char *reason,
1129+ time_t creation, time_t expire, time_t lastmod, time_t lifetime,
1130+ unsigned int flags)
1131+{
1132+
12e82c05 1133+ struct Split *asplit;
1134+ struct Client *acptr;
55404221 1135+ char noreason[BUFSIZE];
12e82c05 1136+
55404221 1137+ assert(0 != cptr);
1138+ assert(0 != sptr);
12e82c05 1139+ assert(0 != server);
1140+ assert(0 != reason);
12e82c05 1141+
1142+ Debug((DEBUG_DEBUG, "split_add(\"%s\", \"%s\", \"%s\", \"%s\", %Tu, %Tu, %Tu, "
1143+ "%Tu, 0x%04x)", cli_name(cptr), cli_name(sptr), server, reason,
1144+ creation, expire, lastmod, lifetime, flags));
1145+
1146+ /* not adding SPLIT for server that is linked
1147+ * if sptr is my user throw error
1148+ * otherwise ignore - SERVER and SPLIT messages can cross,
1149+ * or a server is bursting and it will see our end and destroy the SPLITs
1150+ */
1151+ if ((acptr = FindServer(server))) {
1152+ if (MyUser(sptr))
1153+ sendcmdto_one(&me, CMD_NOTICE, sptr,
1154+ "%C :Cannot add SPLIT %s - server is linked.", sptr, server);
1155+ return 0;
1156+ }
1157+
1158+ /*
1159+ * You cannot set a negative (or zero) duration, nor can you set an
1160+ * duration greater than SPLIT_MAX_EXPIRE.
1161+ */
0c63b280 1162+ if (expire - TStime() <= 0 || expire - TStime() > SPLIT_MAX_EXPIRE) {
55404221 1163+ /* TODO: cptr or sptr here? */
1164+ if (MyUser(sptr))
0c63b280 1165+ return send_reply(cptr, ERR_BADEXPIRE, expire - TStime());
1166+ if (lifetime <= TStime()) /* no point going further */
12e82c05 1167+ /* CHECK: sptr may have the wrong idea about the nettime?
1168+ * or we could be wrong?
12e82c05 1169+ * perhaps raise some sort of warning to ops
65b70298 1170+ */
12e82c05 1171+ return 0;
1172+ }
1173+
55404221 1174+ /* empty reason */
1175+ if (reason[0] == '\0')
1176+ ircd_snprintf(0, noreason, sizeof(noreason), "%C gave no reason.", sptr);
1177+
12e82c05 1178+ /* inform ops and log it */
1179+ sendto_opmask_butone(0, SNO_NETWORK, "%s adding%sSPLIT for %s, expiring at %Tu: %s",
1180+ (feature_bool(FEAT_HIS_SNOTICES) || IsServer(sptr)) ?
cb0b5df1 1181+ get_client_name_and_opername(sptr) : cli_name((cli_user(sptr))->server),
12e82c05 1182+ !(flags & SPLIT_ACTIVE) ? " deactivated " : " ",
55404221 1183+ server, expire, reason[0] == '\0' ? noreason : reason);
12e82c05 1184+
0c63b280 1185+ log_write(LS_SPLIT, L_INFO, LOG_NOSNOTICE,
12e82c05 1186+ "%#C adding%sSPLIT for %s, expiring at %Tu: %s",
1187+ sptr, !(flags & SPLIT_ACTIVE) ? " deactivated " : " ",
55404221 1188+ server, expire, reason[0] == '\0' ? noreason : reason);
12e82c05 1189+
1190+ /* make the split entry */
55404221 1191+ asplit = split_make(server, reason[0] == '\0' ? noreason : reason,
1192+ creation, expire, lastmod, lifetime, flags);
12e82c05 1193+
1194+ assert(asplit);
1195+
1196+ /* and propagate it */
1197+ split_propagate(cptr, sptr, asplit, NULL);
1198+
1199+ return 0;
1200+}
1201+
1202+
1203+/** Modify a split entry.
1204+ * @param[in] cptr Client that sent us the split modification.
1205+ * @param[in] sptr Client that originated the split modification.
1206+ * @param[in] split Split entry being modified.
1207+ * @param[in] action Resultant status of the G-line.
1208+ * @param[in] reason Reason.
1209+ * @param[in] expire Expiration time.
1210+ * @param[in] lastmod Last modification time.
1211+ * @param[in] lifetime Lifetime.
1212+ * @param[in] flags Bitwise combination of SPLIT_* flags.
1213+ * @return Zero.
1214+ */
1215+int
1216+split_modify(struct Client *cptr, struct Client *sptr, struct Split *split,
1217+ enum SplitAction action, const char *reason,
1218+ time_t creation, time_t expire, time_t lastmod, time_t lifetime,
1219+ unsigned int flags)
1220+{
1221+ struct Client* acptr;
1222+ char buf[BUFSIZE];
1223+ int pos = 0;
1224+
55404221 1225+ assert(0 != cptr);
1226+ assert(0 != sptr);
1227+ assert(0 != split);
12e82c05 1228+
55404221 1229+ Debug((DEBUG_DEBUG, "split_modify(cptr \"%s\", sptr \"%s\", split \"%s\", reason \"%s\","
1230+ " action %d, creation %Tu, expire %Tu, lastmod %Tu, lifetime %Tu, flags 0x%04x)",
1231+ cli_name(cptr), cli_name(sptr), split->sp_server, reason, action,
12e82c05 1232+ creation, expire, lastmod, lifetime, flags));
1233+
1234+ /* not modifying SPLIT for server that is linked
1235+ * if sptr is my user throw error
1236+ * otherwise ignore - SERVER and SPLIT messages can cross.
1237+ *
1238+ * note: we cleanup SPLIT entries at end of burst,
1239+ * and not when a server is introduced.
1240+ * so between net junction and end of burst,
1241+ * we can get SPLITs for a linked server.
65b70298 1242+ * also more than one server can be bursting at a time.
12e82c05 1243+ */
12e82c05 1244+ if ((acptr = FindServer(split->sp_server))) {
1245+ if (MyUser(sptr))
1246+ sendcmdto_one(&me, CMD_NOTICE, sptr,
1247+ "%C :Cannot modify SPLIT %s - server is linked.", sptr, split->sp_server);
0c63b280 1248+ /* SPLIT is not for a server part of a netmerge */
1249+ if (!SplitIsBurst(split))
1250+ split_free(split); /* and free it */
12e82c05 1251+ return 0;
1252+ }
55404221 1253+
1254+ /* check lastmod */
0c63b280 1255+ if (split->sp_lastmod > lastmod) { /* we have a more recent version */
55404221 1256+ if (IsBurstOrBurstAck(cptr)) /* middle of a burst, it will resync on its own */
1257+ return 0;
12e82c05 1258+ return split_resend(cptr, split); /* resync the server */
55404221 1259+ } else if (split->sp_lastmod == lastmod) /* same version */
1260+ return 0;
12e82c05 1261+
55404221 1262+ /* there is a change of some sort */
1263+ /* check out the expiration time */
1264+ if (flags & SPLIT_EXPIRE) {
1265+ if (expire != split->sp_expire) {
1266+ /* bad expiration time */
1267+ if (MyUser(sptr) && (expire < TStime() || expire - TStime() > SPLIT_MAX_EXPIRE))
0c63b280 1268+ return send_reply(sptr, ERR_BADEXPIRE, expire - TStime());
55404221 1269+ /* expire timestamp is in the past, so remap to deactivate */
1270+ if (expire < TStime() && action == SPLIT_ACTIVATE)
1271+ action = SPLIT_DEACTIVATE;
1272+ } else
1273+ flags &= ~SPLIT_EXPIRE;
1274+ }
0c63b280 1275+
12e82c05 1276+ /* Next, check out lifetime */
1277+ if (!(flags & SPLIT_LIFETIME) || !lifetime)
1278+ lifetime = split->sp_lifetime; /* use Split lifetime */
1279+
1280+ lifetime = IRCD_MAX(lifetime, expire); /* set lifetime to the max */
1281+
0c63b280 1282+ /* now that we have the correct lifetime, check it */
1283+ if (lifetime - TStime() <= 0) /* no point in going further */
1284+ return 0;
1285+
12e82c05 1286+ /* OK, let's see which is greater... */
1287+ if (lifetime > split->sp_lifetime)
1288+ flags |= SPLIT_LIFETIME; /* have to update lifetime */
1289+ else {
1290+ flags &= ~SPLIT_LIFETIME; /* no change to lifetime */
1291+ lifetime = 0;
1292+ }
55404221 1293+
12e82c05 1294+ /* Finally, let's see if the reason needs to be updated */
55404221 1295+ if ((flags & SPLIT_REASON) && (reason[0] == '\0' || !ircd_strcmp(split->sp_reason, reason)))
12e82c05 1296+ flags &= ~SPLIT_REASON; /* no changes to the reason */
1297+
1298+ /* OK, now let's take a look at the action... */
1299+ if ((action == SPLIT_ACTIVATE && SplitIsActive(split)) ||
1300+ (action == SPLIT_DEACTIVATE && !SplitIsActive(split)) ||
1301+ /* can't activate an expired split entry */
0c63b280 1302+ (IRCD_MAX(split->sp_expire, expire) <= TStime()))
12e82c05 1303+ action = SPLIT_MODIFY; /* no activity state modifications */
1304+
1305+ Debug((DEBUG_DEBUG, "About to perform changes; flags 0x%04x, action %s",
1306+ flags, action == SPLIT_ACTIVATE ? "SPLIT_ACTIVATE" :
1307+ (action == SPLIT_DEACTIVATE ? "SPLIT_DEACTIVATE" :
1308+ (action == SPLIT_MODIFY ? "SPLIT_MODIFY" : "<UNKNOWN>"))));
1309+
1310+ /* If there are no changes to perform, do no changes */
55404221 1311+ if (!(flags & SPLIT_UPDATE) && action == SPLIT_MODIFY) {
1312+ if (MyUser(sptr))
1313+ sendcmdto_one(&me, CMD_NOTICE, sptr,
1314+ "%C :Cannot modify SPLIT %s - nothing to change.", sptr, split->sp_server);
12e82c05 1315+ return 0;
55404221 1316+ }
12e82c05 1317+
1318+ /* Start by updating lastmod, if indicated... */
1319+ split->sp_lastmod = lastmod;
1320+
1321+ /* Then move on to activity status changes... */
1322+ switch (action) {
1323+ case SPLIT_ACTIVATE: /* activating split entry */
1324+ split->sp_flags |= SPLIT_ACTIVE; /* make it active... */
1325+ pos += ircd_snprintf(0, buf, sizeof(buf), " activating SPLIT");
1326+ break;
1327+
1328+ case SPLIT_DEACTIVATE: /* deactivating split entry */
1329+ split->sp_flags &= ~SPLIT_ACTIVE; /* make it inactive... */
1330+ pos += ircd_snprintf(0, buf, sizeof(buf), " deactivating SPLIT");
1331+ break;
1332+
1333+ case SPLIT_MODIFY: /* no change to activity status */
1334+ break;
1335+ }
1336+
55404221 1337+ /* TODO: Handle creation changes... */
1338+
12e82c05 1339+ /* Handle expiration changes... */
1340+ if (flags & SPLIT_EXPIRE) {
1341+ split->sp_expire = expire; /* save new expiration time */
1342+ if (pos < BUFSIZE)
1343+ pos += ircd_snprintf(0, buf + pos, sizeof(buf) - pos,
1344+ "%s%s changing expiration time to %Tu",
1345+ pos ? ";" : "",
1346+ pos && !(flags & (SPLIT_LIFETIME | SPLIT_REASON)) ?
1347+ " and" : "", expire);
1348+ }
1349+
1350+ /* Next, handle lifetime changes... */
1351+ if (flags & SPLIT_LIFETIME) {
1352+ split->sp_lifetime = lifetime; /* save new lifetime */
1353+ if (pos < BUFSIZE)
1354+ pos += ircd_snprintf(0, buf + pos, sizeof(buf) - pos,
1355+ "%s%s extending record lifetime to %Tu",
1356+ pos ? ";" : "", pos && !(flags & SPLIT_REASON) ?
1357+ " and" : "", lifetime);
1358+ }
1359+
1360+ /* Now, handle reason changes... */
1361+ if (flags & SPLIT_REASON) {
1362+ MyFree(split->sp_reason); /* release old reason */
1363+ DupString(split->sp_reason, reason); /* store new reason */
1364+ if (pos < BUFSIZE)
1365+ pos += ircd_snprintf(0, buf + pos, sizeof(buf) - pos,
1366+ "%s%s changing reason to \"%s\"",
1367+ pos ? ";" : "", pos ? " and" : "", reason);
1368+ }
1369+
1370+ /* All right, inform ops... */
1371+ sendto_opmask_butone(0, SNO_NETWORK, "%s modifying SPLIT for %s:%s",
1372+ (feature_bool(FEAT_HIS_SNOTICES) || IsServer(sptr)) ?
cb0b5df1 1373+ get_client_name_and_opername(sptr) : cli_name((cli_user(sptr))->server),
12e82c05 1374+ split->sp_server, buf);
1375+
12e82c05 1376+ /* and log the change */
0c63b280 1377+ log_write(LS_SPLIT, L_INFO, LOG_NOSNOTICE,
12e82c05 1378+ "%#C modifying SPLIT for %s:%s",
1379+ sptr, split->sp_server, buf);
1380+
1381+ /* and propagate it */
1382+ split_propagate(cptr, sptr, split, NULL);
1383+
1384+ return 0;
1385+}
1386+
1387+
1388+/** Remove a split entry.
1389+ * @param[in] cptr Local client that sent us the split entry.
1390+ * @param[in] sptr Originator of the split entry.
1391+ * @param[in] split Split entry to remove.
1392+ * @param[in] reason Reason for removing this split entry.
1393+ * @return Zero.
1394+ */
1395+int
1396+split_remove(struct Client *cptr, struct Client *sptr, struct Split *split, const char *reason)
1397+{
1398+ unsigned int saveflags = 0;
1399+
1400+ assert(0 != split);
1401+ assert(NULL != cptr);
1402+ assert(NULL != sptr);
1403+
1404+ Debug((DEBUG_DEBUG, "split_remove(\"%s\", \"%s\", \"%s\", \"%s\")",
1405+ cli_name(cptr), cli_name(sptr), split->sp_server, reason));
1406+
1407+ /* deactivate entry and mark it for removal (used in split_propagate) */
1408+ split->sp_flags |= SPLIT_REMOVING;
0c63b280 1409+ split->sp_flags &= ~SPLIT_ACTIVE;
12e82c05 1410+
1411+ /* inform ops and log it */
1412+ sendto_opmask_butone(0, SNO_NETWORK, "%s removing SPLIT for %s, expiring at %Tu: %s (%s)",
1413+ (feature_bool(FEAT_HIS_SNOTICES) || IsServer(sptr)) ?
cb0b5df1 1414+ get_client_name_and_opername(sptr) : cli_name((cli_user(sptr))->server),
12e82c05 1415+ split->sp_server, split->sp_expire, split->sp_reason, reason);
1416+
0c63b280 1417+ log_write(LS_SPLIT, L_INFO, LOG_NOSNOTICE,
12e82c05 1418+ "%#C removing SPLIT for %s, expiring at %Tu: %s (%s)",
1419+ sptr, split->sp_server, split->sp_expire, split->sp_reason, reason);
1420+
12e82c05 1421+ /* propagate it */
1422+ split_propagate(cptr, sptr, split, reason);
1423+
1424+ /* destroy it */
1425+ split_free(split);
1426+
1427+ return 0;
1428+}
1429+
1430+
1431+/** Find a split entry by name.
1432+ * @param[in] server Split entry name to search for.
1433+ * @return Matching split entry (or NULL if none match).
1434+ */
1435+struct Split *
1436+split_find(char *server)
1437+{
1438+ struct Split* split;
1439+ struct Split* ssplit;
1440+
1441+ for (split = GlobalSplitList; split; split = ssplit) { /* go through splits */
1442+ ssplit = split->sp_next;
1443+
1444+ if (split_expire(split)) /* expire any that need expiring */
1445+ split_free(split);
1446+ else if (0 == ircd_strcmp(server, split->sp_server)) /* found it yet? */
1447+ return split;
1448+ }
1449+ /* TODO: we return 0 not NULL? */
1450+ return 0;
1451+}
1452+
1453+/** Unlink and free an unused split entry.
1454+ * @param[in] split Server split entry to free.
1455+ */
1456+void
1457+split_free(struct Split* split)
1458+{
1459+ /* TODO: use 0 or NULL ? */
1460+ assert(0 != split);
1461+
1462+ *split->sp_prev_p = split->sp_next; /* squeeze this split entry out */
1463+ if (split->sp_next)
1464+ split->sp_next->sp_prev_p = split->sp_prev_p;
1465+
1466+ /* CHECK: the other fields in this struct are destroyed with MyFree() call? */
1467+ MyFree(split->sp_server); /* and free up the memory */
1468+ MyFree(split->sp_reason);
1469+ MyFree(split);
1470+}
1471+
1472+
1473+/** Check whether a split entry has past its life time.
1474+ * when entry is active and past expire time, but not life time, deactivate it
1475+ * @param[in] split Server split entry to check.
1476+ * @return 1 when entry can be free'd, 0 otherwise.
1477+ */
1478+int
1479+split_expire(struct Split* split)
1480+{
1481+ assert(0 != split);
1482+
65b70298 1483+ /* TODO: also check if the server in SPLIT entry is linked atm? */
12e82c05 1484+ /* past lifetime */
0c63b280 1485+ if (split->sp_lifetime <= TStime())
12e82c05 1486+ return 1;
1487+
1488+ /* past expire time, deactivate entry if it is active */
0c63b280 1489+ if ((split->sp_expire <= TStime()) && SplitIsActive(split))
1490+ split->sp_flags &= ~SPLIT_ACTIVE;
12e82c05 1491+
1492+ return 0;
1493+}
1494+
1495+/** Send the full list of split entries to \a cptr.
1496+ * @param[in] cptr Local server to send split entries to.
1497+ */
1498+void
1499+split_burst(struct Client *cptr)
1500+{
1501+ struct Split *split;
1502+ struct Split *ssplit;
1503+
1504+ assert(NULL != cptr);
1505+
1506+ for (split = GlobalSplitList; split; split = ssplit) { /* go through splits */
1507+ ssplit = split->sp_next;
1508+
1509+ /* expire any that need expiring */
1510+ if (split_expire(split))
1511+ split_free(split);
1512+ /* if we have an entry for cptr, dont send it - but do not free here yet
1513+ * free it at end of burst, to get the correct count for SPLITs removed.
1514+ */
1515+ else if (ircd_strcmp(split->sp_server, cli_name(cptr)) == 0)
1516+ continue;
1517+ else
1518+ sendcmdto_one(&me, CMD_SPLIT, cptr, "%c%s %Tu %Tu %Tu %Tu :%s",
1519+ SplitIsActive(split) ? '+' : '-', split->sp_server,
1520+ split->sp_creation, split->sp_expire,
1521+ split->sp_lastmod, split->sp_lifetime,
1522+ split->sp_reason);
1523+ }
1524+}
1525+
1526+/** Forward a split to another server.
1527+ * @param[in] cptr Server to send split entries to.
1528+ * @param[in] split Split to forward.
1529+ * @return Zero.
1530+ */
1531+int
1532+split_resend(struct Client *cptr, struct Split *split)
1533+{
1534+ sendcmdto_one(&me, CMD_SPLIT, cptr, "%c%s %Tu %Tu %Tu %Tu :%s",
1535+ SplitIsActive(split) ? '+' : '-', split->sp_server,
1536+ split->sp_creation, split->sp_expire,
1537+ split->sp_lastmod, split->sp_lifetime,
1538+ split->sp_reason);
1539+
1540+ return 0;
1541+}
1542+
1543+/** Send a split entry (or a list of split entries) to a server.
1544+ * @param[in] sptr Client searching for split entries.
1545+ * @param[in] server Name of split entry to search for (if NULL, list all).
1546+ * @return Zero.
1547+ */
1548+int
1549+split_list(struct Client *sptr, char *server)
1550+{
1551+ struct Split *split;
0c63b280 1552+ int count = 0;
12e82c05 1553+
1554+ assert(NULL != sptr);
1555+
0c63b280 1556+ /* go through SPLITs */
1557+ for (split = GlobalSplitList; split; split = split->sp_next) {
1558+ if (split_expire(split)) { /* expire any that need expiring */
1559+ split_free(split);
1560+ continue;
1561+ }
1562+ if (server && match(server, split->sp_server))
1563+ continue;
1564+ count++;
12e82c05 1565+ send_reply(sptr, RPL_SPLITLIST, split->sp_server,
0c63b280 1566+ split->sp_creation, split->sp_expire, split->sp_lastmod, split->sp_lifetime,
12e82c05 1567+ SplitIsActive(split) ? '+' : '-', split->sp_reason);
12e82c05 1568+ }
1569+
0c63b280 1570+ /* server given and nothing found */
1571+ if (server && !count)
1572+ return send_reply(sptr, ERR_NOSUCHSPLIT, server);
1573+
12e82c05 1574+ /* end of splitlist information */
1575+ return send_reply(sptr, RPL_ENDOFSPLITLIST);
1576+}
1577+
1578+
0c63b280 1579+/** Mark SPLIT for new server with SPLIT_BURST
1580+ * @param[in] acptr Server introduced to the network
1581+ */
1582+void
1583+split_newserver(struct Client *acptr)
1584+{
1585+ struct Split *split;
1586+
1587+ assert(NULL != acptr);
1588+
1589+ Debug((DEBUG_DEBUG, "split_newserver(\"%s\")", cli_name(acptr)));
1590+
1591+ /* mark the SPLIT with SPLIT_BURST
1592+ * so we know this SPLIT is marked for removal already
1593+ * but not remove it just yet to get the correct count
1594+ * of SPLITs removed at the end of burst.
1595+ * also we need the SPLIT in case the net breaks again
1596+ * before end of burst?
1597+ */
1598+ if ((split = split_find(cli_name(acptr))))
1599+ split->sp_flags |= SPLIT_BURST;
1600+ return;
1601+}
1602+
1603+
12e82c05 1604+/** Auto destroy SPLITs for servers gained in a netmerge
0c63b280 1605+ * @param[in] sptr Server that has linked to the network
12e82c05 1606+ * @return Number of destroyed SPLITs
1607+ */
1608+int
0c63b280 1609+split_netmerge(struct Client *sptr)
12e82c05 1610+{
1611+ struct DLink *lp;
1612+ struct Split *split;
1613+ int count = 0;
1614+
0c63b280 1615+ assert(NULL != sptr);
12e82c05 1616+
0c63b280 1617+ Debug((DEBUG_DEBUG, "split_netmerge(\"%s\")", cli_name(sptr)));
12e82c05 1618+
1619+ /* find the SPLIT for this server */
0c63b280 1620+ if ((split = split_find(cli_name(sptr)))) {
12e82c05 1621+ split_free(split);
1622+ count++;
1623+ }
1624+
1625+ /* go over its downlinks */
0c63b280 1626+ for (lp = cli_serv(sptr)->down; lp; lp = lp->next)
1627+ count += split_netmerge(lp->value.cptr);
12e82c05 1628+ return count;
1629+}
1630+
1631+
1632+/** Auto create SPLITs for servers lost in a netbreak
0c63b280 1633+ * @param[in] victim Server that lost link to network
12e82c05 1634+ * @param[in] reason Reason to add to SPLITs
1635+ * @return Number of created SPLITs
1636+ */
1637+int
0c63b280 1638+split_netbreak(struct Client *victim, const char *reason)
12e82c05 1639+{
1640+ struct DLink *lp;
1641+ struct Split *split;
1642+ int count = 0;
0c63b280 1643+ time_t creation = TStime(), expire = TStime() + feature_int(FEAT_SPLIT_AUTO_EXPIRE),
1644+ lastmod = TStime(), lifetime = expire;
12e82c05 1645+ unsigned int flags = SPLIT_ACTIVE;
1646+
0c63b280 1647+ assert(NULL != victim);
12e82c05 1648+
0c63b280 1649+ Debug((DEBUG_DEBUG, "split_netbreak(\"%s\", \"%s\")", cli_name(victim), reason));
12e82c05 1650+
65b70298 1651+ /* bad auto expire value */
1652+ if (feature_int(FEAT_SPLIT_AUTO_EXPIRE) <= 0)
1653+ return 0;
1654+
0c63b280 1655+ /* TODO: what if we do have a SPLIT for this server?
1656+ * we remove splits in end of burst
1657+ * and if the netbreaks again before that
1658+ * free the SPLIT and create a new one?
1659+ * or simply update lastmod/lifetime/reason but leave the status?
1660+ */
12e82c05 1661+ /* find the SPLIT for this server */
0c63b280 1662+ if (!(split = split_find(cli_name(victim)))) {
1663+ split_make(cli_name(victim), reason, creation, expire, lastmod, lifetime, flags);
12e82c05 1664+ count++;
0c63b280 1665+ } else
1666+ split->sp_flags &= SPLIT_BURST; /* clear the burst flag for it */
12e82c05 1667+
1668+ /* go over its downlinks */
0c63b280 1669+ for (lp = cli_serv(victim)->down; lp; lp = lp->next)
1670+ count += split_netbreak(lp->value.cptr, reason);
12e82c05 1671+ return count;
1672+}
1673+
1674+
1675+/** Auto create SPLITs for servers we have a Connect Block for
1676+ *
1677+ */
1678+void
1679+split_conf()
1680+{
1681+ struct ConfItem *conf;
12e82c05 1682+ struct Split *split;
0c63b280 1683+ time_t creation = TStime(), expire = TStime() + feature_int(FEAT_SPLIT_AUTO_EXPIRE),
1684+ lastmod = TStime(), lifetime = expire;
12e82c05 1685+ unsigned int flags = SPLIT_ACTIVE;
1686+ char reason[BUFSIZE];
1687+
1688+ Debug((DEBUG_DEBUG, "split_conf()"));
1689+
1690+ /* we are not set to generate SPLITs */
1691+ if (!feature_bool(FEAT_SPLIT))
1692+ return;
1693+
0c63b280 1694+ /* TODO: log this problem? */
65b70298 1695+ /* bad auto expire value */
1696+ if (feature_int(FEAT_SPLIT_AUTO_EXPIRE) <= 0)
1697+ return;
1698+
12e82c05 1699+ ircd_snprintf(0, reason, sizeof(reason),
0c63b280 1700+ "Generated upon loading conf file on %C", &me);
12e82c05 1701+
1702+ /* go over the conf contents */
1703+ for (conf = GlobalConfList; conf; conf = conf->next) {
cb0b5df1 1704+
12e82c05 1705+ /* not a Connect Block */
1706+ if (CONF_SERVER != conf->status)
1707+ continue;
cb0b5df1 1708+
12e82c05 1709+ /* server is linked */
cb0b5df1 1710+ if (FindServer(conf->name))
12e82c05 1711+ continue;
cb0b5df1 1712+
12e82c05 1713+ /* we have a SPLIT for this server already */
cb0b5df1 1714+ if (split_find(conf->name))
12e82c05 1715+ continue;
cb0b5df1 1716+
65b70298 1717+ /* TODO: use SNO_OLDSNO here?
1718+ * just like nickcollisions on local users go there?
1719+ * or at least change the source / message format?
1720+ * /REHASH goes to SNO_OLDSNO? so this too?
1721+ * if oper did rehash, add his name here?
1722+ */
12e82c05 1723+ /* inform ops and log it */
1724+ sendto_opmask_butone(0, SNO_NETWORK, "%C adding SPLIT for %s, expiring at %Tu: %s",
1725+ &me, conf->name, expire, reason);
1726+
0c63b280 1727+ log_write(LS_SPLIT, L_INFO, LOG_NOSNOTICE,
12e82c05 1728+ "%C adding SPLIT for %s, expiring at %Tu: %s",
1729+ &me, conf->name, expire, reason);
1730+
1731+ /* make the split entry */
cb0b5df1 1732+ split = split_make(conf->name, reason, creation, expire, lastmod, lifetime, flags);
12e82c05 1733+
cb0b5df1 1734+ assert(split);
12e82c05 1735+
1736+ /* and propagate it */
cb0b5df1 1737+ split_propagate(&me, &me, split, NULL);
12e82c05 1738+ }
1739+}
1740+
1741+
1742+/** Statistics callback to list SPLITs.
1743+ * @param[in] sptr Client requesting statistics.
1744+ * @param[in] sd Stats descriptor for request (ignored).
1745+ * @param[in] param Extra parameter from user (ignored).
1746+ */
1747+void
cb0b5df1 1748+split_stats(struct Client *sptr, const struct StatDesc *sd, char *param)
12e82c05 1749+{
1750+ struct Split *split;
12e82c05 1751+
65b70298 1752+ /* send header so the client knows what we are showing */
1753+ send_reply(sptr, SND_EXPLICIT | RPL_STATSHEADER,
1754+ "S Server Creation Expire Lastmod Lifetime Status :Reason");
1755+
0c63b280 1756+ for (split = GlobalSplitList; split; split = split->sp_next) {
cb0b5df1 1757+
1758+ /* does not match */
0c63b280 1759+ if (param && match(param, split->sp_server))
1760+ continue;
cb0b5df1 1761+
1762+ /* send it */
12e82c05 1763+ send_reply(sptr, RPL_STATSSPLIT, split->sp_server,
1764+ split->sp_creation, split->sp_expire, split->sp_lastmod, split->sp_lifetime,
1765+ SplitIsActive(split) ? '+' : '-', split->sp_reason);
0c63b280 1766+ }
12e82c05 1767+}
1768+
1769+
1770+/** Count split entries and memory used by them.
1771+ * @param[out] sp_size Receives total number of bytes allocated for split entries.
1772+ * @return Number of split entries currently allocated.
1773+ */
1774+int
1775+split_memory_count(size_t *sp_size)
1776+{
1777+ struct Split *split;
1778+ unsigned int sp = 0;
1779+
0c63b280 1780+ for (split = GlobalSplitList; split; split = split->sp_next) {
12e82c05 1781+ sp++;
1782+ *sp_size += sizeof(struct Split);
1783+ *sp_size += split->sp_server ? (strlen(split->sp_server) + 1) : 0;
1784+ *sp_size += split->sp_reason ? (strlen(split->sp_reason) + 1) : 0;
1785+ }
1786+ return sp;
1787+}
1788+