2 * charybdis: an advanced Internet Relay Chat Daemon(ircd).
3 * m_scan.c: Provides information about various targets on various topics
5 * Copyright (c) 2006 William Pitcock <nenolod -at- nenolod.net>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are
11 * 1.Redistributions of source code must retain the above copyright notice,
12 * this list of conditions and the following disclaimer.
13 * 2.Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3.The name of the author may not be used to endorse or promote products
17 * derived from this software without specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
23 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
25 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
27 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
28 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
31 * $Id: m_scan.c 1853 2006-08-24 18:30:52Z jilles $
46 #include "s_newconf.h"
53 static int mo_scan(struct Client
*, struct Client
*, int, const char **);
54 static int scan_umodes(struct Client
*, struct Client
*, int, const char **);
55 /*static int scan_cmodes(struct Client *, struct Client *, int, const char **);*/
57 struct Message scan_msgtab
= {
58 "SCAN", 0, 0, 0, MFLG_SLOW
,
59 {mg_ignore
, mg_not_oper
, mg_ignore
, mg_ignore
, mg_ignore
, {mo_scan
, 2}}
62 mapi_clist_av1 scan_clist
[] = { &scan_msgtab
, NULL
};
63 DECLARE_MODULE_AV1(scan
, NULL
, NULL
, scan_clist
, NULL
, NULL
, "$Revision: 1853 $");
65 typedef int (*scan_handler
)(struct Client
*, struct Client
*, int,
73 {"UMODES", L_OPER
, scan_umodes
},
77 static const char *empty_sockhost
= "255.255.255.255";
78 static const char *spoofed_sockhost
= "0";
82 * parv[1] = options [or target]
86 mo_scan(struct Client
*client_p
, struct Client
*source_p
, int parc
,
89 struct scan_cmd
*sptr
;
91 for (sptr
= scan_cmds
; sptr
->name
!= NULL
; sptr
++)
93 if (!irccmp(sptr
->name
, parv
[1]))
95 if (sptr
->operlevel
== L_ADMIN
&&
96 !IsOperAdmin(source_p
))
99 return sptr
->handler(client_p
, source_p
, parc
, parv
);
103 sendto_one_notice(source_p
, ":*** %s is not an implemented SCAN target",
110 scan_umodes(struct Client
*client_p
, struct Client
*source_p
, int parc
,
113 unsigned int allowed_umodes
= 0, disallowed_umodes
= 0;
116 int list_users
= YES
;
118 int list_count
= 0, count
= 0;
119 const char *mask
= NULL
;
121 struct Client
*target_p
;
122 rb_dlink_list
*target_list
= &lclient_list
; /* local clients only by default */
125 const char *sockhost
;
130 if (MyClient(source_p
))
131 sendto_one(source_p
, form_str(ERR_NEEDMOREPARAMS
),
132 me
.name
, source_p
->name
, "SCAN UMODES");
137 if (parv
[2][0] != '+' && parv
[2][0] != '-')
139 sendto_one_notice(source_p
, ":SCAN UMODES: umodes parameter must start with '+' or '-'");
143 for (c
= parv
[2]; *c
; c
++)
154 if ((mode
= user_modes
[(unsigned char) *c
]) != 0)
156 if (what
== MODE_ADD
)
157 allowed_umodes
|= mode
;
159 disallowed_umodes
|= mode
;
164 for (i
= 3; i
< parc
; i
++)
166 if (!irccmp(parv
[i
], "no-list"))
168 else if (!irccmp(parv
[i
], "list"))
170 else if (!irccmp(parv
[i
], "global"))
171 target_list
= &global_client_list
;
172 else if (i
< (parc
- 1))
174 if (!irccmp(parv
[i
], "list-max"))
175 list_max
= atoi(parv
[++i
]);
176 else if (!irccmp(parv
[i
], "mask"))
180 sendto_one_notice(source_p
, ":SCAN UMODES: invalid parameters");
186 sendto_one_notice(source_p
, ":SCAN UMODES: invalid parameters");
190 if (target_list
== &global_client_list
&& list_users
)
192 if (IsOperSpy(source_p
))
194 if (!ConfigFileEntry
.operspy_dont_care_user_info
)
196 rb_strlcpy(buf
, "UMODES", sizeof buf
);
197 for (i
= 2; i
< parc
; i
++)
199 rb_strlcat(buf
, " ", sizeof buf
);
200 rb_strlcat(buf
, parv
[i
], sizeof buf
);
202 report_operspy(source_p
, "SCAN", buf
);
207 sendto_one(source_p
, form_str(ERR_NOPRIVS
),
208 me
.name
, source_p
->name
, "oper_spy");
213 RB_DLINK_FOREACH(tn
, target_list
->head
)
215 unsigned int working_umodes
= 0;
216 char maskbuf
[BUFSIZE
];
220 if (!IsClient(target_p
))
223 if(EmptyString(target_p
->sockhost
))
224 sockhost
= empty_sockhost
;
225 else if(!show_ip(source_p
, target_p
))
226 sockhost
= spoofed_sockhost
;
228 sockhost
= target_p
->sockhost
;
230 working_umodes
= target_p
->umodes
;
232 /* require that we have the allowed umodes... */
233 if ((working_umodes
& allowed_umodes
) != allowed_umodes
)
236 /* require that we have NONE of the disallowed ones */
237 if ((working_umodes
& disallowed_umodes
) != 0)
242 rb_snprintf(maskbuf
, BUFSIZE
, "%s!%s@%s",
243 target_p
->name
, target_p
->username
, target_p
->host
);
245 if (!match(mask
, maskbuf
))
249 if (list_users
&& (!list_max
|| (list_count
< list_max
)))
251 char modebuf
[BUFSIZE
];
256 for (i
= 0; i
< 128; i
++)
258 if (target_p
->umodes
& user_modes
[i
])
266 sendto_one_numeric(source_p
, RPL_SCANUMODES
,
267 form_str(RPL_SCANUMODES
),
268 target_p
->name
, target_p
->username
,
269 target_p
->host
, sockhost
,
270 target_p
->servptr
->name
, modebuf
,
276 sendto_one_numeric(source_p
, RPL_SCANMATCHED
,
277 form_str(RPL_SCANMATCHED
), count
);