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"
48 um_callerid_modinit(void)
50 user_modes
['g'] = find_umode_slot();
53 ierror("um_callerid: unable to allocate usermode slot for +g; unloading module.");
57 user_modes
['G'] = find_umode_slot();
60 ierror("um_callerid: unable to allocate usermode slot for +G; unloading module.");
66 add_isupport("CALLERID", isupport_string
, "g");
72 um_callerid_moddeinit(void)
78 delete_isupport("CALLERID");
81 #define IsSetCallerID(c) ((c->umodes & user_modes['g']) == user_modes['g'])
82 #define IsSetRelaxedCallerID(c) ((c->umodes & user_modes['G']) == user_modes['G'])
84 static const char um_callerid_desc
[] =
85 "Provides usermode +g which restricts messages from unauthorized users.";
88 has_common_channel(struct Client
*source_p
, struct Client
*target_p
)
92 RB_DLINK_FOREACH(ptr
, source_p
->user
->channel
.head
)
94 struct membership
*msptr
= ptr
->data
;
95 if (IsMember(target_p
, msptr
->chptr
))
103 allow_message(struct Client
*source_p
, struct Client
*target_p
)
105 if (!MyClient(target_p
))
108 if (!IsSetCallerID(target_p
))
111 if (IsSetRelaxedCallerID(target_p
) && has_common_channel(source_p
, target_p
))
114 if (IsServer(source_p
))
117 /* XXX: controversial? allow opers to send through +g */
118 if (IsOper(source_p
))
121 if (accept_message(source_p
, target_p
))
128 send_callerid_notice(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
130 if (!MyClient(target_p
))
133 if (msgtype
== MESSAGE_TYPE_NOTICE
)
136 sendto_one_numeric(source_p
, ERR_TARGUMODEG
, form_str(ERR_TARGUMODEG
),
139 if ((target_p
->localClient
->last_caller_id_time
+ ConfigFileEntry
.caller_id_wait
) < rb_current_time())
141 sendto_one_numeric(source_p
, RPL_TARGNOTIFY
, form_str(RPL_TARGNOTIFY
),
144 sendto_one(target_p
, form_str(RPL_UMODEGMSG
),
145 me
.name
, target_p
->name
, source_p
->name
,
146 source_p
->username
, source_p
->host
);
148 target_p
->localClient
->last_caller_id_time
= rb_current_time();
153 add_callerid_accept_for_source(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
156 * XXX: Controversial? Allow target users to send replies
157 * through a +g. Rationale is that people can presently use +g
158 * as a way to taunt users, e.g. harass them and hide behind +g
159 * as a way of griefing. --nenolod
161 if(msgtype
!= MESSAGE_TYPE_NOTICE
&&
162 IsSetCallerID(source_p
) &&
163 !accept_message(target_p
, source_p
) &&
166 if(rb_dlink_list_length(&source_p
->localClient
->allow_list
) <
167 (unsigned long)ConfigFileEntry
.max_accept
)
169 rb_dlinkAddAlloc(target_p
, &source_p
->localClient
->allow_list
);
170 rb_dlinkAddAlloc(source_p
, &target_p
->on_allow_list
);
174 sendto_one_numeric(source_p
, ERR_OWNMODE
,
175 form_str(ERR_OWNMODE
),
176 target_p
->name
, "+g");
185 h_hdl_invite(void *vdata
)
187 hook_data_channel_approval
*data
= vdata
;
188 struct Client
*source_p
= data
->client
;
189 struct Client
*target_p
= data
->target
;
194 if (!add_callerid_accept_for_source(MESSAGE_TYPE_PRIVMSG
, source_p
, target_p
))
196 data
->approved
= ERR_TARGUMODEG
;
200 if (allow_message(source_p
, target_p
))
203 send_callerid_notice(MESSAGE_TYPE_PRIVMSG
, source_p
, target_p
);
205 data
->approved
= ERR_TARGUMODEG
;
209 h_hdl_privmsg_user(void *vdata
)
211 hook_data_privmsg_user
*data
= vdata
;
212 enum message_type msgtype
= data
->msgtype
;
213 struct Client
*source_p
= data
->source_p
;
214 struct Client
*target_p
= data
->target_p
;
219 if (!add_callerid_accept_for_source(msgtype
, source_p
, target_p
))
221 data
->approved
= ERR_TARGUMODEG
;
225 if (allow_message(source_p
, target_p
))
228 send_callerid_notice(msgtype
, source_p
, target_p
);
230 data
->approved
= ERR_TARGUMODEG
;
233 static mapi_hfn_list_av1 um_callerid_hfnlist
[] = {
234 { "invite", h_hdl_invite
},
235 { "privmsg_user", h_hdl_privmsg_user
},
239 DECLARE_MODULE_AV2(um_callerid
, um_callerid_modinit
, um_callerid_moddeinit
,
240 NULL
, NULL
, um_callerid_hfnlist
, NULL
, NULL
, um_callerid_desc
);