d1ec72686044489c0d492380762c02c9ed44fae2
[citadel.git] / webcit / webcit.h
1 /*
2  * Copyright (c) 1987-2016 by the citadel.org team
3  *
4  * This program is open source software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License version 3.
6  *
7  * This program is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10  * GNU General Public License for more details.
11  */
12
13 #include "sysdep.h"
14 #include <sys/select.h>
15 #include <ctype.h>
16 #include <stdlib.h>
17 #ifdef HAVE_UNISTD_H
18 #include <unistd.h>
19 #endif
20 #include <stdio.h>
21 #ifdef HAVE_FCNTL_H
22 #include <fcntl.h>
23 #endif
24 #include <signal.h>
25 #include <sys/types.h>
26 #include <sys/wait.h>
27 #include <sys/socket.h>
28 #include <sys/stat.h>
29 #ifdef HAVE_LIMITS_H
30 #include <limits.h>
31 #endif
32 #include <sys/socket.h>
33 #include <netinet/in.h>
34 #include <arpa/inet.h>
35 #include <netdb.h>
36 #include <sys/un.h>
37 #include <sys/poll.h>
38 #include <string.h>
39 #include <pwd.h>
40 #include <errno.h>
41 #include <stdarg.h>
42 #include <pthread.h>
43 #include <signal.h>
44 #include <syslog.h>
45 #include <sys/utsname.h>
46 #include <expat.h>
47 #include <libcitadel.h>
48
49 #ifdef HAVE_ICONV
50 #include <iconv.h>
51 #endif
52
53 #ifdef ENABLE_NLS
54 #ifdef HAVE_XLOCALE_H
55 #include <xlocale.h>
56 #endif
57 #include <libintl.h>
58 #include <locale.h>
59 #define _(string)       gettext(string)
60 #else
61 #define _(string)       (string)
62 #endif
63
64 #define DO_DBG_QR 0
65 #define DBG_QR(x) if(DO_DBG_QR) _DBG_QR(x)
66 #define DBG_QR2(x) if(DO_DBG_QR) _DBG_QR2(x)
67
68 #include <zlib.h>
69
70 #include <libical/ical.h>
71
72 #undef PACKAGE
73 #undef VERSION
74 #undef PACKAGE_NAME
75 #undef PACKAGE_STRING
76 #undef PACKAGE_TARNAME
77 #undef PACKAGE_VERSION
78 #undef PACKAGE_BUGREPORT
79
80 typedef struct wcsession wcsession;
81
82 #include "sysdep.h"
83
84 #include "subst.h"
85 #include "messages.h"
86 #include "paramhandling.h"
87 #include "roomops.h"
88 #include "preferences.h"
89
90 #include "tcp_sockets.h"
91 #include "utils.h"
92 #ifdef HAVE_OPENSSL
93 /* Work around RedHat's b0rken OpenSSL includes */
94 #define OPENSSL_NO_KRB5
95 #include <openssl/ssl.h>
96 #include <openssl/err.h>
97 #include <openssl/rand.h>
98 extern char *ssl_cipher_list;
99 #define DEFAULT_SSL_CIPHER_LIST "DEFAULT"       /* See http://openssl.org/docs/apps/ciphers.html */
100 #endif
101
102 #if SIZEOF_SIZE_T == SIZEOF_INT 
103 #define SIZE_T_FMT "%d"
104 #else
105 #define SIZE_T_FMT "%ld"
106 #endif
107
108 #if SIZEOF_LONG_UNSIGNED_INT == SIZEOF_INT
109 #define ULONG_FMT "%d"
110 #else
111 #define ULONG_FMT "%ld"
112 #endif
113
114 #define CALENDAR_ROOM_NAME      "Calendar"
115 #define PRODID "-//Citadel//NONSGML Citadel Calendar//EN"
116
117 #define SIZ                     4096            /* generic buffer size */
118
119 #define TRACE syslog(LOG_DEBUG, "\033[3%dmCHECKPOINT: %s:%d\033[0m", ((__LINE__%6)+1), __FILE__, __LINE__)
120
121 #ifdef UNDEF_MEMCPY
122 #undef memcpy
123 #endif
124
125 #define SLEEPING                180             /* TCP connection timeout */
126 #define WEBCIT_TIMEOUT          900             /* WebCit session timeout */
127 #define PORT_NUM                2000            /* port number to listen on */
128 #define DEVELOPER_ID            0
129 #define CLIENT_ID               4
130 #define CLIENT_VERSION          902             /* This version of WebCit */
131 #define MINIMUM_CIT_VERSION     902             /* Minimum required version of Citadel server */
132 #define LIBCITADEL_MIN          902             /* Minimum required version of libcitadel */
133 #define DEFAULT_HOST            "localhost"     /* Default Citadel server */
134 #define DEFAULT_PORT            "504"
135 #define TARGET                  "webcit01"      /* Window target for inline URL's */
136 #define HOUSEKEEPING            15              /* Housekeeping frequency */
137 #define MAX_WORKER_THREADS      250
138 #define LISTEN_QUEUE_LENGTH     100             /* listen() backlog queue */
139
140 #define USERCONFIGROOM          "My Citadel Config"
141 #define DEFAULT_MAXMSGS         20
142
143
144 #ifdef LIBCITADEL_VERSION_NUMBER
145 #if LIBCITADEL_VERSION_NUMBER < LIBCITADEL_MIN
146 #error libcitadel is too old.  Please upgrade it before continuing.
147 #endif
148 #endif
149
150
151
152
153 #define SRV_STATUS_MSG(ServerLineBuf) (ChrPtr(ServerLineBuf) + 4), (StrLength(ServerLineBuf) - 4)
154 #define MAJORCODE(a) (((int)(a / 100) ) * 100)
155
156 #define LISTING_FOLLOWS 100
157 #define CIT_OK          200     
158 #define MORE_DATA       300
159 #define SEND_LISTING    400
160 #define ERROR           500
161 #define BINARY_FOLLOWS  600
162 #define SEND_BINARY     700
163 #define START_CHAT_MODE 800
164 #define ASYNC_MSG       900
165
166 #define MINORCODE(a) (a % 100)
167 #define ASYNC_GEXP                      02      
168 #define INTERNAL_ERROR                  10      
169 #define TOO_BIG                         11      
170 #define ILLEGAL_VALUE                   12      
171 #define NOT_LOGGED_IN                   20      
172 #define CMD_NOT_SUPPORTED               30      
173 #define SERVER_SHUTTING_DOWN            31      
174 #define PASSWORD_REQUIRED               40      
175 #define ALREADY_LOGGED_IN               41      
176 #define USERNAME_REQUIRED               42      
177 #define HIGHER_ACCESS_REQUIRED          50      
178 #define MAX_SESSIONS_EXCEEDED           51      
179 #define RESOURCE_BUSY                   52      
180 #define RESOURCE_NOT_OPEN               53      
181 #define NOT_HERE                        60      
182 #define INVALID_FLOOR_OPERATION         61      
183 #define NO_SUCH_USER                    70      
184 #define FILE_NOT_FOUND                  71      
185 #define ROOM_NOT_FOUND                  72      
186 #define NO_SUCH_SYSTEM                  73      
187 #define ALREADY_EXISTS                  74      
188 #define MESSAGE_NOT_FOUND               75
189
190 /*
191  * NLI is the string that shows up in a who's online listing for sessions
192  * that are active but do not (yet) have a user logged in.
193  */
194 #define NLI     "(not logged in)"
195
196 /*
197  * Expiry policy for the autopurger
198  */
199
200 typedef struct __ExpirePolicy {
201         int expire_mode;
202         int expire_value;
203 } ExpirePolicy;
204
205 /*
206  * Linked list of session variables encoded in an x-www-urlencoded content type
207  */
208 typedef struct urlcontent urlcontent;
209 struct urlcontent {
210         char url_key[32];               /* key */
211         long klen;
212         StrBuf *url_data;               /* value */
213         HashList *sub;
214 };
215
216 /*
217  * Information about the Citadel server to which we are connected
218  */ 
219 typedef struct _serv_info {
220         int serv_pid;                   /* Process ID of the Citadel server */
221         StrBuf *serv_nodename;          /* Node name of the Citadel server */
222         StrBuf *serv_humannode;         /* Juman readable node name of the Citadel server */
223         StrBuf *serv_fqdn;              /* Fully qualified Domain Name (such as uncensored.citadel.org) */
224         StrBuf *serv_software;          /* Free form text description of the server software in use */
225         int serv_rev_level;             /* Server version number (times 100) */
226         StrBuf *serv_bbs_city;          /* Geographic location of the Citadel server */
227         StrBuf *serv_sysadm;            /* Name of system administrator */
228         int serv_supports_ldap;         /* is the server linked against an ldap tree for adresses? */
229         int serv_newuser_disabled;      /* Has the server disabled self-service new user creation? */
230         StrBuf *serv_default_cal_zone;  /* Default timezone for unspecified calendar items */
231         int serv_supports_sieve;        /* Server supports Sieve mail filtering */
232         int serv_fulltext_enabled;      /* Full text index is enabled */
233         StrBuf *serv_svn_revision;      /* svn or git revision of the server */
234         int serv_supports_openid;       /* Server supports authentication via OpenID */
235         int serv_supports_guest;        /* Server supports unauthenticated guest logins */
236 } ServInfo;
237
238
239 typedef struct _disp_cal {                                      
240         icalcomponent *cal;             /* cal items for display */
241         long cal_msgnum;                /* cal msgids for display */
242         char *from;                     /* owner of this component */
243         int unread;                     /* already seen by the user? */
244
245         time_t event_start;
246         time_t event_end;
247
248         int multi_day_event;
249         int is_repeat;
250         icalcomponent *SortBy;          /* cal items for display */
251         icalproperty_status Status;
252 } disp_cal;                                             
253
254 typedef struct _IcalKindEnumMap {
255         const char *Name;
256         long NameLen;
257         icalproperty_kind map;
258 } IcalKindEnumMap;
259 typedef struct _IcalMethodEnumMap {
260         const char *Name;
261         long NameLen;
262         icalproperty_method map;
263 } IcalMethodEnumMap;
264
265
266 #define AJAX (1<<0)
267 #define ANONYMOUS (1<<1)
268 #define NEED_URL (1<<2)
269 #define XHTTP_COMMANDS (1<<3)
270 #define BOGUS (1<<4)
271 #define URLNAMESPACE (1<<4)
272 #define LOGCHATTY (1<<5)
273 #define COOKIEUNNEEDED (1<<6)
274 #define ISSTATIC (1<<7)
275 #define FORCE_SESSIONCLOSE (1<<8)
276 #define PARSE_REST_URL (1<<9)
277 #define PROHIBIT_STARTPAGE (1<<10)
278
279
280 #define DATEFMT_FULL 0
281 #define DATEFMT_BRIEF 1
282 #define DATEFMT_RAWDATE 2
283 #define DATEFMT_LOCALEDATE 3
284 long webcit_fmt_date(char *buf, size_t siz, time_t thetime, int Format);
285
286
287 typedef enum _RESTDispatchID {
288         ExistsID,
289         PutID,
290         DeleteID
291 } RESTDispatchID;
292
293 typedef int (*WebcitRESTDispatchID)(RESTDispatchID WhichAction, int IgnoreFloor);
294 typedef void (*WebcitHandlerFunc)(void);
295 typedef struct  _WebcitHandler{
296         WebcitHandlerFunc F;
297         WebcitRESTDispatchID RID;
298         long Flags;
299         StrBuf *Name;
300         StrBuf *DisplayName;
301 } WebcitHandler;
302
303
304 void WebcitAddUrlHandler(const char * UrlString, long UrlSLen, const char *DisplayName, long dslen, WebcitHandlerFunc F, long Flags);
305
306 typedef struct _headereval {
307         ExamineMsgHeaderFunc evaluator;
308         int Type;
309 } headereval;
310
311
312 struct attach_link {
313         char partnum[32];
314         char html[1024];
315 };
316
317
318 enum {
319         eUp,
320         eDown,
321         eNone
322 };
323
324 <<<<<<< HEAD
325 enum {
326         eGET,
327         ePOST,
328         eOPTIONS,
329         ePROPFIND,
330         ePUT,
331         eDELETE,
332         eHEAD,
333         eMOVE,
334         eCOPY,
335         eREPORT,
336         eNONE
337 };
338 extern const char *ReqStrs[eNONE];
339
340 #define NO_AUTH 0
341 #define AUTH_COOKIE 1
342 #define AUTH_BASIC 2
343
344
345
346 typedef struct _HdrRefs {
347         long eReqType;                          /* HTTP method */
348         int desired_session;
349         int SessionKey;
350
351         int got_auth;
352         int DontNeedAuth;
353         long ContentLength;
354         time_t if_modified_since;
355         int gzip_ok;                            /* Nonzero if Accept-encoding: gzip */
356         int prohibit_caching;
357         int dav_depth;
358         int Static;
359
360         /* these are references into Hdr->HTTPHeaders, so we don't need to free them. */
361         StrBuf *ContentType;
362         StrBuf *RawCookie;
363         StrBuf *ReqLine;
364         StrBuf *browser_host;
365         StrBuf *browser_language;
366         StrBuf *user_agent;
367         StrBuf *plainauth;
368         StrBuf *dav_ifmatch;
369
370         const WebcitHandler *Handler;
371 } HdrRefs;
372
373 typedef struct _ParsedHttpHdrs {
374         int http_sock;                          /* HTTP server socket */
375         long HaveRange;
376         long RangeStart;
377         long RangeTil;
378         long TotalBytes;
379         const char *Pos;
380         StrBuf *ReadBuf;
381
382         StrBuf *c_username;
383         StrBuf *c_password;
384         StrBuf *c_roomname;
385         StrBuf *c_language;
386         StrBuf *this_page;                      /* URL of current page */
387         StrBuf *PlainArgs; 
388         StrBuf *HostHeader;
389
390         HashList *urlstrings;                   /* variables passed to webcit in a URL */
391         HashList *HTTPHeaders;                  /* the headers the client sent us */
392         int nWildfireHeaders;                   /* how many wildfire headers did we already send? */
393
394         HdrRefs HR;
395 } ParsedHttpHdrs;
396
397
398 /*
399  * One of these is kept for each active Citadel session.
400  * HTTP transactions are bound to one at a time.
401  */
402 struct wcsession {
403 /* infrastructural members */
404         wcsession *next;                        /* Linked list */
405         pthread_mutex_t SessionMutex;           /* mutex for exclusive access */
406         int wc_session;                         /* WebCit session ID */
407         int killthis;                           /* Nonzero == purge this session */
408         int ctdl_pid;                           /* Session ID on the Citadel server */
409         int nonce;                              /* session nonce (to prevent session riding) */
410         int inuse;                              /* set to nonzero if bound to a running thread */
411         int isFailure;                          /* Http 2xx or 5xx? */
412
413 /* Session local Members */
414         int serv_sock;                          /* Client socket to Citadel server */
415         StrBuf *ReadBuf;                        /* linebuffered reads from the server */
416         StrBuf *MigrateReadLineBuf;             /* here we buffer legacy server read stuff */
417         const char *ReadPos;                    /* whats our read position in ReadBuf? */
418         int last_chat_seq;                      /* When in chat - last message seq# we saw */
419         time_t lastreq;                         /* Timestamp of most recent HTTP */
420         time_t last_pager_check;                /* last time we polled for instant msgs */
421         ServInfo *serv_info;                    /* Information about the citserver we're connected to */
422         StrBuf *PushedDestination;              /* Where to go after login, registration, etc. */
423
424 /* Request local Members */
425         StrBuf *CLineBuf;                       /* linebuffering client stuff */
426         ParsedHttpHdrs *Hdr;
427         StrBuf *WBuf;                           /* Our output buffer */
428         StrBuf *HBuf;                           /* Our HeaderBuffer */
429         StrBuf *WFBuf;                          /* Wildfire error logging buffer */
430         StrBuf *trailing_javascript;            /* extra javascript to be appended to page */
431         StrBuf *ImportantMsg;
432         HashList *Directory;                    /* Parts of the directory URL in snippets */
433         const Floor *CurrentFloor;              /* when Parsing REST, which floor are we on? */
434
435 /* accounting */
436         StrBuf *wc_username;                    /* login name of current user */
437         StrBuf *wc_fullname;                    /* Screen name of current user */
438         StrBuf *wc_password;                    /* Password of current user */
439         StrBuf *httpauth_pass;                  /* only for GroupDAV sessions */
440         int axlevel;                            /* this user's access level */
441         int is_aide;                            /* nonzero == this user is an Admin */
442         int connected;                          /* nonzero == we are connected to Citadel */
443         int logged_in;                          /* nonzero == we are logged in  */
444         int need_regi;                          /* This user needs to register. */
445         int need_vali;                          /* New users require validation. */
446
447 /* Preferences */
448         StrBuf *cs_inet_email;                  /* User's preferred Internet addr. */
449         HashList *hash_prefs;                   /* WebCit preferences for this user */
450         StrBuf *DefaultCharset;                 /* Charset the user preferes */
451         int downloaded_prefs;                   /* Has the client download its prefs yet? */
452         int SavePrefsToServer;                  /* Should we save our preferences to the server at the end of the request? */
453         int selected_language;                  /* Language selected by user */
454         int time_format_cache;                  /* which timeformat does our user like? */
455
456         folder CurRoom;                         /* information about our current room */
457         const folder *ThisRoom;                 /* if REST found a room, remember it here. */
458 /* next/previous room thingabob */
459         struct march *march;                    /* march mode room list */
460         char ugname[128];                       /* where does 'ungoto' take us */
461         long uglsn;                             /* last seen message number for ungoto */
462
463 /* Uploading; mime attachments for composing messages */
464         HashList *attachments;                  /* list of attachments for 'enter message' */
465         int upload_length;                      /* content length of http-uploaded data */
466         StrBuf *upload;                         /* pointer to http-uploaded data */
467         StrBuf *upload_filename;                /* filename of http-uploaded data */
468         char upload_content_type[256];          /* content type of http-uploaded data */
469
470         int remember_new_mail;                  /* last count of new mail messages */
471
472 /* Roomiew control */
473         HashList *Floors;                       /* floors our citserver has hashed numeric for quicker access*/
474         HashList *FloorsByName;                 /* same but hashed by its name */
475         HashList *Rooms;                        /* our directory structure as loaded by LKRA */
476         HashList *summ;                         /* list of messages for mailbox summary view */
477   /** Perhaps these should be within a struct instead */
478         long startmsg;                          /* message number to start at */
479         long maxmsgs;                           /* maximum messages to display */
480         long num_displayed;                     /* number of messages actually displayed */
481         HashList *disp_cal_items;               /* sorted list of calendar items; startdate is the sort criteria. */
482
483
484         char last_chat_user[256];
485
486         StrBuf *IconTheme;                      /* Icontheme setting */
487
488 /* Iconbar controls */
489         int cache_max_folders;
490         int cache_num_floors;
491         long *IBSettingsVec;                    /* which icons should be shown / not shown? */
492         const StrBuf *floordiv_expanded;        /* which floordiv currently expanded */
493         int ib_wholist_expanded;
494         int ib_roomlist_expanded;
495
496 /* our known Sieve scripts; loaded by SIEVE:SCRIPTS iterator. */
497         HashList *KnownSieveScripts;
498
499 /* Transcoding cache buffers; used to avoid to frequent realloc */
500         StrBuf *ConvertBuf1;
501         StrBuf *ConvertBuf2;
502
503 /* cache stuff for templates. TODO: find a smarter way */
504         HashList *ServCfg;                      /* cache our server config for editing */
505         HashList *InetCfg;                      /* Our inet server config for editing */
506         ExpirePolicy Policy[maxpolicy];
507
508 };
509
510
511 typedef void (*Header_Evaluator)(StrBuf *Line, ParsedHttpHdrs *hdr);
512
513 typedef struct _HttpHeader {
514         Header_Evaluator H;
515         StrBuf *Val;
516         int HaveEvaluator;
517 } OneHttpHeader;
518
519 void RegisterHeaderHandler(const char *Name, long Len, Header_Evaluator F);
520
521
522 enum {
523         S_SHUTDOWN,
524         S_SPAWNER,
525         MAX_SEMAPHORES
526 };
527
528 #ifndef num_parms
529 #define num_parms(source)               num_tokens(source, '|') 
530 #endif
531
532 #define site_prefix     (WC ? (WC->Hdr->HostHeader) : NULL)
533
534 /* Per-session data */
535 #define WC ((struct wcsession *)pthread_getspecific(MyConKey))
536 extern pthread_key_t MyConKey;
537
538 /* Per-thread SSL context */
539 #ifdef HAVE_OPENSSL
540 #define THREADSSL ((SSL *)pthread_getspecific(ThreadSSL))
541 extern pthread_key_t ThreadSSL;
542 extern char ctdl_key_dir[PATH_MAX];
543 extern char file_crpt_file_key[PATH_MAX];
544 extern char file_crpt_file_csr[PATH_MAX];
545 extern char file_crpt_file_cer[PATH_MAX];
546
547 void init_ssl(void);
548 void endtls(void);
549 int starttls(int sock);
550 extern SSL_CTX *ssl_ctx;  
551 int client_read_sslbuffer(StrBuf *buf, int timeout);
552 int client_write_ssl(const StrBuf *Buf);
553 #endif
554
555 extern int is_https;
556 extern int follow_xff;
557 extern char *server_cookie;
558 extern char *ctdlhost, *ctdlport;
559 extern char *axdefs[];
560 extern int num_threads_existing;
561 extern int num_threads_executing;
562 extern int setup_wizard;
563 extern char wizard_filename[];
564
565 void InitialiseSemaphores(void);
566 void begin_critical_section(int which_one);
567 void end_critical_section(int which_one);
568
569 void CheckGZipCompressionAllowed(const char *MimeType, long MLen);
570
571 extern void do_404(void);
572 void http_redirect(const char *);
573
574
575 #ifdef UBER_VERBOSE_DEBUGGING
576 #define wc_printf(...) wcc_printf(__FILE__, __FUNCTION__, __LINE__, __VA_ARGS__)
577 void wcc_printf(const char *FILE, const char *FUNCTION, long LINE, const char *format, ...);
578 #else 
579 void wc_printf(const char *format,...)__attribute__((__format__(__printf__,1,2)));
580 #endif
581
582 void hprintf(const char *format,...)__attribute__((__format__(__printf__,1,2)));
583
584 void CheckAuthBasic(ParsedHttpHdrs *hdr);
585 void GetAuthBasic(ParsedHttpHdrs *hdr);
586
587 void sleeeeeeeeeep(int);
588
589 size_t wc_strftime(char *s, size_t max, const char *format, const struct tm *tm);
590 void fmt_time(char *buf, size_t siz, time_t thetime);
591 void httpdate(char *buf, time_t thetime);
592 time_t httpdate_to_timestamp(StrBuf *buf);
593
594
595
596
597 void end_webcit_session(void);
598
599
600
601
602 void cookie_to_stuff(StrBuf *cookie,
603                 int *session,
604                 StrBuf *user,
605                 StrBuf *pass,
606                 StrBuf *room,
607                 StrBuf *language
608 );
609 void locate_host(StrBuf *TBuf, int);
610 void become_logged_in(const StrBuf *user, const StrBuf *pass, StrBuf *serv_response);
611
612 void display_login(void);
613 void display_openids(void);
614 void display_default_landing_page(void);
615 void do_welcome(void);
616
617 void display_reg(int during_login);
618 void display_main_menu(void);
619 void display_aide_menu(void);
620
621 void RegisterEmbeddableMimeType(const char *MimeType, long MTLen, int Priority);
622 void CreateMimeStr(void);
623
624
625 void pop_destination(void);
626
627 void FmOut(StrBuf *Target, const char *align, const StrBuf *Source);
628 void wDumpContent(int);
629
630
631 void PutRequestLocalMem(void *Data, DeleteHashDataFunc DeleteIt);
632
633 void output_headers(    int do_httpheaders,
634                         int do_htmlhead,
635                         int do_room_banner,
636                         int unset_cookies,
637                         int suppress_check,
638                         int cache);
639 void cdataout(char *rawdata);
640
641
642 void url(char *buf, size_t bufsize);
643 void UrlizeText(StrBuf* Target, StrBuf *Source, StrBuf *WrkBuf);
644
645 void display_success(const char *successmessage);
646
647 void shutdown_sessions(void);
648
649
650
651 StrBuf *load_mimepart(long msgnum, char *partnum);
652 void MimeLoadData(wc_mime_attachment *Mime);
653 void do_edit_vcard(long msgnum, char *partnum, 
654                    message_summary *VCMsg,
655                    wc_mime_attachment *VCAtt,
656                    const char *return_to, 
657                    const char *force_room);
658
659 void select_user_to_edit(const char *preselect);
660
661 void convenience_page(const char *titlebarcolor, const char *titlebarmsg, const char *messagetext);
662 void output_html(const char *, int, int, StrBuf *, StrBuf *);
663
664 ssize_t write(int fd, const void *buf, size_t count);
665 void cal_process_attachment(wc_mime_attachment *Mime);
666
667 void begin_ajax_response(void);
668 void end_ajax_response(void);
669
670 extern char *months[];
671 extern char *days[];
672 long locate_user_vcard_in_this_room(message_summary **VCMsg,
673                                     wc_mime_attachment **VCAtt);
674 void http_transmit_thing(const char *content_type, int is_static);
675 void http_transmit_headers(const char *content_type, int is_static, long is_chunked, int is_gzip);
676 long unescape_input(char *buf);
677 void check_thread_pool_size(void);
678 void StrEndTab(StrBuf *Target, int tabnum, int num_tabs);
679 void StrBeginTab(StrBuf *Target, int tabnum, int num_tabs, StrBuf **Names);
680 void StrTabbedDialog(StrBuf *Target, int num_tabs, StrBuf *tabnames[]);
681 void tabbed_dialog(int num_tabs, const char *tabnames[]);
682 void begin_tab(int tabnum, int num_tabs);
683 void end_tab(int tabnum, int num_tabs);
684
685 int get_time_format_cached (void);
686 void display_wiki_pagelist(void);
687 void str_wiki_index(StrBuf *);
688
689 HashList *GetRoomListHashLKRA(StrBuf *Target, WCTemplputParams *TP);
690
691 /* actual supported locales */
692 void TmplGettext(StrBuf *Target, WCTemplputParams *TP);
693
694 void set_selected_language(const char *);
695 void go_selected_language(void);
696 const char *get_selected_language(void);
697
698 void utf8ify_rfc822_string(char **buf);
699
700 void begin_burst(void);
701 long end_burst(void);
702
703 void AppendImportantMessage(const char *pch, long len);
704
705 void http_datestring(char *buf, size_t n, time_t xtime);
706
707
708 /* These should be empty, but we have them for testing */
709 #define DEFAULT_HTTPAUTH_USER   ""
710 #define DEFAULT_HTTPAUTH_PASS   ""
711
712
713 /* Exit codes 101 through 109 are initialization failures so we don't want to
714  * just keep respawning indefinitely.
715  */
716 #define WC_EXIT_BIND            101     /* Can't bind to the port */
717 #define WC_EXIT_SSL             102     /* Can't initialize SSL */
718
719
720 #define WC_TIMEFORMAT_NONE 0
721 #define WC_TIMEFORMAT_AMPM 1
722 #define WC_TIMEFORMAT_24 2
723
724 extern int time_to_die;                 /* Nonzero if server is shutting down */
725 extern int DisableGzip;
726
727 void display_summary_page(void);
728
729 HashList *GetValidDomainNames(StrBuf *Target, WCTemplputParams *TP);
730 void output_error_pic(const char *ErrMsg1, const char *ErrMsg2);
731 =======
732 #define TRACE syslog(LOG_DEBUG, "\033[3%dmCHECKPOINT: %s:%d\033[0m", ((__LINE__%6)+1), __FILE__, __LINE__)
733 #define SLEEPING                180             // TCP connection timeout
734 #define MAX_WORKER_THREADS      32              // Maximum number of worker threads permitted to exist
735
736 int webserver(char *webserver_interface, int webserver_port, int webserver_protocol);
737 int webcit_tcp_server(const char *ip_addr, int port_number, int queue_len);
738 int webcit_uds_server(char *sockpath, int queue_len);
739 int lingering_close(int fd);
740 void perform_one_http_transaction(int ssock);
741 void worker_entry(int *pointer_to_master_socket);
742 >>>>>>> af64ffb... textclient: don't try to stat citadel.config; it doesn't exist anymore.