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();
62 ierror("um_callerid: unable to allocate usermode slot for +G; unloading module.");
68 add_isupport("CALLERID", isupport_umode
, "g");
74 um_callerid_moddeinit(void)
80 delete_isupport("CALLERID");
83 #define IsSetStrictCallerID(c) ((c->umodes & user_modes['g']) == user_modes['g'])
84 #define IsSetRelaxedCallerID(c) ((c->umodes & user_modes['G']) == user_modes['G'])
85 #define IsSetAnyCallerID(c) (IsSetStrictCallerID(c) || IsSetRelaxedCallerID(c))
87 static const char um_callerid_desc
[] =
88 "Provides usermodes +g and +G which restrict messages from unauthorized users.";
91 has_common_channel(struct Client
*source_p
, struct Client
*target_p
)
95 RB_DLINK_FOREACH(ptr
, source_p
->user
->channel
.head
)
97 struct membership
*msptr
= ptr
->data
;
98 if (IsMember(target_p
, msptr
->chptr
))
106 allow_message(struct Client
*source_p
, struct Client
*target_p
)
108 if (!MyClient(target_p
))
111 if (!IsSetAnyCallerID(target_p
))
114 if (IsSetRelaxedCallerID(target_p
) && has_common_channel(source_p
, target_p
) && !IsSetStrictCallerID(target_p
))
117 if (IsServer(source_p
))
120 /* XXX: controversial? allow opers to send through +g */
121 if (IsOperGeneral(source_p
))
124 if (accept_message(source_p
, target_p
))
131 send_callerid_notice(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
133 if (!MyClient(target_p
))
136 if (msgtype
== MESSAGE_TYPE_NOTICE
)
139 sendto_one_numeric(source_p
, ERR_TARGUMODEG
, form_str(ERR_TARGUMODEG
),
140 target_p
->name
, IsSetStrictCallerID(target_p
) ? "+g" : "+G");
142 if ((target_p
->localClient
->last_caller_id_time
+ ConfigFileEntry
.caller_id_wait
) < rb_current_time())
144 sendto_one_numeric(source_p
, RPL_TARGNOTIFY
, form_str(RPL_TARGNOTIFY
),
147 sendto_one(target_p
, form_str(RPL_UMODEGMSG
),
148 me
.name
, target_p
->name
, source_p
->name
,
149 source_p
->username
, source_p
->host
, IsSetStrictCallerID(target_p
) ? "+g" : "+G");
151 target_p
->localClient
->last_caller_id_time
= rb_current_time();
156 add_callerid_accept_for_source(enum message_type msgtype
, struct Client
*source_p
, struct Client
*target_p
)
158 /* only do this on source_p's server */
159 if (!MyClient(source_p
))
163 * XXX: Controversial? Allow target users to send replies
164 * through a +g. Rationale is that people can presently use +g
165 * as a way to taunt users, e.g. harass them and hide behind +g
166 * as a way of griefing. --nenolod
168 if(msgtype
!= MESSAGE_TYPE_NOTICE
&&
169 IsSetAnyCallerID(source_p
) &&
170 !accept_message(target_p
, source_p
) &&
171 !IsOperGeneral(target_p
))
173 if(rb_dlink_list_length(&source_p
->localClient
->allow_list
) <
174 (unsigned long)ConfigFileEntry
.max_accept
)
176 rb_dlinkAddAlloc(target_p
, &source_p
->localClient
->allow_list
);
177 rb_dlinkAddAlloc(source_p
, &target_p
->on_allow_list
);
181 sendto_one_numeric(source_p
, ERR_OWNMODE
,
182 form_str(ERR_OWNMODE
),
183 target_p
->name
, IsSetStrictCallerID(target_p
) ? "+g" : "+G");
192 h_hdl_invite(void *vdata
)
194 hook_data_channel_approval
*data
= vdata
;
195 struct Client
*source_p
= data
->client
;
196 struct Client
*target_p
= data
->target
;
197 static char errorbuf
[BUFSIZE
];
202 if (!add_callerid_accept_for_source(MESSAGE_TYPE_PRIVMSG
, source_p
, target_p
))
204 data
->approved
= ERR_TARGUMODEG
;
208 if (allow_message(source_p
, target_p
))
211 snprintf(errorbuf
, sizeof errorbuf
, form_str(ERR_TARGUMODEG
),
212 target_p
->name
, IsSetStrictCallerID(target_p
) ? "+g" : "+G");
214 data
->approved
= ERR_TARGUMODEG
;
215 data
->error
= errorbuf
;
219 h_hdl_privmsg_user(void *vdata
)
221 hook_data_privmsg_user
*data
= vdata
;
222 enum message_type msgtype
= data
->msgtype
;
223 struct Client
*source_p
= data
->source_p
;
224 struct Client
*target_p
= data
->target_p
;
229 if (!add_callerid_accept_for_source(msgtype
, source_p
, target_p
))
231 data
->approved
= ERR_TARGUMODEG
;
235 if (allow_message(source_p
, target_p
))
238 send_callerid_notice(msgtype
, source_p
, target_p
);
240 data
->approved
= ERR_TARGUMODEG
;
243 static mapi_hfn_list_av1 um_callerid_hfnlist
[] = {
244 { "invite", h_hdl_invite
},
245 { "privmsg_user", h_hdl_privmsg_user
},
249 DECLARE_MODULE_AV2(um_callerid
, um_callerid_modinit
, um_callerid_moddeinit
,
250 NULL
, NULL
, um_callerid_hfnlist
, NULL
, NULL
, um_callerid_desc
);