2 * modules/um_callerid.c
3 * Copyright (c) 2020 Ariadne Conill <ariadne@dereferenced.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
9 * 1. Redistributions of source code must retain the above copyright notice,
10 * this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
21 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
25 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
26 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 * POSSIBILITY OF SUCH DAMAGE.
41 #include "privilege.h"
42 #include "s_newconf.h"
44 #include "supported.h"
47 um_callerid_modinit(void)
49 user_modes
['g'] = find_umode_slot();
50 user_modes
['G'] = find_umode_slot();
53 add_isupport("CALLERID", isupport_string
, "g");
59 um_callerid_moddeinit(void)
65 delete_isupport("CALLERID");
68 #define IsSetCallerID(c) ((c->umodes & user_modes['g']) == user_modes['g'])
69 #define IsSetRelaxedCallerID(c) ((c->umodes & user_modes['G']) == user_modes['G'])
71 static const char um_callerid_desc
[] =
72 "Provides usermode +g which restricts messages from unauthorized users.";
75 has_common_channel(struct Client
*source_p
, struct Client
*target_p
)
79 RB_DLINK_FOREACH(ptr
, source_p
->user
->channel
.head
)
81 struct membership
*msptr
= ptr
->data
;
82 if (IsMember(target_p
, msptr
->chptr
))
90 allow_message(struct Client
*source_p
, struct Client
*target_p
)
92 if (!MyClient(target_p
))
95 if (!IsSetCallerID(target_p
))
98 if (IsSetRelaxedCallerID(target_p
) && has_common_channel(source_p
, target_p
))
101 if (IsServer(source_p
))
104 /* XXX: controversial? allow opers to send through +g */
105 if (IsOper(source_p
))
108 if (accept_message(source_p
, target_p
))
115 send_callerid_notice(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
117 if (!MyClient(target_p
))
120 if (msgtype
== MESSAGE_TYPE_NOTICE
)
123 sendto_one_numeric(source_p
, ERR_TARGUMODEG
, form_str(ERR_TARGUMODEG
),
126 if ((target_p
->localClient
->last_caller_id_time
+ ConfigFileEntry
.caller_id_wait
) < rb_current_time())
128 sendto_one_numeric(source_p
, RPL_TARGNOTIFY
, form_str(RPL_TARGNOTIFY
),
131 sendto_one(target_p
, form_str(RPL_UMODEGMSG
),
132 me
.name
, target_p
->name
, source_p
->name
,
133 source_p
->username
, source_p
->host
);
135 target_p
->localClient
->last_caller_id_time
= rb_current_time();
140 add_callerid_accept_for_source(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
143 * XXX: Controversial? Allow target users to send replies
144 * through a +g. Rationale is that people can presently use +g
145 * as a way to taunt users, e.g. harass them and hide behind +g
146 * as a way of griefing. --nenolod
148 if(msgtype
!= MESSAGE_TYPE_NOTICE
&&
149 IsSetCallerID(source_p
) &&
150 !accept_message(target_p
, source_p
) &&
153 if(rb_dlink_list_length(&source_p
->localClient
->allow_list
) <
154 (unsigned long)ConfigFileEntry
.max_accept
)
156 rb_dlinkAddAlloc(target_p
, &source_p
->localClient
->allow_list
);
157 rb_dlinkAddAlloc(source_p
, &target_p
->on_allow_list
);
161 sendto_one_numeric(source_p
, ERR_OWNMODE
,
162 form_str(ERR_OWNMODE
),
163 target_p
->name
, "+g");
172 h_hdl_invite(void *vdata
)
174 hook_data_channel_approval
*data
= vdata
;
175 struct Client
*source_p
= data
->client
;
176 struct Client
*target_p
= data
->target
;
181 if (!add_callerid_accept_for_source(MESSAGE_TYPE_PRIVMSG
, source_p
, target_p
))
183 data
->approved
= ERR_TARGUMODEG
;
187 if (allow_message(source_p
, target_p
))
190 send_callerid_notice(MESSAGE_TYPE_PRIVMSG
, source_p
, target_p
);
192 data
->approved
= ERR_TARGUMODEG
;
196 h_hdl_privmsg_user(void *vdata
)
198 hook_data_privmsg_user
*data
= vdata
;
199 enum message_type msgtype
= data
->msgtype
;
200 struct Client
*source_p
= data
->source_p
;
201 struct Client
*target_p
= data
->target_p
;
206 if (!add_callerid_accept_for_source(msgtype
, source_p
, target_p
))
208 data
->approved
= ERR_TARGUMODEG
;
212 if (allow_message(source_p
, target_p
))
215 send_callerid_notice(msgtype
, source_p
, target_p
);
217 data
->approved
= ERR_TARGUMODEG
;
220 static mapi_hfn_list_av1 um_callerid_hfnlist
[] = {
221 { "invite", h_hdl_invite
},
222 { "privmsg_user", h_hdl_privmsg_user
},
226 DECLARE_MODULE_AV2(um_callerid
, um_callerid_modinit
, um_callerid_moddeinit
,
227 NULL
, NULL
, um_callerid_hfnlist
, NULL
, NULL
, um_callerid_desc
);