]> code.citadel.org Git - citadel.git/blob - webcit/webcit.h
b78b1f85d3127dd016913ffeec085a941220506b
[citadel.git] / webcit / webcit.h
1 /* $Id$ */
2
3 #include <ctype.h>
4 #include <stdlib.h>
5 #ifdef HAVE_UNISTD_H
6 #include <unistd.h>
7 #endif
8 #include <stdio.h>
9 #ifdef HAVE_FCNTL_H
10 #include <fcntl.h>
11 #endif
12 #include <signal.h>
13 #include <sys/types.h>
14 #include <sys/wait.h>
15 #include <sys/socket.h>
16 #ifdef HAVE_SYS_TIME_H
17 #include <sys/time.h>
18 #endif
19 #include <sys/stat.h>
20 #ifdef HAVE_LIMITS_H
21 #include <limits.h>
22 #endif
23 #include <netinet/in.h>
24 #include <arpa/inet.h>
25 #include <sys/un.h>
26 #include <netdb.h>
27 #include <sys/poll.h>
28 #include <string.h>
29 #include <pwd.h>
30 #include <errno.h>
31 #include <stdarg.h>
32 #include <pthread.h>
33 #include <signal.h>
34 #include <sys/utsname.h>
35
36 #ifndef INADDR_NONE
37 #define INADDR_NONE 0xffffffff
38 #endif
39
40 #ifdef HAVE_ICONV
41 #include <iconv.h>
42 #endif
43
44 #include "gettext.h"
45
46 #if ENABLE_NLS
47 #include <locale.h>
48 #define _(string)       gettext(string)
49 #else
50 #define _(string)       (string)
51 #endif
52
53 /*
54  * Uncomment to dump an HTTP trace to stderr
55 #define HTTP_TRACING 1
56  */
57
58 #ifdef HTTP_TRACING
59 #undef HAVE_ZLIB_H
60 #undef HAVE_ZLIB
61 #endif
62
63 #ifdef HAVE_ZLIB_H
64 #include <zlib.h>
65 #endif
66
67 #ifdef HAVE_ICAL_H
68 #ifdef HAVE_LIBICAL
69 #define WEBCIT_WITH_CALENDAR_SERVICE 1
70 #endif
71 #endif
72
73
74
75 #ifdef WEBCIT_WITH_CALENDAR_SERVICE
76 /* Work around PACKAGE/VERSION defs that are (not supposed to be?) in ical.h */
77 #ifdef PACKAGE
78 # define CTDL_PACKAGE PACKAGE
79 # undef PACKAGE
80 #endif
81 #ifdef VERSION
82 # define CTDL_VERSION VERSION
83 # undef VERSION
84 #endif
85 #include <ical.h>
86 #ifdef CTDL_PACKAGE
87 # ifdef PACKAGE
88 #  undef PACKAGE
89 # endif
90 # define PACKAGE CTDL_PACKAGE
91 # undef CTDL_PACKAGE
92 #endif
93 #ifdef CTDL_VERSION
94 # ifdef VERSION
95 #  undef VERSION
96 # endif
97 # define VERSION CTDL_VERSION
98 # undef CTDL_VERSION
99 #endif
100 #endif
101
102
103
104 #ifdef HAVE_OPENSSL
105 /* Work around RedHat's b0rken OpenSSL includes */
106 #define OPENSSL_NO_KRB5
107 #include <openssl/ssl.h>
108 #include <openssl/err.h>
109 #include <openssl/rand.h>
110 #endif
111
112 #define CALENDAR_ROOM_NAME      "Calendar"
113 #define PRODID "-//Citadel//NONSGML Citadel Calendar//EN"
114
115 #define SIZ                     4096            /* generic buffer size */
116
117 #define TRACE fprintf(stderr, "Checkpoint: %s, %d\n", __FILE__, __LINE__)
118
119 #define SLEEPING                180             /* TCP connection timeout */
120 #define WEBCIT_TIMEOUT          900             /* WebCit session timeout */
121 #define PORT_NUM                2000            /* port number to listen on */
122 #define SERVER                  "WebCit v6.40"  /* who's in da house */
123 #define DEVELOPER_ID            0
124 #define CLIENT_ID               4
125 #define CLIENT_VERSION          640             /* This version of WebCit */
126 #define MINIMUM_CIT_VERSION     661             /* min required Citadel ver. */
127 #define DEFAULT_HOST            "localhost"     /* Default Citadel server */
128 #define DEFAULT_PORT            "504"
129 #define LB                      (1)             /* Internal escape chars */
130 #define RB                      (2)
131 #define QU                      (3)
132 #define TARGET                  "webcit01"      /* Target for inline URL's */
133 #define HOUSEKEEPING            15              /* Housekeeping frequency */
134 #define MIN_WORKER_THREADS      5
135 #define MAX_WORKER_THREADS      250
136 #define LISTEN_QUEUE_LENGTH     100             /* listen() backlog queue */
137
138 #define USERCONFIGROOM          "My Citadel Config"
139 #define DEFAULT_MAXMSGS         20
140
141
142 /*
143  * Room flags (from Citadel)
144  *
145  * bucket one...
146  */
147 #define QR_PERMANENT    1               /* Room does not purge        */
148 #define QR_INUSE        2               /* Set if in use, clear if avail    */
149 #define QR_PRIVATE      4               /* Set for any type of private room */
150 #define QR_PASSWORDED   8               /* Set if there's a password too    */
151 #define QR_GUESSNAME    16              /* Set if it's a guessname room     */
152 #define QR_DIRECTORY    32              /* Directory room                  */
153 #define QR_UPLOAD       64              /* Allowed to upload            */
154 #define QR_DOWNLOAD     128             /* Allowed to download        */
155 #define QR_VISDIR       256             /* Visible directory            */
156 #define QR_ANONONLY     512             /* Anonymous-Only room        */
157 #define QR_ANONOPT      1024            /* Anonymous-Option room            */
158 #define QR_NETWORK      2048            /* Shared network room        */
159 #define QR_PREFONLY     4096            /* Preferred status needed to enter */
160 #define QR_READONLY     8192            /* Aide status required to post     */
161 #define QR_MAILBOX      16384           /* Set if this is a private mailbox */
162
163 /*
164  * bucket two...
165  */
166 #define QR2_SYSTEM      1               /* System room; hide by default     */
167 #define QR2_SELFLIST    2               /* Self-service mailing list mgmt   */
168
169 /*
170  * user/room access
171  */
172 #define UA_KNOWN        2
173 #define UA_GOTOALLOWED  4
174 #define UA_HASNEWMSGS   8
175 #define UA_ZAPPED       16
176
177
178 /*
179  * User flags (from Citadel)
180  */
181 #define US_NEEDVALID    1               /* User needs to be validated       */
182 #define US_PERM         4               /* Permanent user                   */
183 #define US_LASTOLD      16              /* Print last old message with new  */
184 #define US_EXPERT       32              /* Experienced user                 */
185 #define US_UNLISTED     64              /* Unlisted userlog entry           */
186 #define US_NOPROMPT     128             /* Don't prompt after each message  */
187 #define US_PROMPTCTL    256             /* <N>ext & <S>top work at prompt   */
188 #define US_DISAPPEAR    512             /* Use "disappearing msg prompts"   */
189 #define US_REGIS        1024            /* Registered user                  */
190 #define US_PAGINATOR    2048            /* Pause after each screen of text  */
191 #define US_INTERNET     4096            /* Internet mail privileges         */
192 #define US_FLOORS       8192            /* User wants to see floors         */
193 #define US_COLOR        16384           /* User wants ANSI color support    */
194 #define US_USER_SET     (US_LASTOLD | US_EXPERT | US_UNLISTED | \
195                         US_NOPROMPT | US_DISAPPEAR | US_PAGINATOR | \
196                         US_FLOORS | US_COLOR | US_PROMPTCTL )
197
198
199
200
201 struct httprequest {
202         struct httprequest *next;
203         char line[SIZ];
204 };
205
206 struct urlcontent {
207         struct urlcontent *next;
208         char url_key[32];
209         char *url_data;
210 };
211
212 struct serv_info {
213         int serv_pid;
214         char serv_nodename[32];
215         char serv_humannode[64];
216         char serv_fqdn[64];
217         char serv_software[64];
218         int serv_rev_level;
219         char serv_bbs_city[64];
220         char serv_sysadm[64];
221         char serv_moreprompt[SIZ];
222         int serv_ok_floors;
223         int serv_supports_ldap;
224 };
225
226
227
228 /*
229  * This struct holds a list of rooms for <G>oto operations.
230  */
231 struct march {
232         struct march *next;
233         char march_name[128];
234         int march_floor;
235         int march_order;
236 };
237
238 /* 
239  * This struct holds a list of rooms for client display.
240  * (oooh, a tree!)
241  */
242 struct roomlisting {
243         struct roomlisting *lnext;
244         struct roomlisting *rnext;
245         char rlname[128];
246         unsigned rlflags;
247         int rlfloor;
248         int rlorder;
249 };
250
251
252
253 /*
254  * Dynamic content for variable substitution in templates
255  */
256 struct wcsubst {
257         struct wcsubst *next;
258         int wcs_type;
259         char wcs_key[32];
260         void *wcs_value;
261         void (*wcs_function)(void);
262 };
263
264 /*
265  * Values for wcs_type
266  */
267 enum {
268         WCS_STRING,
269         WCS_FUNCTION,
270         WCS_SERVCMD
271 };
272
273
274 struct wc_attachment {
275         struct wc_attachment *next;
276         size_t length;
277         char content_type[SIZ];
278         char filename[SIZ];
279         char *data;
280 };
281
282 struct message_summary {
283         time_t date;
284         long msgnum;
285         char from[128];
286         char to[128];
287         char subj[128];
288         int hasattachments;
289         int is_new;
290 };
291
292 /* Data structure for roomlist-to-folderlist conversion */
293 struct folder {
294         int floor;
295         char room[SIZ];
296         char name[SIZ];
297         int hasnewmsgs;
298         int is_mailbox;
299         int selectable;
300         int view;
301 };
302
303 /*
304  * One of these is kept for each active Citadel session.
305  * HTTP transactions are bound to one at a time.
306  */
307 struct wcsession {
308         struct wcsession *next;         /* Linked list */
309         int wc_session;                 /* WebCit session ID */
310         char wc_username[128];
311         char wc_password[128];
312         char wc_roomname[256];
313         int connected;
314         int logged_in;
315         int axlevel;
316         int is_aide;
317         int is_room_aide;
318         int http_sock;
319         int serv_sock;
320         int chat_sock;
321         unsigned room_flags;
322         int wc_view;
323         int wc_default_view;
324         int wc_floor;
325         char ugname[128];
326         long uglsn;
327         int upload_length;
328         char *upload;
329         char upload_filename[PATH_MAX];
330         char upload_content_type[256];
331         int new_mail;
332         int remember_new_mail;
333         int need_regi;                  /* This user needs to register. */
334         int need_vali;                  /* New users require validation. */
335         char cs_inet_email[256];        /* User's preferred Internet addr. */
336         pthread_mutex_t SessionMutex;   /* mutex for exclusive access */
337         time_t lastreq;                 /* Timestamp of most recent HTTP */
338         int killthis;                   /* Nonzero == purge this session */
339         struct march *march;            /* march mode room list */
340         char reply_to[512];             /* reply-to address */
341
342         long msgarr[10000];             /* for read operations */
343         int num_summ;
344         struct message_summary *summ;
345
346         int is_wap;                     /* Client is a WAP gateway */
347         struct urlcontent *urlstrings;
348         int HaveInstantMessages;        /* Nonzero if incoming msgs exist */
349         struct wcsubst *vars;
350         char this_page[512];            /* address of current page */
351         char http_host[512];            /* HTTP Host: header */
352         char *preferences;
353 #ifdef WEBCIT_WITH_CALENDAR_SERVICE
354         struct disp_cal {
355                 icalcomponent *cal;             /* cal items for display */
356                 long cal_msgnum;                /* cal msgids for display */
357         } *disp_cal;
358         int num_cal;
359 #endif
360         struct wc_attachment *first_attachment;
361         char ImportantMessage[SIZ];
362         char last_chat_user[256];
363         int ctdl_pid;                   /* Session ID on the Citadel server */
364         char httpauth_user[256];        /* only for GroupDAV sessions */
365         char httpauth_pass[256];        /* only for GroupDAV sessions */
366
367         size_t burst_len;
368         char *burst;
369         int gzip_ok;                    /* Nonzero if Accept-encoding: gzip */
370         int is_mailbox;                 /* the current room is a private mailbox */
371
372         struct folder *cache_fold;      /* cache the iconbar room list */
373         int cache_max_folders;
374         int cache_num_floors;
375         time_t cache_timestamp;
376
377         int current_iconbar;            /* What's currently in the iconbar? */
378         char floordiv_expanded[32];     /* which floordiv currently expanded */
379 };
380
381 /* values for WC->current_iconbar */
382 enum {
383         current_iconbar_menu,
384         current_iconbar_roomlist
385 };
386
387
388 #define num_parms(source)               num_tokens(source, '|')
389
390 /* Per-session data */
391 #define WC ((struct wcsession *)pthread_getspecific(MyConKey))
392 extern pthread_key_t MyConKey;
393
394 /* Per-thread SSL context */
395 #ifdef HAVE_OPENSSL
396 #define THREADSSL ((SSL *)pthread_getspecific(ThreadSSL))
397 extern pthread_key_t ThreadSSL;
398 #endif
399
400 struct serv_info serv_info;
401 extern char floorlist[128][SIZ];
402 extern char *axdefs[];
403 extern char *ctdlhost, *ctdlport;
404 extern char *server_cookie;
405 extern int is_https;
406 extern int setup_wizard;
407 extern char wizard_filename[];
408 extern time_t if_modified_since;
409 extern int follow_xff;
410 void do_setup_wizard(void);
411
412 void stuff_to_cookie(char *cookie, int session,
413                         char *user, char *pass, char *room);
414 void cookie_to_stuff(char *cookie, int *session,
415                 char *user, size_t user_len,
416                 char *pass, size_t pass_len,
417                 char *room, size_t room_len);
418 void locate_host(char *, int);
419 void become_logged_in(char *, char *, char *);
420 void do_login(void);
421 void display_login(char *mesg);
422 void do_welcome(void);
423 void do_logout(void);
424 void display_main_menu(void);
425 void display_aide_menu(void);
426 void display_advanced_menu(void);
427 void slrp_highest(void);
428 void gotonext(void);
429 void ungoto(void);
430 void get_serv_info(char *, char *);
431 int uds_connectsock(char *);
432 int tcp_connectsock(char *, char *);
433 void serv_getln(char *strbuf, int bufsize);
434 void serv_puts(char *string);
435 void who(void);
436 void who_inner_div(void);
437 void fmout(char *align);
438 void pullquote_fmout(void);
439 void wDumpContent(int);
440 void serv_printf(const char *format,...);
441 char *bstr(char *key);
442 void urlesc(char *, char *);
443 void urlescputs(char *);
444 void jsesc(char *, char *);
445 void jsescputs(char *);
446 void output_headers(    int do_httpheaders,
447                         int do_htmlhead,
448                         int do_room_banner,
449                         int unset_cookies,
450                         int suppress_check,
451                         int cache);
452 void wprintf(const char *format,...);
453 void output_static(char *what);
454 void stresc(char *target, char *strbuf, int nbsp, int nolinebreaks);
455 void escputs(char *strbuf);
456 void url(char *buf);
457 void escputs1(char *strbuf, int nbsp, int nolinebreaks);
458 void msgesc(char *target, char *strbuf);
459 void msgescputs(char *strbuf);
460 int extract_int(const char *source, int parmnum);
461 long extract_long(const char *source, int parmnum);
462 void stripout(char *str, char leftboundary, char rightboundary);
463 void dump_vars(void);
464 void embed_main_menu(void);
465 void serv_read(char *buf, int bytes);
466 int haschar(char *, char);
467 void readloop(char *oper);
468 void embed_message(char *msgnum_as_string);
469 void print_message(char *msgnum_as_string);
470 void text_to_server(char *ptr, int convert_to_html);
471 void display_enter(void);
472 void post_message(void);
473 void confirm_delete_msg(void);
474 void delete_msg(void);
475 void confirm_move_msg(void);
476 void move_msg(void);
477 void userlist(void);
478 void showuser(void);
479 void display_page(void);
480 void page_user(void);
481 void do_chat(void);
482 void display_private(char *rname, int req_pass);
483 void goto_private(void);
484 void zapped_list(void);
485 void display_zap(void);
486 void zap(void);
487 void display_success(char *);
488 void authorization_required(const char *message);
489 void display_entroom(void);
490 void entroom(void);
491 void display_editroom(void);
492 void netedit(void);
493 void editroom(void);
494 void display_whok(void);
495 void do_invt_kick(void);
496 void server_to_text(void);
497 void save_edit(char *description, char *enter_cmd, int regoto);
498 void display_edit(char *description, char *check_cmd,
499                   char *read_cmd, char *save_cmd, int with_room_banner);
500 int gotoroom(char *gname);
501 void confirm_delete_room(void);
502 void delete_room(void);
503 void validate(void);
504 void display_graphics_upload(char *, char *, char *);
505 void do_graphics_upload(char *upl_cmd);
506 void serv_read(char *buf, int bytes);
507 void serv_gets(char *strbuf);
508 void serv_write(char *buf, int nbytes);
509 void serv_puts(char *string);
510 void serv_printf(const char *format,...);
511 void load_floorlist(void);
512 void display_reg(int);
513 void display_changepw(void);
514 void changepw(void);
515 void display_edit_node(void);
516 void edit_node(void);
517 void display_netconf(void);
518 void display_confirm_delete_node(void);
519 void delete_node(void);
520 void display_add_node(void);
521 void add_node(void);
522 void terminate_session(void);
523 void edit_me(void);
524 void display_siteconfig(void);
525 void siteconfig(void);
526 void display_generic(void);
527 void do_generic(void);
528 void ajax_servcmd(void);
529 void display_menubar(int);
530 void smart_goto(char *);
531 void worker_entry(void);
532 void session_loop(struct httprequest *);
533 void fmt_date(char *buf, time_t thetime, int brief);
534 void fmt_time(char *buf, time_t thetime);
535 void httpdate(char *buf, time_t thetime);
536 time_t httpdate_to_timestamp(const char *buf);
537 void end_webcit_session(void);
538 void page_popup(void);
539 void chat_recv(void);
540 void chat_send(void);
541 void http_redirect(char *);
542 void clear_local_substs(void);
543 void svprintf(char *keyname, int keytype, const char *format,...);
544 void svcallback(char *keyname, void (*fcn_ptr)() );
545 void do_template(void *templatename);
546 int lingering_close(int fd);
547 char *memreadline(char *start, char *buf, int maxlen);
548 int num_tokens (char *source, char tok);
549 void extract_token(char *dest, const char *source, int parmnum, char separator, int maxlen);
550 void remove_token(char *source, int parmnum, char separator);
551 char *load_mimepart(long msgnum, char *partnum);
552 int pattern2(char *search, char *patn);
553 void do_edit_vcard(long, char *, char *);
554 void edit_vcard(void);
555 void submit_vcard(void);
556 void striplt(char *);
557 void select_user_to_edit(char *message, char *preselect);
558 void delete_user(char *);
559 void display_edituser(char *who, int is_new);
560 void create_user(void);
561 void edituser(void);
562 void do_change_view(int);
563 void change_view(void);
564 void folders(void);
565 void load_preferences(void);
566 void save_preferences(void);
567 void get_preference(char *key, char *value, size_t value_len);
568 void set_preference(char *key, char *value, int save_to_server);
569 void knrooms(void);
570 int is_msg_in_mset(char *mset, long msgnum);
571 char *safestrncpy(char *dest, const char *src, size_t n);
572 void display_addressbook(long msgnum, char alpha);
573 void offer_start_page(void);
574 void convenience_page(char *titlebarcolor, char *titlebarmsg, char *messagetext);
575 void change_start_page(void);
576 void output_html(char *);
577 void display_floorconfig(char *);
578 void delete_floor(void);
579 void create_floor(void);
580 void rename_floor(void);
581 void do_listsub(void);
582 void toggle_self_service(void);
583 void summary(void);
584 void summary_inner_div(void);
585 ssize_t write(int fd, const void *buf, size_t count);
586 void cal_process_attachment(char *part_source, long msgnum, char *cal_partnum);
587 void display_calendar(long msgnum);
588 void display_task(long msgnum);
589 void display_note(long msgnum);
590 void do_calendar_view(void);
591 void do_tasks_view(void);
592 void free_calendar_buffer(void);
593 void calendar_summary_view(void);
594 int load_msg_ptrs(char *servcmd, int with_headers);
595 void CtdlEncodeBase64(char *dest, const char *source, size_t sourcelen);
596 int CtdlDecodeBase64(char *dest, const char *source, size_t length);
597 void free_attachments(struct wcsession *sess);
598 void set_room_policy(void);
599 void display_inetconf(void);
600 void save_inetconf(void);
601 void generate_uuid(char *);
602 void CtdlMakeTempFileName(char *, int);
603 void display_preferences(void);
604 void set_preferences(void);
605 void recp_autocomplete(char *);
606 void begin_ajax_response(void);
607 void end_ajax_response(void);
608 void initialize_viewdefs(void);
609 void initialize_axdefs(void);
610 void list_all_rooms_by_floor(char *viewpref);
611
612 #ifdef WEBCIT_WITH_CALENDAR_SERVICE
613 void display_edit_task(void);
614 void save_task(void);
615 void display_edit_event(void);
616 void save_event(void);
617 void display_icaltimetype_as_webform(struct icaltimetype *, char *);
618 void icaltime_from_webform(struct icaltimetype *result, char *prefix);
619 void icaltime_from_webform_dateonly(struct icaltimetype *result, char *prefix);
620 void display_edit_individual_event(icalcomponent *supplied_vtodo, long msgnum);
621 void save_individual_event(icalcomponent *supplied_vtodo, long msgnum);
622 void respond_to_request(void);
623 void handle_rsvp(void);
624 void ical_dezonify(icalcomponent *cal);
625 void partstat_as_string(char *buf, icalproperty *attendee);
626 icalcomponent *ical_encapsulate_subcomponent(icalcomponent *subcomp);
627 void check_attendee_availability(icalcomponent *supplied_vevent);
628 void do_freebusy(char *req);
629 #endif
630
631 extern char *months[];
632 extern char *days[];
633 void read_server_binary(char *buffer, size_t total_len);
634 char *read_server_text(void);
635 int goto_config_room(void);
636 long locate_user_vcard(char *username, long usernum);
637 void sleeeeeeeeeep(int);
638 void http_transmit_thing(char *thing, size_t length, char *content_type,
639                          int is_static);
640 void unescape_input(char *buf);
641 void do_iconbar(void);
642 void do_iconbar_roomlist(void);
643 void do_selected_iconbar(void);
644 void display_customize_iconbar(void);
645 void commit_iconbar(void);
646 int CtdlDecodeQuotedPrintable(char *decoded, char *encoded, int sourcelen);
647 void spawn_another_worker_thread(void);
648 void display_rss(char *roomname, char *request_method);
649 void set_floordiv_expanded(char *which_floordiv);
650
651 void embed_room_banner(char *, int);
652 /* navbar types that can be passed to embed_room_banner */
653 enum {
654         navbar_none,
655         navbar_default
656 };
657
658
659 #ifdef HAVE_OPENSSL
660 void init_ssl(void);
661 void endtls(void);
662 void ssl_lock(int mode, int n, const char *file, int line);
663 int starttls(int sock);
664 extern SSL_CTX *ssl_ctx;  
665 int client_read_ssl(char *buf, int bytes, int timeout);
666 void client_write_ssl(char *buf, int nbytes);
667 #endif
668
669 #ifdef HAVE_ZLIB
670 #include <zlib.h>
671 int ZEXPORT compress_gzip(Bytef * dest, uLongf * destLen,
672                           const Bytef * source, uLong sourceLen, int level);
673 #endif
674
675 #ifdef HAVE_ICONV
676 void utf8ify_rfc822_string(char *buf);
677 #endif
678
679 void begin_burst(void);
680 void end_burst(void);
681
682 extern char *ascmonths[];
683 extern char *hourname[];
684 void http_datestring(char *buf, size_t n, time_t xtime);
685
686 /* Views (from citadel.h) */
687 #define VIEW_BBS                0       /* Traditional Citadel BBS view */
688 #define VIEW_MAILBOX            1       /* Mailbox summary */
689 #define VIEW_ADDRESSBOOK        2       /* Address book view */
690 #define VIEW_CALENDAR           3       /* Calendar view */
691 #define VIEW_TASKS              4       /* Tasks view */
692 #define VIEW_NOTES              5       /* Notes view */
693
694
695 /* These should be empty, but we have them for testing */
696 #define DEFAULT_HTTPAUTH_USER   ""
697 #define DEFAULT_HTTPAUTH_PASS   ""
698