1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
|
.TH "users.cpp" 3 "19 Dec 2005" "Version 1.0Betareleases" "InspIRCd" \" -*- nroff -*-
.ad l
.nh
.SH NAME
users.cpp \-
.SH SYNOPSIS
.br
.PP
\fC#include 'inspircd_config.h'\fP
.br
\fC#include 'channels.h'\fP
.br
\fC#include 'connection.h'\fP
.br
\fC#include 'users.h'\fP
.br
\fC#include 'inspircd.h'\fP
.br
\fC#include <stdio.h>\fP
.br
\fC#include 'inspstring.h'\fP
.br
\fC#include 'commands.h'\fP
.br
\fC#include 'helperfuncs.h'\fP
.br
\fC#include 'typedefs.h'\fP
.br
\fC#include 'socketengine.h'\fP
.br
\fC#include 'hashcomp.h'\fP
.br
\fC#include 'message.h'\fP
.br
\fC#include 'wildcard.h'\fP
.br
\fC#include 'xline.h'\fP
.br
.SS "Functions"
.in +1c
.ti -1c
.RI "template<typename T> \fBstring\fP \fBConvToStr\fP (const T &in)"
.br
.ti -1c
.RI "void \fBAddOper\fP (\fBuserrec\fP *user)"
.br
.ti -1c
.RI "void \fBDeleteOper\fP (\fBuserrec\fP *user)"
.br
.ti -1c
.RI "void \fBkill_link\fP (\fBuserrec\fP *user, const char *r)"
.br
.ti -1c
.RI "void \fBkill_link_silent\fP (\fBuserrec\fP *user, const char *r)"
.br
.ti -1c
.RI "void \fBAddWhoWas\fP (\fBuserrec\fP *u)"
.br
.ti -1c
.RI "void \fBAddClient\fP (int socket, char *host, int port, bool iscached, char *ip)"
.br
.ti -1c
.RI "void \fBFullConnectUser\fP (\fBuserrec\fP *user)"
.br
.ti -1c
.RI "void \fBConnectUser\fP (\fBuserrec\fP *user)"
.br
.ti -1c
.RI "\fBuserrec\fP * \fBReHashNick\fP (char *Old, char *New)"
.br
.ti -1c
.RI "void \fBforce_nickchange\fP (\fBuserrec\fP *user, const char *newnick)"
.br
.in -1c
.SS "Variables"
.in +1c
.ti -1c
.RI "\fBInspIRCd\fP * \fBServerInstance\fP"
.br
.ti -1c
.RI "int \fBWHOWAS_STALE\fP"
.br
.ti -1c
.RI "int \fBWHOWAS_MAX\fP"
.br
.ti -1c
.RI "std::vector< \fBModule\fP * > \fBmodules\fP"
.br
.ti -1c
.RI "std::vector< \fBircd_module\fP * > \fBfactory\fP"
.br
.ti -1c
.RI "std::vector< \fBInspSocket\fP * > \fBmodule_sockets\fP"
.br
.ti -1c
.RI "int \fBMODCOUNT\fP"
.br
.ti -1c
.RI "\fBInspSocket\fP * \fBsocket_ref\fP [65535]"
.br
.ti -1c
.RI "time_t \fBTIME\fP"
.br
.ti -1c
.RI "\fBuserrec\fP * \fBfd_ref_table\fP [65536]"
.br
.ti -1c
.RI "\fBServerConfig\fP * \fBConfig\fP"
.br
.ti -1c
.RI "\fBuser_hash\fP \fBclientlist\fP"
.br
.ti -1c
.RI "\fBwhowas_hash\fP \fBwhowas\fP"
.br
.ti -1c
.RI "std::vector< \fBuserrec\fP * > \fBlocal_users\fP"
.br
.ti -1c
.RI "std::vector< \fBuserrec\fP * > \fBall_opers\fP"
.br
.in -1c
.SH "Function Documentation"
.PP
.SS "void AddClient (int socket, char * host, int port, bool iscached, char * ip)"
.PP
Definition at line 524 of file users.cpp.
.PP
References SocketEngine::AddFd(), CC_ALLOW, ucrec::channel, ServerConfig::Classes, clientlist, ConvToStr(), DEBUG, ServerConfig::dns_timeout, FindServerNamePtr(), kill_link(), local_users, log(), matches_exception(), matches_zline(), InspIRCd::SE, ServerConfig::ServerName, ServerConfig::SoftLimit, TIME, ucrec::uc_modes, and X_ESTAB_CLIENT.
.PP
.nf
525 {
526 string tempnick;
527 char tn2[MAXBUF];
528 user_hash::iterator iter;
529
530 tempnick = ConvToStr(socket) + '-unknown';
531 sprintf(tn2,'%lu-unknown',(unsigned long)socket);
532
533 iter = clientlist.find(tempnick);
534
535 // fix by brain.
536 // as these nicknames are 'RFC impossible', we can be sure nobody is going to be
537 // using one as a registered connection. As theyre per fd, we can also safely assume
538 // that we wont have collisions. Therefore, if the nick exists in the list, its only
539 // used by a dead socket, erase the iterator so that the new client may reclaim it.
540 // this was probably the cause of 'server ignores me when i hammer it with reconnects'
541 // issue in earlier alphas/betas
542 if (iter != clientlist.end())
543 {
544 userrec* goner = iter->second;
545 delete goner;
546 clientlist.erase(iter);
547 }
548
549 /*
550 * It is OK to access the value here this way since we know
551 * it exists, we just created it above.
552 *
553 * At NO other time should you access a value in a map or a
554 * hash_map this way.
555 */
556 clientlist[tempnick] = new userrec();
557
558 log(DEBUG,'AddClient: %lu %s %d %s',(unsigned long)socket,host,port,ip);
559
560 clientlist[tempnick]->fd = socket;
561 strlcpy(clientlist[tempnick]->nick, tn2,NICKMAX);
562 strlcpy(clientlist[tempnick]->host, host,160);
563 strlcpy(clientlist[tempnick]->dhost, host,160);
564 clientlist[tempnick]->server = (char*)FindServerNamePtr(Config->ServerName);
565 strlcpy(clientlist[tempnick]->ident, 'unknown',IDENTMAX);
566 clientlist[tempnick]->registered = 0;
567 clientlist[tempnick]->signon = TIME + Config->dns_timeout;
568 clientlist[tempnick]->lastping = 1;
569 clientlist[tempnick]->port = port;
570 strlcpy(clientlist[tempnick]->ip,ip,16);
571
572 // set the registration timeout for this user
573 unsigned long class_regtimeout = 90;
574 int class_flood = 0;
575 long class_threshold = 5;
576 long class_sqmax = 262144; // 256kb
577 long class_rqmax = 4096; // 4k
578
579 for (ClassVector::iterator i = Config->Classes.begin(); i != Config->Classes.end(); i++)
580 {
581 if (match(clientlist[tempnick]->host,i->host) && (i->type == CC_ALLOW))
582 {
583 class_regtimeout = (unsigned long)i->registration_timeout;
584 class_flood = i->flood;
585 clientlist[tempnick]->pingmax = i->pingtime;
586 class_threshold = i->threshold;
587 class_sqmax = i->sendqmax;
588 class_rqmax = i->recvqmax;
589 break;
590 }
591 }
592
593 clientlist[tempnick]->nping = TIME+clientlist[tempnick]->pingmax + Config->dns_timeout;
594 clientlist[tempnick]->timeout = TIME+class_regtimeout;
595 clientlist[tempnick]->flood = class_flood;
596 clientlist[tempnick]->threshold = class_threshold;
597 clientlist[tempnick]->sendqmax = class_sqmax;
598 clientlist[tempnick]->recvqmax = class_rqmax;
599
600 ucrec a;
601 a.channel = NULL;
602 a.uc_modes = 0;
603 for (int i = 0; i < MAXCHANS; i++)
604 clientlist[tempnick]->chans.push_back(a);
605
606 if (clientlist.size() > Config->SoftLimit)
607 {
608 kill_link(clientlist[tempnick],'No more connections allowed');
609 return;
610 }
611
612 if (clientlist.size() >= MAXCLIENTS)
613 {
614 kill_link(clientlist[tempnick],'No more connections allowed');
615 return;
616 }
617
618 // this is done as a safety check to keep the file descriptors within range of fd_ref_table.
619 // its a pretty big but for the moment valid assumption:
620 // file descriptors are handed out starting at 0, and are recycled as theyre freed.
621 // therefore if there is ever an fd over 65535, 65536 clients must be connected to the
622 // irc server at once (or the irc server otherwise initiating this many connections, files etc)
623 // which for the time being is a physical impossibility (even the largest networks dont have more
624 // than about 10,000 users on ONE server!)
625 if ((unsigned)socket > 65534)
626 {
627 kill_link(clientlist[tempnick],'Server is full');
628 return;
629 }
630 char* e = matches_exception(ip);
631 if (!e)
632 {
633 char* r = matches_zline(ip);
634 if (r)
635 {
636 char reason[MAXBUF];
637 snprintf(reason,MAXBUF,'Z-Lined: %s',r);
638 kill_link(clientlist[tempnick],reason);
639 return;
640 }
641 }
642 fd_ref_table[socket] = clientlist[tempnick];
643 local_users.push_back(clientlist[tempnick]);
644 ServerInstance->SE->AddFd(socket,true,X_ESTAB_CLIENT);
645 }
.fi
.PP
.SS "void AddOper (\fBuserrec\fP * user)"
.PP
Definition at line 330 of file users.cpp.
.PP
References all_opers, DEBUG, and log().
.PP
.nf
331 {
332 log(DEBUG,'Oper added to optimization list');
333 all_opers.push_back(user);
334 }
.fi
.PP
.SS "void AddWhoWas (\fBuserrec\fP * u)"
.PP
Definition at line 471 of file users.cpp.
.PP
References DEBUG, WhoWasUser::dhost, userrec::dhost, WhoWasUser::fullname, userrec::fullname, WhoWasUser::host, connection::host, WhoWasUser::ident, userrec::ident, log(), userrec::nick, WhoWasUser::nick, WhoWasUser::server, userrec::server, WhoWasUser::signon, connection::signon, TIME, whowas, WHOWAS_MAX, and WHOWAS_STALE.
.PP
Referenced by kill_link().
.PP
.nf
472 {
473 whowas_hash::iterator iter = whowas.find(u->nick);
474 WhoWasUser *a = new WhoWasUser();
475 strlcpy(a->nick,u->nick,NICKMAX);
476 strlcpy(a->ident,u->ident,IDENTMAX);
477 strlcpy(a->dhost,u->dhost,160);
478 strlcpy(a->host,u->host,160);
479 strlcpy(a->fullname,u->fullname,MAXGECOS);
480 strlcpy(a->server,u->server,256);
481 a->signon = u->signon;
482
483 /* MAX_WHOWAS: max number of /WHOWAS items
484 * WHOWAS_STALE: number of hours before a WHOWAS item is marked as stale and
485 * can be replaced by a newer one
486 */
487
488 if (iter == whowas.end())
489 {
490 if (whowas.size() >= (unsigned)WHOWAS_MAX)
491 {
492 for (whowas_hash::iterator i = whowas.begin(); i != whowas.end(); i++)
493 {
494 // 3600 seconds in an hour ;)
495 if ((i->second->signon)<(TIME-(WHOWAS_STALE*3600)))
496 {
497 // delete the old one
498 if (i->second) delete i->second;
499 // replace with new one
500 i->second = a;
501 log(DEBUG,'added WHOWAS entry, purged an old record');
502 return;
503 }
504 }
505 // no space left and user doesnt exist. Don't leave ram in use!
506 log(DEBUG,'Not able to update whowas (list at WHOWAS_MAX entries and trying to add new?), freeing excess ram');
507 delete a;
508 }
509 else
510 {
511 log(DEBUG,'added fresh WHOWAS entry');
512 whowas[a->nick] = a;
513 }
514 }
515 else
516 {
517 log(DEBUG,'updated WHOWAS entry');
518 if (iter->second) delete iter->second;
519 iter->second = a;
520 }
521 }
.fi
.PP
.SS "void ConnectUser (\fBuserrec\fP * user)"
.PP
Definition at line 731 of file users.cpp.
.PP
References userrec::dns_done, FullConnectUser(), and connection::registered.
.PP
.nf
732 {
733 // dns is already done, things are fast. no need to wait for dns to complete just pass them straight on
734 if ((user->dns_done) && (user->registered >= 3) && (AllModulesReportReady(user)))
735 {
736 FullConnectUser(user);
737 }
738 }
.fi
.PP
.SS "template<typename T> \fBstring\fP ConvToStr (const T & in)\fC [inline]\fP"
.PP
Definition at line 56 of file users.cpp.
.PP
Referenced by AddClient().
.PP
.nf
57 {
58 stringstream tmp;
59 if (!(tmp << in)) return string();
60 return tmp.str();
61 }
.fi
.PP
.SS "void DeleteOper (\fBuserrec\fP * user)"
.PP
Definition at line 336 of file users.cpp.
.PP
References all_opers, DEBUG, and log().
.PP
.nf
337 {
338 for (std::vector<userrec*>::iterator a = all_opers.begin(); a < all_opers.end(); a++)
339 {
340 if (*a == user)
341 {
342 log(DEBUG,'Oper removed from optimization list');
343 all_opers.erase(a);
344 return;
345 }
346 }
347 }
.fi
.PP
.SS "void force_nickchange (\fBuserrec\fP * user, const char * newnick)"
.PP
Definition at line 769 of file users.cpp.
.PP
References FOREACH_RESULT, kill_link(), matches_qline(), InspIRCd::Parser, connection::registered, InspIRCd::stats, and serverstats::statsCollisions.
.PP
Referenced by Server::ChangeUserNick().
.PP
.nf
770 {
771 char nick[MAXBUF];
772 int MOD_RESULT = 0;
773
774 strcpy(nick,'');
775
776 FOREACH_RESULT(OnUserPreNick(user,newnick));
777 if (MOD_RESULT) {
778 ServerInstance->stats->statsCollisions++;
779 kill_link(user,'Nickname collision');
780 return;
781 }
782 if (matches_qline(newnick))
783 {
784 ServerInstance->stats->statsCollisions++;
785 kill_link(user,'Nickname collision');
786 return;
787 }
788
789 if (user)
790 {
791 if (newnick)
792 {
793 strncpy(nick,newnick,MAXBUF);
794 }
795 if (user->registered == 7)
796 {
797 char* pars[1];
798 pars[0] = nick;
799 std::string cmd = 'NICK';
800 ServerInstance->Parser->CallHandler(cmd,pars,1,user);
801 }
802 }
803 }
.fi
.PP
.SS "void FullConnectUser (\fBuserrec\fP * user)"
.PP
Definition at line 647 of file users.cpp.
.PP
References DEBUG, connection::fd, FOREACH_MOD, connection::haspassed, connection::host, userrec::ident, connection::idle_lastmsg, connection::ip, kill_link(), kill_link_silent(), log(), matches_exception(), matches_gline(), matches_kline(), ServerConfig::Network, userrec::nick, connection::port, connection::registered, ServerConfig::ServerName, InspIRCd::stats, serverstats::statsConnects, TIME, WriteOpers(), and WriteServ().
.PP
Referenced by ConnectUser().
.PP
.nf
648 {
649 ServerInstance->stats->statsConnects++;
650 user->idle_lastmsg = TIME;
651 log(DEBUG,'ConnectUser: %s',user->nick);
652
653 if ((strcmp(Passwd(user),'')) && (!user->haspassed))
654 {
655 kill_link(user,'Invalid password');
656 return;
657 }
658 if (IsDenied(user))
659 {
660 kill_link(user,'Unauthorised connection');
661 return;
662 }
663
664 char match_against[MAXBUF];
665 snprintf(match_against,MAXBUF,'%s@%s',user->ident,user->host);
666 char* e = matches_exception(match_against);
667 if (!e)
668 {
669 char* r = matches_gline(match_against);
670 if (r)
671 {
672 char reason[MAXBUF];
673 snprintf(reason,MAXBUF,'G-Lined: %s',r);
674 kill_link_silent(user,reason);
675 return;
676 }
677 r = matches_kline(user->host);
678 if (r)
679 {
680 char reason[MAXBUF];
681 snprintf(reason,MAXBUF,'K-Lined: %s',r);
682 kill_link_silent(user,reason);
683 return;
684 }
685 }
686
687
688 WriteServ(user->fd,'NOTICE Auth :Welcome to \002%s\002!',Config->Network);
689 WriteServ(user->fd,'001 %s :Welcome to the %s IRC Network %s!%s@%s',user->nick,Config->Network,user->nick,user->ident,user->host);
690 WriteServ(user->fd,'002 %s :Your host is %s, running version %s',user->nick,Config->ServerName,VERSION);
691 WriteServ(user->fd,'003 %s :This server was created %s %s',user->nick,__TIME__,__DATE__);
692 WriteServ(user->fd,'004 %s %s %s iowghraAsORVSxNCWqBzvdHtGI lvhopsmntikrRcaqOALQbSeKVfHGCuzN',user->nick,Config->ServerName,VERSION);
693 // the neatest way to construct the initial 005 numeric, considering the number of configure constants to go in it...
694 std::stringstream v;
695 v << 'WALLCHOPS MODES=13 CHANTYPES=# PREFIX=(ohv)@%+ MAP SAFELIST MAXCHANNELS=' << MAXCHANS;
696 v << ' MAXBANS=60 NICKLEN=' << NICKMAX;
697 v << ' TOPICLEN=' << MAXTOPIC << ' KICKLEN=' << MAXKICK << ' MAXTARGETS=20 AWAYLEN=' << MAXAWAY << ' CHANMODES=ohvb,k,l,psmnti NETWORK=';
698 v << Config->Network;
699 std::string data005 = v.str();
700 FOREACH_MOD On005Numeric(data005);
701 // anfl @ #ratbox, efnet reminded me that according to the RFC this cant contain more than 13 tokens per line...
702 // so i'd better split it :)
703 std::stringstream out(data005);
704 std::string token = '';
705 std::string line5 = '';
706 int token_counter = 0;
707 while (!out.eof())
708 {
709 out >> token;
710 line5 = line5 + token + ' ';
711 token_counter++;
712 if ((token_counter >= 13) || (out.eof() == true))
713 {
714 WriteServ(user->fd,'005 %s %s:are supported by this server',user->nick,line5.c_str());
715 line5 = '';
716 token_counter = 0;
717 }
718 }
719 ShowMOTD(user);
720
721 // fix 3 by brain, move registered = 7 below these so that spurious modes and host changes dont go out
722 // onto the network and produce 'fake direction'
723 FOREACH_MOD OnUserConnect(user);
724 FOREACH_MOD OnGlobalConnect(user);
725 user->registered = 7;
726 WriteOpers('*** Client connecting on port %lu: %s!%s@%s [%s]',(unsigned long)user->port,user->nick,user->ident,user->host,user->ip);
727 }
.fi
.PP
.SS "void kill_link (\fBuserrec\fP * user, const char * r)"
.PP
Definition at line 349 of file users.cpp.
.PP
References AddWhoWas(), clientlist, userrec::CloseSocket(), DEBUG, SocketEngine::DelFd(), connection::fd, userrec::FlushWriteBuf(), FOREACH_MOD, ServerConfig::GetIOHook(), connection::host, userrec::ident, local_users, log(), userrec::nick, Module::OnRawSocketClose(), connection::port, connection::registered, InspIRCd::SE, Write(), WriteCommonExcept(), and WriteOpers().
.PP
Referenced by AddClient(), force_nickchange(), FullConnectUser(), Server::PseudoToUser(), and Server::QuitUser().
.PP
.nf
350 {
351 user_hash::iterator iter = clientlist.find(user->nick);
352
353 char reason[MAXBUF];
354
355 strncpy(reason,r,MAXBUF);
356
357 if (strlen(reason)>MAXQUIT)
358 {
359 reason[MAXQUIT-1] = '\0';
360 }
361
362 log(DEBUG,'kill_link: %s '%s'',user->nick,reason);
363 Write(user->fd,'ERROR :Closing link (%s@%s) [%s]',user->ident,user->host,reason);
364 log(DEBUG,'closing fd %lu',(unsigned long)user->fd);
365
366 if (user->registered == 7) {
367 FOREACH_MOD OnUserQuit(user,reason);
368 WriteCommonExcept(user,'QUIT :%s',reason);
369 }
370
371 user->FlushWriteBuf();
372
373 FOREACH_MOD OnUserDisconnect(user);
374
375 if (user->fd > -1)
376 {
377 if (Config->GetIOHook(user->port))
378 {
379 Config->GetIOHook(user->port)->OnRawSocketClose(user->fd);
380 }
381 ServerInstance->SE->DelFd(user->fd);
382 user->CloseSocket();
383 }
384
385 // this must come before the WriteOpers so that it doesnt try to fill their buffer with anything
386 // if they were an oper with +s.
387 if (user->registered == 7) {
388 purge_empty_chans(user);
389 // fix by brain: only show local quits because we only show local connects (it just makes SENSE)
390 if (user->fd > -1)
391 WriteOpers('*** Client exiting: %s!%s@%s [%s]',user->nick,user->ident,user->host,reason);
392 AddWhoWas(user);
393 }
394
395 if (iter != clientlist.end())
396 {
397 log(DEBUG,'deleting user hash value %lu',(unsigned long)user);
398 if (user->fd > -1)
399 {
400 fd_ref_table[user->fd] = NULL;
401 if (find(local_users.begin(),local_users.end(),user) != local_users.end())
402 {
403 local_users.erase(find(local_users.begin(),local_users.end(),user));
404 log(DEBUG,'Delete local user');
405 }
406 }
407 clientlist.erase(iter);
408 }
409 delete user;
410 }
.fi
.PP
.SS "void kill_link_silent (\fBuserrec\fP * user, const char * r)"
.PP
Definition at line 412 of file users.cpp.
.PP
References clientlist, userrec::CloseSocket(), DEBUG, SocketEngine::DelFd(), connection::fd, userrec::FlushWriteBuf(), FOREACH_MOD, ServerConfig::GetIOHook(), connection::host, userrec::ident, local_users, log(), userrec::nick, Module::OnRawSocketClose(), connection::port, connection::registered, InspIRCd::SE, Write(), and WriteCommonExcept().
.PP
Referenced by FullConnectUser().
.PP
.nf
413 {
414 user_hash::iterator iter = clientlist.find(user->nick);
415
416 char reason[MAXBUF];
417
418 strncpy(reason,r,MAXBUF);
419
420 if (strlen(reason)>MAXQUIT)
421 {
422 reason[MAXQUIT-1] = '\0';
423 }
424
425 log(DEBUG,'kill_link: %s '%s'',user->nick,reason);
426 Write(user->fd,'ERROR :Closing link (%s@%s) [%s]',user->ident,user->host,reason);
427 log(DEBUG,'closing fd %lu',(unsigned long)user->fd);
428
429 user->FlushWriteBuf();
430
431 if (user->registered == 7) {
432 FOREACH_MOD OnUserQuit(user,reason);
433 WriteCommonExcept(user,'QUIT :%s',reason);
434 }
435
436 FOREACH_MOD OnUserDisconnect(user);
437
438 if (user->fd > -1)
439 {
440 if (Config->GetIOHook(user->port))
441 {
442 Config->GetIOHook(user->port)->OnRawSocketClose(user->fd);
443 }
444 ServerInstance->SE->DelFd(user->fd);
445 user->CloseSocket();
446 }
447
448 if (user->registered == 7) {
449 purge_empty_chans(user);
450 }
451
452 if (iter != clientlist.end())
453 {
454 log(DEBUG,'deleting user hash value %lu',(unsigned long)user);
455 if (user->fd > -1)
456 {
457 fd_ref_table[user->fd] = NULL;
458 if (find(local_users.begin(),local_users.end(),user) != local_users.end())
459 {
460 log(DEBUG,'Delete local user');
461 local_users.erase(find(local_users.begin(),local_users.end(),user));
462 }
463 }
464 clientlist.erase(iter);
465 }
466 delete user;
467 }
.fi
.PP
.SS "\fBuserrec\fP* ReHashNick (char * Old, char * New)"
.PP
Definition at line 743 of file users.cpp.
.PP
References clientlist, DEBUG, and log().
.PP
.nf
744 {
745 //user_hash::iterator newnick;
746 user_hash::iterator oldnick = clientlist.find(Old);
747
748 log(DEBUG,'ReHashNick: %s %s',Old,New);
749
750 if (!strcasecmp(Old,New))
751 {
752 log(DEBUG,'old nick is new nick, skipping');
753 return oldnick->second;
754 }
755
756 if (oldnick == clientlist.end()) return NULL; /* doesnt exist */
757
758 log(DEBUG,'ReHashNick: Found hashed nick %s',Old);
759
760 userrec* olduser = oldnick->second;
761 clientlist[New] = olduser;
762 clientlist.erase(oldnick);
763
764 log(DEBUG,'ReHashNick: Nick rehashed as %s',New);
765
766 return clientlist[New];
767 }
.fi
.PP
.SH "Variable Documentation"
.PP
.SS "std::vector<\fBuserrec\fP*> \fBall_opers\fP"
.PP
Definition at line 54 of file users.cpp.
.PP
Referenced by AddOper(), and DeleteOper().
.SS "\fBuser_hash\fP \fBclientlist\fP"
.PP
.SS "\fBServerConfig\fP* \fBConfig\fP"
.PP
.SS "std::vector<\fBircd_module\fP*> factory"
.PP
.SS "\fBuserrec\fP* \fBfd_ref_table\fP[65536]"
.PP
.SS "std::vector<\fBuserrec\fP*> \fBlocal_users\fP"
.PP
Definition at line 52 of file users.cpp.
.PP
Referenced by AddClient(), kill_link(), and kill_link_silent().
.SS "int \fBMODCOUNT\fP"
.PP
.SS "std::vector<\fBInspSocket\fP*> \fBmodule_sockets\fP"
.PP
.SS "std::vector<\fBModule\fP*> modules"
.PP
.SS "\fBInspIRCd\fP* \fBServerInstance\fP"
.PP
.SS "\fBInspSocket\fP* \fBsocket_ref\fP[65535]"
.PP
Definition at line 43 of file socket.cpp.
.SS "time_t \fBTIME\fP"
.PP
.SS "\fBwhowas_hash\fP \fBwhowas\fP"
.PP
Referenced by AddWhoWas().
.SS "int \fBWHOWAS_MAX\fP"
.PP
.SS "int \fBWHOWAS_STALE\fP"
.PP
.SH "Author"
.PP
Generated automatically by Doxygen for InspIRCd from the source code.
|