2 * Copyright (c) 1987-2011 by the citadel.org team
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 as published
6 * by the Free Software Foundation; either version 3 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 #include <sys/select.h>
31 #include <sys/types.h>
33 #include <sys/socket.h>
34 #ifdef HAVE_SYS_TIME_H
41 #include <sys/socket.h>
42 #include <netinet/in.h>
43 #include <arpa/inet.h>
54 #include <sys/utsname.h>
56 #include <libcitadel.h>
68 #define _(string) gettext(string)
70 #define _(string) (string)
73 #define IsEmptyStr(a) ((a)[0] == '\0')
76 #define DBG_QR(x) if(DO_DBG_QR) _DBG_QR(x)
77 #define DBG_QR2(x) if(DO_DBG_QR) _DBG_QR2(x)
81 #include <libical/ical.h>
87 #undef PACKAGE_TARNAME
88 #undef PACKAGE_VERSION
89 #undef PACKAGE_BUGREPORT
91 typedef struct wcsession wcsession;
97 #include "paramhandling.h"
99 #include "preferences.h"
101 #include "tcp_sockets.h"
104 /* Work around RedHat's b0rken OpenSSL includes */
105 #define OPENSSL_NO_KRB5
106 #include <openssl/ssl.h>
107 #include <openssl/err.h>
108 #include <openssl/rand.h>
109 extern char *ssl_cipher_list;
110 #define DEFAULT_SSL_CIPHER_LIST "DEFAULT" /* See http://openssl.org/docs/apps/ciphers.html */
113 #if SIZEOF_SIZE_T == SIZEOF_INT
114 #define SIZE_T_FMT "%d"
116 #define SIZE_T_FMT "%ld"
119 #if SIZEOF_LONG_UNSIGNED_INT == SIZEOF_INT
120 #define ULONG_FMT "%d"
122 #define ULONG_FMT "%ld"
125 #define CALENDAR_ROOM_NAME "Calendar"
126 #define PRODID "-//Citadel//NONSGML Citadel Calendar//EN"
128 #define SIZ 4096 /* generic buffer size */
130 #define TRACE syslog(LOG_DEBUG, "\033[3%dmCHECKPOINT: %s:%d\033[0m", ((__LINE__%6)+1), __FILE__, __LINE__)
132 #define SLEEPING 180 /* TCP connection timeout */
133 #define WEBCIT_TIMEOUT 900 /* WebCit session timeout */
134 #define PORT_NUM 2000 /* port number to listen on */
135 #define DEVELOPER_ID 0
137 #define CLIENT_VERSION 802 /* This version of WebCit */
138 #define MINIMUM_CIT_VERSION 802 /* min required Citadel ver */
139 #define LIBCITADEL_MIN 802 /* min required libcitadel ver */
140 #define DEFAULT_HOST "localhost" /* Default Citadel server */
141 #define DEFAULT_PORT "504"
142 #define TARGET "webcit01" /* Target for inline URL's */
143 #define HOUSEKEEPING 15 /* Housekeeping frequency */
144 #define MAX_WORKER_THREADS 250
145 #define LISTEN_QUEUE_LENGTH 100 /* listen() backlog queue */
147 #define USERCONFIGROOM "My Citadel Config"
148 #define DEFAULT_MAXMSGS 20
151 #ifdef LIBCITADEL_VERSION_NUMBER
152 #if LIBCITADEL_VERSION_NUMBER < LIBCITADEL_MIN
153 #error libcitadel is too old. Please upgrade it before continuing.
159 * Room flags (from Citadel)
163 #define QR_PERMANENT 1 /* Room does not purge */
164 #define QR_INUSE 2 /* Set if in use, clear if avail */
165 #define QR_PRIVATE 4 /* Set for any type of private room */
166 #define QR_PASSWORDED 8 /* Set if there's a password too */
167 #define QR_GUESSNAME 16 /* Set if it's a guessname room */
168 #define QR_DIRECTORY 32 /* Directory room */
169 #define QR_UPLOAD 64 /* Allowed to upload */
170 #define QR_DOWNLOAD 128 /* Allowed to download */
171 #define QR_VISDIR 256 /* Visible directory */
172 #define QR_ANONONLY 512 /* Anonymous-Only room */
173 #define QR_ANONOPT 1024 /* Anonymous-Option room */
174 #define QR_NETWORK 2048 /* Shared network room */
175 #define QR_PREFONLY 4096 /* Preferred status needed to enter */
176 #define QR_READONLY 8192 /* Aide status required to post */
177 #define QR_MAILBOX 16384 /* Set if this is a private mailbox */
182 #define QR2_SYSTEM 1 /* System room; hide by default */
183 #define QR2_SELFLIST 2 /* Self-service mailing list mgmt */
184 #define QR2_COLLABDEL 4 /* Anyone who can post can also delete */
185 #define QR2_SUBJECTREQ 8 /* Subject strongly recommended */
186 #define QR2_SMTP_PUBLIC 16 /* smtp public postable room */
187 #define QR2_MODERATED 32 /* Listservice aide has to permit posts */
193 #define UA_GOTOALLOWED 4
194 #define UA_HASNEWMSGS 8
196 #define UA_POSTALLOWED 32
197 #define UA_ADMINALLOWED 64
198 #define UA_DELETEALLOWED 128
199 #define UA_ISTRASH 256 /* Only available in room view... */
203 * User flags (from Citadel)
205 #define US_NEEDVALID 1 /* User needs to be validated */
206 #define US_PERM 4 /* Permanent user */
207 #define US_LASTOLD 16 /* Print last old message with new */
208 #define US_EXPERT 32 /* Experienced user */
209 #define US_UNLISTED 64 /* Unlisted userlog entry */
210 #define US_NOPROMPT 128 /* Don't prompt after each message */
211 #define US_PROMPTCTL 256 /* <N>ext & <S>top work at prompt */
212 #define US_DISAPPEAR 512 /* Use "disappearing msg prompts" */
213 #define US_REGIS 1024 /* Registered user */
214 #define US_PAGINATOR 2048 /* Pause after each screen of text */
215 #define US_INTERNET 4096 /* Internet mail privileges */
216 #define US_FLOORS 8192 /* User wants to see floors */
217 #define US_COLOR 16384 /* User wants ANSI color support */
218 #define US_USER_SET (US_LASTOLD | US_EXPERT | US_UNLISTED | \
219 US_NOPROMPT | US_DISAPPEAR | US_PAGINATOR | \
220 US_FLOORS | US_COLOR | US_PROMPTCTL )
224 #define SRV_STATUS_MSG(ServerLineBuf) (ChrPtr(ServerLineBuf) + 4), (StrLength(ServerLineBuf) - 4)
225 #define MAJORCODE(a) (((int)(a / 100) ) * 100)
227 #define LISTING_FOLLOWS 100
229 #define MORE_DATA 300
230 #define SEND_LISTING 400
232 #define BINARY_FOLLOWS 600
233 #define SEND_BINARY 700
234 #define START_CHAT_MODE 800
235 #define ASYNC_MSG 900
237 #define MINORCODE(a) (a % 100)
238 #define ASYNC_GEXP 02
239 #define INTERNAL_ERROR 10
241 #define ILLEGAL_VALUE 12
242 #define NOT_LOGGED_IN 20
243 #define CMD_NOT_SUPPORTED 30
244 #define SERVER_SHUTTING_DOWN 31
245 #define PASSWORD_REQUIRED 40
246 #define ALREADY_LOGGED_IN 41
247 #define USERNAME_REQUIRED 42
248 #define HIGHER_ACCESS_REQUIRED 50
249 #define MAX_SESSIONS_EXCEEDED 51
250 #define RESOURCE_BUSY 52
251 #define RESOURCE_NOT_OPEN 53
253 #define INVALID_FLOOR_OPERATION 61
254 #define NO_SUCH_USER 70
255 #define FILE_NOT_FOUND 71
256 #define ROOM_NOT_FOUND 72
257 #define NO_SUCH_SYSTEM 73
258 #define ALREADY_EXISTS 74
259 #define MESSAGE_NOT_FOUND 75
262 * NLI is the string that shows up in a who's online listing for sessions
263 * that are active but do not (yet) have a user logged in.
265 #define NLI "(not logged in)"
268 * Expiry policy for the autopurger
270 #define EXPIRE_NEXTLEVEL 0 /* Inherit expiration policy */
271 #define EXPIRE_MANUAL 1 /* Don't expire messages at all */
272 #define EXPIRE_NUMMSGS 2 /* Keep only latest n messages */
273 #define EXPIRE_AGE 3 /* Expire messages after n days */
274 typedef struct __ExpirePolicy {
275 int loaded; /* has this been loaded from the server? */
279 void LoadExpirePolicy(GPEXWhichPolicy which);
280 void SaveExpirePolicyFromHTTP(GPEXWhichPolicy which);
283 * Linked list of session variables encoded in an x-www-urlencoded content type
285 typedef struct urlcontent urlcontent;
287 char url_key[32]; /* key */
288 StrBuf *url_data; /* value */
292 * Information about the Citadel server to which we are connected
294 typedef struct _serv_info {
295 int serv_pid; /* Process ID of the Citadel server */
296 StrBuf *serv_nodename; /* Node name of the Citadel server */
297 StrBuf *serv_humannode; /* Juman readable node name of the Citadel server */
298 StrBuf *serv_fqdn; /* Fully qualified Domain Name (such as uncensored.citadel.org) */
299 StrBuf *serv_software; /* Free form text description of the server software in use */
300 int serv_rev_level; /* Server version number (times 100) */
301 StrBuf *serv_bbs_city; /* Geographic location of the Citadel server */
302 StrBuf *serv_sysadm; /* Name of system administrator */
303 int serv_supports_ldap; /* is the server linked against an ldap tree for adresses? */
304 int serv_newuser_disabled; /* Has the server disabled self-service new user creation? */
305 StrBuf *serv_default_cal_zone; /* Default timezone for unspecified calendar items */
306 int serv_supports_sieve; /* Server supports Sieve mail filtering */
307 int serv_fulltext_enabled; /* Full text index is enabled */
308 StrBuf *serv_svn_revision; /* svn or git revision of the server */
309 int serv_supports_openid; /* Server supports authentication via OpenID */
310 int serv_supports_guest; /* Server supports unauthenticated guest logins */
314 typedef struct _disp_cal {
315 icalcomponent *cal; /* cal items for display */
316 long cal_msgnum; /* cal msgids for display */
317 char *from; /* owner of this component */
318 int unread; /* already seen by the user? */
325 icalcomponent *SortBy; /* cal items for display */
326 icalproperty_status Status;
329 typedef struct _IcalEnumMap {
332 icalproperty_kind map;
336 * Address book entry (keep it short and sweet, it's just a quickie lookup
337 * which we can use to get to the real meat and bones later)
339 typedef struct _addrbookent {
340 char ab_name[64]; /* name string */
341 long ab_msgnum; /* message number of address book entry */
346 #define ANONYMOUS (1<<1)
347 #define NEED_URL (1<<2)
348 #define XHTTP_COMMANDS (1<<3)
350 #define URLNAMESPACE (1<<4)
351 #define LOGCHATTY (1<<5)
352 #define COOKIEUNNEEDED (1<<6)
353 #define ISSTATIC (1<<7)
354 #define FORCE_SESSIONCLOSE (1<<8)
355 #define PARSE_REST_URL (1<<9)
356 #define PROHIBIT_STARTPAGE (1<<10)
359 #define DATEFMT_FULL 0
360 #define DATEFMT_BRIEF 1
361 #define DATEFMT_RAWDATE 2
362 #define DATEFMT_LOCALEDATE 3
363 void webcit_fmt_date(char *buf, size_t siz, time_t thetime, int Format);
366 typedef enum _RESTDispatchID {
372 typedef int (*WebcitRESTDispatchID)(RESTDispatchID WhichAction, int IgnoreFloor);
373 typedef void (*WebcitHandlerFunc)(void);
374 typedef struct _WebcitHandler{
376 WebcitRESTDispatchID RID;
383 void WebcitAddUrlHandler(const char * UrlString, long UrlSLen, const char *DisplayName, long dslen, WebcitHandlerFunc F, long Flags);
385 typedef struct _headereval {
386 ExamineMsgHeaderFunc evaluator;
415 extern const char *ReqStrs[eNONE];
418 #define AUTH_COOKIE 1
423 typedef struct _HdrRefs {
424 long eReqType; /* HTTP method */
431 time_t if_modified_since;
432 int gzip_ok; /* Nonzero if Accept-encoding: gzip */
433 int prohibit_caching;
437 /* these are references into Hdr->HTTPHeaders, so we don't need to free them. */
441 StrBuf *browser_host;
442 StrBuf *browser_language;
447 const WebcitHandler *Handler;
450 typedef struct _ParsedHttpHdrs {
451 int http_sock; /* HTTP server socket */
459 StrBuf *this_page; /* URL of current page */
463 HashList *urlstrings; /* variables passed to webcit in a URL */
464 HashList *HTTPHeaders; /* the headers the client sent us */
465 int nWildfireHeaders; /* how many wildfire headers did we already send? */
472 * One of these is kept for each active Citadel session.
473 * HTTP transactions are bound to one at a time.
476 /* infrastructural members */
477 wcsession *next; /* Linked list */
478 pthread_mutex_t SessionMutex; /* mutex for exclusive access */
479 int wc_session; /* WebCit session ID */
480 int killthis; /* Nonzero == purge this session */
481 int ctdl_pid; /* Session ID on the Citadel server */
482 int nonce; /* session nonce (to prevent session riding) */
483 int inuse; /* set to nonzero if bound to a running thread */
485 /* Session local Members */
486 int serv_sock; /* Client socket to Citadel server */
487 StrBuf *ReadBuf; /* linebuffered reads from the server */
488 StrBuf *MigrateReadLineBuf; /* here we buffer legacy server read stuff */
489 const char *ReadPos; /* whats our read position in ReadBuf? */
490 int last_chat_seq; /* When in chat - last message seq# we saw */
491 time_t lastreq; /* Timestamp of most recent HTTP */
492 time_t last_pager_check; /* last time we polled for instant msgs */
493 ServInfo *serv_info; /* Information about the citserver we're connected to */
494 int is_ajax; /* are we doing an ajax request? */
495 StrBuf *PushedDestination; /* Where to go after login, registration, etc. */
497 /* Request local Members */
498 StrBuf *CLineBuf; /* linebuffering client stuff */
500 StrBuf *WBuf; /* Our output buffer */
501 StrBuf *HBuf; /* Our HeaderBuffer */
502 StrBuf *WFBuf; /* Wildfire error logging buffer */
503 StrBuf *trailing_javascript; /* extra javascript to be appended to page */
504 StrBuf *ImportantMsg;
505 HashList *Directory; /* Parts of the directory URL in snippets */
506 const Floor *CurrentFloor; /* when Parsing REST, which floor are we on? */
509 StrBuf *wc_username; /* login name of current user */
510 StrBuf *wc_fullname; /* Screen name of current user */
511 StrBuf *wc_password; /* Password of current user */
512 StrBuf *httpauth_pass; /* only for GroupDAV sessions */
513 int axlevel; /* this user's access level */
514 int is_aide; /* nonzero == this user is an Aide */
515 int connected; /* nonzero == we are connected to Citadel */
516 int logged_in; /* nonzero == we are logged in */
517 int need_regi; /* This user needs to register. */
518 int need_vali; /* New users require validation. */
521 StrBuf *cs_inet_email; /* User's preferred Internet addr. */
522 HashList *hash_prefs; /* WebCit preferences for this user */
523 StrBuf *DefaultCharset; /* Charset the user preferes */
524 int downloaded_prefs; /* Has the client download its prefs yet? */
525 int SavePrefsToServer; /* Should we save our preferences to the server at the end of the request? */
526 int selected_language; /* Language selected by user */
527 int time_format_cache; /* which timeformat does our user like? */
529 folder CurRoom; /* information about our current room */
530 const folder *ThisRoom; /* if REST found a room, remember it here. */
531 /* next/previous room thingabob */
532 struct march *march; /* march mode room list */
533 char ugname[128]; /* where does 'ungoto' take us */
534 long uglsn; /* last seen message number for ungoto */
536 /* Uploading; mime attachments for composing messages */
537 HashList *attachments; /* list of attachments for 'enter message' */
538 int upload_length; /* content length of http-uploaded data */
539 StrBuf *upload; /* pointer to http-uploaded data */
540 StrBuf *upload_filename; /* filename of http-uploaded data */
541 char upload_content_type[256]; /* content type of http-uploaded data */
543 int remember_new_mail; /* last count of new mail messages */
545 /* Roomiew control */
546 HashList *Floors; /* floors our citserver has hashed numeric for quicker access*/
547 HashList *FloorsByName; /* same but hashed by its name */
548 HashList *Rooms; /* our directory structure as loaded by LKRA */
549 HashList *summ; /* list of messages for mailbox summary view */
550 /** Perhaps these should be within a struct instead */
551 long startmsg; /* message number to start at */
552 long maxmsgs; /* maximum messages to display */
553 long num_displayed; /* number of messages actually displayed */
554 HashList *disp_cal_items; /* sorted list of calendar items; startdate is the sort criteria. */
557 char last_chat_user[256];
559 /* Iconbar controls */
560 int cache_max_folders;
561 int cache_num_floors;
562 long *IBSettingsVec; /* which icons should be shown / not shown? */
563 const StrBuf *floordiv_expanded; /* which floordiv currently expanded */
564 int ib_wholist_expanded;
565 int ib_roomlist_expanded;
567 /* our known Sieve scripts; loaded by SIEVE:SCRIPTS iterator. */
568 HashList *KnownSieveScripts;
570 /* Transcoding cache buffers; used to avoid to frequent realloc */
574 /* cache stuff for templates. TODO: find a smartrer way */
575 HashList *ServCfg; /* cache our server config for editing */
576 HashList *InetCfg; /* Our inet server config for editing */
577 ExpirePolicy Policy[maxpolicy];
579 /* used by the blog viewer */
580 int bptlid; /* hash of thread currently being rendered */
584 typedef void (*Header_Evaluator)(StrBuf *Line, ParsedHttpHdrs *hdr);
586 typedef struct _HttpHeader {
592 void RegisterHeaderHandler(const char *Name, long Len, Header_Evaluator F);
602 #define num_parms(source) num_tokens(source, '|')
605 #define site_prefix (WC ? (WC->Hdr->HostHeader) : NULL)
607 /* Per-session data */
608 #define WC ((struct wcsession *)pthread_getspecific(MyConKey))
609 extern pthread_key_t MyConKey;
611 /* Per-thread SSL context */
613 #define THREADSSL ((SSL *)pthread_getspecific(ThreadSSL))
614 extern pthread_key_t ThreadSSL;
615 extern char ctdl_key_dir[PATH_MAX];
616 extern char file_crpt_file_key[PATH_MAX];
617 extern char file_crpt_file_csr[PATH_MAX];
618 extern char file_crpt_file_cer[PATH_MAX];
622 void ssl_lock(int mode, int n, const char *file, int line);
623 int starttls(int sock);
624 extern SSL_CTX *ssl_ctx;
625 int client_read_sslbuffer(StrBuf *buf, int timeout);
626 void client_write_ssl(const StrBuf *Buf);
630 extern int follow_xff;
631 extern char *server_cookie;
632 extern char *ctdlhost, *ctdlport;
633 extern char *axdefs[];
634 extern int num_threads_existing;
635 extern int num_threads_executing;
636 extern int setup_wizard;
637 extern char wizard_filename[];
639 void InitialiseSemaphores(void);
640 void begin_critical_section(int which_one);
641 void end_critical_section(int which_one);
644 extern void do_404(void);
645 void http_redirect(const char *);
648 #ifdef UBER_VERBOSE_DEBUGGING
649 #define wc_printf(...) wcc_printf(__FILE__, __FUNCTION__, __LINE__, __VA_ARGS__)
650 void wcc_printf(const char *FILE, const char *FUNCTION, long LINE, const char *format, ...);
652 void wc_printf(const char *format,...)__attribute__((__format__(__printf__,1,2)));
655 void hprintf(const char *format,...)__attribute__((__format__(__printf__,1,2)));
657 void CheckAuthBasic(ParsedHttpHdrs *hdr);
658 void GetAuthBasic(ParsedHttpHdrs *hdr);
660 void sleeeeeeeeeep(int);
662 size_t wc_strftime(char *s, size_t max, const char *format, const struct tm *tm);
663 void fmt_time(char *buf, size_t siz, time_t thetime);
664 void httpdate(char *buf, time_t thetime);
665 time_t httpdate_to_timestamp(StrBuf *buf);
670 void end_webcit_session(void);
675 void cookie_to_stuff(StrBuf *cookie,
682 void locate_host(StrBuf *TBuf, int);
683 void become_logged_in(const StrBuf *user, const StrBuf *pass, StrBuf *serv_response);
685 void display_login(void);
686 void display_openids(void);
687 void display_default_landing_page(void);
688 void do_welcome(void);
690 void display_reg(int during_login);
691 void display_main_menu(void);
692 void display_aide_menu(void);
694 void RegisterEmbeddableMimeType(const char *MimeType, long MTLen, int Priority);
695 void CreateMimeStr(void);
698 void pop_destination(void);
700 void fmout(const char *align);
701 void _fmout(StrBuf *Targt, const char *align);
702 void FmOut(StrBuf *Target, const char *align, const StrBuf *Source);
703 void wDumpContent(int);
706 void PutRequestLocalMem(void *Data, DeleteHashDataFunc DeleteIt);
708 void output_headers( int do_httpheaders,
714 void output_custom_content_header(const char *ctype);
715 void cdataout(char *rawdata);
718 void url(char *buf, size_t bufsize);
719 void UrlizeText(StrBuf* Target, StrBuf *Source, StrBuf *WrkBuf);
722 void display_vcard(StrBuf *Target, wc_mime_attachment *Mime, char alpha, int full, char **storename, long msgnum);
724 void display_success(const char *successmessage);
726 void shutdown_sessions(void);
730 StrBuf *load_mimepart(long msgnum, char *partnum);
731 void MimeLoadData(wc_mime_attachment *Mime);
732 void do_edit_vcard(long msgnum, char *partnum,
733 message_summary *VCMsg,
734 wc_mime_attachment *VCAtt,
735 const char *return_to,
736 const char *force_room);
738 void select_user_to_edit(const char *preselect);
740 void convenience_page(const char *titlebarcolor, const char *titlebarmsg, const char *messagetext);
741 void output_html(const char *, int, int, StrBuf *, StrBuf *);
743 ssize_t write(int fd, const void *buf, size_t count);
744 void cal_process_attachment(wc_mime_attachment *Mime);
746 void generate_uuid(char *);
748 void address_book_popup(void);
749 void begin_ajax_response(void);
750 void end_ajax_response(void);
753 extern char *months[];
755 long locate_user_vcard_in_this_room(message_summary **VCMsg,
756 wc_mime_attachment **VCAtt);
757 void http_transmit_thing(const char *content_type, int is_static);
758 long unescape_input(char *buf);
759 void check_thread_pool_size(void);
760 void StrEndTab(StrBuf *Target, int tabnum, int num_tabs);
761 void StrBeginTab(StrBuf *Target, int tabnum, int num_tabs, StrBuf **Names);
762 void StrTabbedDialog(StrBuf *Target, int num_tabs, StrBuf *tabnames[]);
763 void tabbed_dialog(int num_tabs, char *tabnames[]);
764 void begin_tab(int tabnum, int num_tabs);
765 void end_tab(int tabnum, int num_tabs);
768 int get_time_format_cached (void);
769 void display_wiki_pagelist(void);
770 HashList *GetRoomListHashLKRA(StrBuf *Target, WCTemplputParams *TP);
772 /* actual supported locales */
773 void TmplGettext(StrBuf *Target, WCTemplputParams *TP);
775 void set_selected_language(const char *);
776 void go_selected_language(void);
777 void stop_selected_language(void);
778 const char *get_selected_language(void);
780 void utf8ify_rfc822_string(char **buf);
782 void begin_burst(void);
783 long end_burst(void);
785 void AppendImportantMessage(const char *pch, long len);
787 void http_datestring(char *buf, size_t n, time_t xtime);
790 /* These should be empty, but we have them for testing */
791 #define DEFAULT_HTTPAUTH_USER ""
792 #define DEFAULT_HTTPAUTH_PASS ""
795 /* Exit codes 101 through 109 are initialization failures so we don't want to
796 * just keep respawning indefinitely.
798 #define WC_EXIT_BIND 101 /* Can't bind to the port */
799 #define WC_EXIT_SSL 102 /* Can't initialize SSL */
802 #define WC_TIMEFORMAT_NONE 0
803 #define WC_TIMEFORMAT_AMPM 1
804 #define WC_TIMEFORMAT_24 2
806 extern int time_to_die; /* Nonzero if server is shutting down */
809 * Array type for a blog post. The first message is the post; the rest are comments
813 long *msgs; /* Array of msgnums for messages we are displaying */
814 int num_msgs; /* Number of msgnums stored in 'msgs' */
815 int alloc_msgs; /* Currently allocated size of array */
820 * Data which gets returned from a call to blogview_learn_thread_references()
828 struct bltr blogview_learn_thread_references(long msgnum);
829 void tmplput_blog_permalink(StrBuf *Target, WCTemplputParams *TP);