2 * ircd-ratbox: an advanced Internet Relay Chat Daemon(ircd).
3 * m_etrace.c: Gives local opers a trace output with added info.
5 * Copyright (C) 2002-2003 Lee Hardy <lee@leeh.co.uk>
6 * Copyright (C) 2002-2005 ircd-ratbox development team
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions are
12 * 1.Redistributions of source code must retain the above copyright notice,
13 * this list of conditions and the following disclaimer.
14 * 2.Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3.The name of the author may not be used to endorse or promote products
18 * derived from this software without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
21 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
22 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
24 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
25 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
26 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
29 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30 * POSSIBILITY OF SUCH DAMAGE.
44 #include "s_newconf.h"
50 #include "supported.h"
52 static const char etrace_desc
[] =
53 "Provides enhanced tracing facilities to opers (ETRACE, CHANTRACE, and MASKTRACE)";
55 static void mo_etrace(struct MsgBuf
*, struct Client
*, struct Client
*, int, const char **);
56 static void me_etrace(struct MsgBuf
*, struct Client
*, struct Client
*, int, const char **);
57 static void m_chantrace(struct MsgBuf
*, struct Client
*, struct Client
*, int, const char **);
58 static void mo_masktrace(struct MsgBuf
*, struct Client
*, struct Client
*, int, const char **);
60 struct Message etrace_msgtab
= {
62 {mg_ignore
, mg_not_oper
, mg_ignore
, mg_ignore
, {me_etrace
, 0}, {mo_etrace
, 0}}
64 struct Message chantrace_msgtab
= {
65 "CHANTRACE", 0, 0, 0, 0,
66 {mg_ignore
, {m_chantrace
, 2}, mg_ignore
, mg_ignore
, mg_ignore
, {m_chantrace
, 2}}
68 struct Message masktrace_msgtab
= {
69 "MASKTRACE", 0, 0, 0, 0,
70 {mg_ignore
, mg_not_oper
, mg_ignore
, mg_ignore
, mg_ignore
, {mo_masktrace
, 2}}
76 add_isupport("ETRACE", isupport_string
, "");
84 delete_isupport("ETRACE");
87 mapi_clist_av1 etrace_clist
[] = { &etrace_msgtab
, &chantrace_msgtab
, &masktrace_msgtab
, NULL
};
89 DECLARE_MODULE_AV2(etrace
, _modinit
, _moddeinit
, etrace_clist
, NULL
, NULL
, NULL
, NULL
, etrace_desc
);
91 static void do_etrace(struct Client
*source_p
, int ipv4
, int ipv6
);
92 static void do_etrace_full(struct Client
*source_p
);
93 static void do_single_etrace(struct Client
*source_p
, struct Client
*target_p
);
95 static const char *empty_sockhost
= "255.255.255.255";
96 static const char *spoofed_sockhost
= "0";
100 * parv[1] = options [or target]
104 mo_etrace(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
106 if(parc
> 1 && !EmptyString(parv
[1]))
108 if(!irccmp(parv
[1], "-full"))
109 do_etrace_full(source_p
);
110 else if(!irccmp(parv
[1], "-v6"))
111 do_etrace(source_p
, 0, 1);
112 else if(!irccmp(parv
[1], "-v4"))
113 do_etrace(source_p
, 1, 0);
116 struct Client
*target_p
= find_named_person(parv
[1]);
120 if(!MyClient(target_p
))
121 sendto_one(target_p
, ":%s ENCAP %s ETRACE %s",
122 get_id(source_p
, target_p
),
123 target_p
->servptr
->name
,
124 get_id(target_p
, target_p
));
126 do_single_etrace(source_p
, target_p
);
129 sendto_one_numeric(source_p
, ERR_NOSUCHNICK
,
130 form_str(ERR_NOSUCHNICK
), parv
[1]);
134 do_etrace(source_p
, 1, 1);
138 me_etrace(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
140 struct Client
*target_p
;
142 if(!IsOper(source_p
) || parc
< 2 || EmptyString(parv
[1]))
145 /* we cant etrace remote clients.. we shouldnt even get sent them */
146 if((target_p
= find_person(parv
[1])) && MyClient(target_p
))
147 do_single_etrace(source_p
, target_p
);
149 sendto_one_numeric(source_p
, RPL_ENDOFTRACE
, form_str(RPL_ENDOFTRACE
),
150 target_p
? target_p
->name
: parv
[1]);
154 do_etrace(struct Client
*source_p
, int ipv4
, int ipv6
)
156 struct Client
*target_p
;
159 /* report all direct connections */
160 RB_DLINK_FOREACH(ptr
, lclient_list
.head
)
162 target_p
= ptr
->data
;
164 if((!ipv4
&& GET_SS_FAMILY(&target_p
->localClient
->ip
) == AF_INET
) ||
165 (!ipv6
&& GET_SS_FAMILY(&target_p
->localClient
->ip
) == AF_INET6
))
168 sendto_one(source_p
, form_str(RPL_ETRACE
),
169 me
.name
, source_p
->name
,
170 IsOper(target_p
) ? "Oper" : "User",
171 get_client_class(target_p
),
172 target_p
->name
, target_p
->username
, target_p
->host
,
173 show_ip(source_p
, target_p
) ? target_p
->sockhost
: "255.255.255.255",
177 sendto_one_numeric(source_p
, RPL_ENDOFTRACE
, form_str(RPL_ENDOFTRACE
), me
.name
);
181 do_etrace_full(struct Client
*source_p
)
185 RB_DLINK_FOREACH(ptr
, lclient_list
.head
)
187 do_single_etrace(source_p
, ptr
->data
);
190 sendto_one_numeric(source_p
, RPL_ENDOFTRACE
, form_str(RPL_ENDOFTRACE
), me
.name
);
194 * do_single_etrace - searches local clients and displays those matching
196 * input - source client, target client
197 * output - etrace results
198 * side effects - etrace results are displayed
201 do_single_etrace(struct Client
*source_p
, struct Client
*target_p
)
203 /* note, we hide fullcaps for spoofed users, as mirc can often
204 * advertise its internal ip address in the field --fl
206 if(!show_ip(source_p
, target_p
))
207 sendto_one(source_p
, form_str(RPL_ETRACEFULL
),
208 me
.name
, source_p
->name
,
209 IsOper(target_p
) ? "Oper" : "User",
210 get_client_class(target_p
),
211 target_p
->name
, target_p
->username
, target_p
->host
,
212 "255.255.255.255", "<hidden> <hidden>", target_p
->info
);
214 sendto_one(source_p
, form_str(RPL_ETRACEFULL
),
215 me
.name
, source_p
->name
,
216 IsOper(target_p
) ? "Oper" : "User",
217 get_client_class(target_p
),
218 target_p
->name
, target_p
->username
,
219 target_p
->host
, target_p
->sockhost
,
220 target_p
->localClient
->fullcaps
, target_p
->info
);
224 m_chantrace(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
226 struct Client
*target_p
;
227 struct Channel
*chptr
;
228 struct membership
*msptr
;
229 const char *sockhost
;
236 if(IsOperSpy(source_p
) && parv
[1][0] == '!')
241 if(EmptyString(name
))
243 sendto_one(source_p
, form_str(ERR_NEEDMOREPARAMS
),
244 me
.name
, source_p
->name
, "CHANTRACE");
249 if((chptr
= find_channel(name
)) == NULL
)
251 sendto_one_numeric(source_p
, ERR_NOSUCHCHANNEL
, form_str(ERR_NOSUCHCHANNEL
),
256 /* dont report operspys for nonexistant channels. */
258 report_operspy(source_p
, "CHANTRACE", chptr
->chname
);
260 if(!operspy
&& !IsMember(client_p
, chptr
))
262 sendto_one_numeric(source_p
, ERR_NOTONCHANNEL
, form_str(ERR_NOTONCHANNEL
),
267 RB_DLINK_FOREACH(ptr
, chptr
->members
.head
)
270 target_p
= msptr
->client_p
;
272 if(EmptyString(target_p
->sockhost
))
273 sockhost
= empty_sockhost
;
274 else if(!show_ip(source_p
, target_p
))
275 sockhost
= spoofed_sockhost
;
277 sockhost
= target_p
->sockhost
;
279 sendto_one(source_p
, form_str(RPL_ETRACE
),
280 me
.name
, source_p
->name
,
281 IsOper(target_p
) ? "Oper" : "User",
282 /* class field -- pretend its server.. */
283 target_p
->servptr
->name
,
284 target_p
->name
, target_p
->username
, target_p
->host
,
285 sockhost
, target_p
->info
);
288 sendto_one_numeric(source_p
, RPL_ENDOFTRACE
, form_str(RPL_ENDOFTRACE
), me
.name
);
292 match_masktrace(struct Client
*source_p
, rb_dlink_list
*list
,
293 const char *username
, const char *hostname
, const char *name
,
296 struct Client
*target_p
;
298 const char *sockhost
;
300 RB_DLINK_FOREACH(ptr
, list
->head
)
302 target_p
= ptr
->data
;
303 if(!IsPerson(target_p
))
306 if(EmptyString(target_p
->sockhost
))
307 sockhost
= empty_sockhost
;
308 else if(!show_ip(source_p
, target_p
))
309 sockhost
= spoofed_sockhost
;
311 sockhost
= target_p
->sockhost
;
313 if(match(username
, target_p
->username
) &&
314 (match(hostname
, target_p
->host
) ||
315 match(hostname
, target_p
->orighost
) ||
316 match(hostname
, sockhost
) || match_ips(hostname
, sockhost
)))
318 if(name
!= NULL
&& !match(name
, target_p
->name
))
321 if(gecos
!= NULL
&& !match_esc(gecos
, target_p
->info
))
324 sendto_one(source_p
, form_str(RPL_ETRACE
),
325 me
.name
, source_p
->name
,
326 IsOper(target_p
) ? "Oper" : "User",
327 /* class field -- pretend its server.. */
328 target_p
->servptr
->name
,
329 target_p
->name
, target_p
->username
, target_p
->host
,
330 sockhost
, target_p
->info
);
336 mo_masktrace(struct MsgBuf
*msgbuf_p
, struct Client
*client_p
, struct Client
*source_p
, int parc
,
339 char *name
, *username
, *hostname
, *gecos
;
344 name
= LOCAL_COPY(parv
[1]);
347 if(IsOperSpy(source_p
) && parv
[1][0] == '!')
354 if(parc
> 2 && !EmptyString(parv
[2]))
356 gecos
= LOCAL_COPY(parv
[2]);
362 if((hostname
= strchr(name
, '@')) == NULL
)
364 sendto_one_notice(source_p
, ":Invalid parameters");
370 if((username
= strchr(name
, '!')) == NULL
)
377 if(EmptyString(username
) || EmptyString(hostname
))
379 sendto_one_notice(source_p
, ":Invalid parameters");
384 if (!ConfigFileEntry
.operspy_dont_care_user_info
)
387 rb_strlcpy(buf
, mask
, sizeof(buf
));
388 if(!EmptyString(gecos
)) {
389 rb_strlcat(buf
, " ", sizeof(buf
));
390 rb_strlcat(buf
, gecos
, sizeof(buf
));
393 report_operspy(source_p
, "MASKTRACE", buf
);
395 match_masktrace(source_p
, &global_client_list
, username
, hostname
, name
, gecos
);
397 match_masktrace(source_p
, &lclient_list
, username
, hostname
, name
, gecos
);
399 sendto_one_numeric(source_p
, RPL_ENDOFTRACE
, form_str(RPL_ENDOFTRACE
), me
.name
);