1 /*----------------------------------------------------------------------------*/
6 * message index functions
10 void DestroyMimeParts(wc_mime_attachment *Mime)
12 FreeStrBuf(&Mime->Name);
13 FreeStrBuf(&Mime->FileName);
14 FreeStrBuf(&Mime->PartNum);
15 FreeStrBuf(&Mime->Disposition);
16 FreeStrBuf(&Mime->ContentType);
17 FreeStrBuf(&Mime->Charset);
18 FreeStrBuf(&Mime->Data);
21 void DestroyMime(void *vMime)
23 wc_mime_attachment *Mime = (wc_mime_attachment*)vMime;
24 DestroyMimeParts(Mime);
28 void DestroyMessageSummary(void *vMsg)
30 message_summary *Msg = (message_summary*) vMsg;
32 FreeStrBuf(&Msg->from);
34 FreeStrBuf(&Msg->subj);
35 FreeStrBuf(&Msg->reply_inreplyto);
36 FreeStrBuf(&Msg->reply_references);
37 FreeStrBuf(&Msg->cccc);
38 FreeStrBuf(&Msg->hnod);
39 FreeStrBuf(&Msg->AllRcpt);
40 FreeStrBuf(&Msg->Room);
41 FreeStrBuf(&Msg->Rfca);
42 FreeStrBuf(&Msg->OtherNode);
44 FreeStrBuf(&Msg->reply_to);
46 DeleteHash(&Msg->Attachments); /**< list of Accachments */
47 DeleteHash(&Msg->Submessages);
48 DeleteHash(&Msg->AttachLinks);
49 DeleteHash(&Msg->AllAttach);
55 void RegisterMsgHdr(const char *HeaderName, long HdrNLen, ExamineMsgHeaderFunc evaluator, int type)
58 ev = (headereval*) malloc(sizeof(headereval));
59 ev->evaluator = evaluator;
61 Put(MsgHeaderHandler, HeaderName, HdrNLen, ev, NULL);
64 void RegisterMimeRenderer(const char *HeaderName, long HdrNLen, RenderMimeFunc MimeRenderer)
66 RenderMimeFuncStruct *f;
68 f = (RenderMimeFuncStruct*) malloc(sizeof(RenderMimeFuncStruct));
70 Put(MimeRenderHandler, HeaderName, HdrNLen, f, NULL);
74 /*----------------------------------------------------------------------------*/
77 * comparator for two longs in descending order.
79 int longcmp_r(const void *s1, const void *s2) {
83 l1 = *(long *)GetSearchPayload(s1);
84 l2 = *(long *)GetSearchPayload(s2);
86 if (l1 > l2) return(-1);
87 if (l1 < l2) return(+1);
92 * comparator for longs; descending order.
94 int qlongcmp_r(const void *s1, const void *s2) {
98 if (l1 > l2) return(-1);
99 if (l1 < l2) return(+1);
105 * comparator for message summary structs by ascending subject.
107 int summcmp_subj(const void *s1, const void *s2) {
108 message_summary *summ1;
109 message_summary *summ2;
111 summ1 = (message_summary *)GetSearchPayload(s1);
112 summ2 = (message_summary *)GetSearchPayload(s2);
113 return strcasecmp(ChrPtr(summ1->subj), ChrPtr(summ2->subj));
117 * comparator for message summary structs by descending subject.
119 int summcmp_rsubj(const void *s1, const void *s2) {
120 message_summary *summ1;
121 message_summary *summ2;
123 summ1 = (message_summary *)GetSearchPayload(s1);
124 summ2 = (message_summary *)GetSearchPayload(s2);
125 return strcasecmp(ChrPtr(summ2->subj), ChrPtr(summ1->subj));
128 * comparator for message summary structs by descending subject.
130 int groupchange_subj(const void *s1, const void *s2) {
131 message_summary *summ1;
132 message_summary *summ2;
134 summ1 = (message_summary *)s1;
135 summ2 = (message_summary *)s2;
136 return ChrPtr(summ2->subj)[0] != ChrPtr(summ1->subj)[0];
140 * comparator for message summary structs by ascending sender.
142 int summcmp_sender(const void *s1, const void *s2) {
143 message_summary *summ1;
144 message_summary *summ2;
146 summ1 = (message_summary *)GetSearchPayload(s1);
147 summ2 = (message_summary *)GetSearchPayload(s2);
148 return strcasecmp(ChrPtr(summ1->from), ChrPtr(summ2->from));
152 * comparator for message summary structs by descending sender.
154 int summcmp_rsender(const void *s1, const void *s2) {
155 message_summary *summ1;
156 message_summary *summ2;
158 summ1 = (message_summary *)GetSearchPayload(s1);
159 summ2 = (message_summary *)GetSearchPayload(s2);
160 return strcasecmp(ChrPtr(summ2->from), ChrPtr(summ1->from));
163 * comparator for message summary structs by descending sender.
165 int groupchange_sender(const void *s1, const void *s2) {
166 message_summary *summ1;
167 message_summary *summ2;
169 summ1 = (message_summary *)s1;
170 summ2 = (message_summary *)s2;
171 return strcasecmp(ChrPtr(summ2->from), ChrPtr(summ1->from)) != 0;
176 * comparator for message summary structs by ascending date.
178 int summcmp_date(const void *s1, const void *s2) {
179 message_summary *summ1;
180 message_summary *summ2;
182 summ1 = (message_summary *)GetSearchPayload(s1);
183 summ2 = (message_summary *)GetSearchPayload(s2);
185 if (summ1->date < summ2->date) return -1;
186 else if (summ1->date > summ2->date) return +1;
191 * comparator for message summary structs by descending date.
193 int summcmp_rdate(const void *s1, const void *s2) {
194 message_summary *summ1;
195 message_summary *summ2;
197 summ1 = (message_summary *)GetSearchPayload(s1);
198 summ2 = (message_summary *)GetSearchPayload(s2);
200 if (summ1->date < summ2->date) return +1;
201 else if (summ1->date > summ2->date) return -1;
206 * comparator for message summary structs by descending date.
208 const long DAYSECONDS = 24 * 60 * 60;
209 int groupchange_date(const void *s1, const void *s2) {
210 message_summary *summ1;
211 message_summary *summ2;
213 summ1 = (message_summary *)s1;
214 summ2 = (message_summary *)s2;
216 return (summ1->date % DAYSECONDS) != (summ2->date %DAYSECONDS);
220 /*----------------------------------------------------------------------------*/
221 /* Don't wanna know... or? */
222 void examine_pref(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset) {return;}
223 void examine_suff(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset) {return;}
224 void examine_path(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset) {return;}
225 void examine_content_encoding(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
227 /* TODO: do we care? */
230 void examine_nhdr(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
233 if (!strncasecmp(ChrPtr(HdrLine), "yes", 8))
236 int Conditional_ANONYMOUS_MESSAGE(StrBuf *Target, WCTemplputParams *TP)
238 message_summary *Msg = (message_summary*) CTX;
239 return Msg->nhdr != 0;
243 void examine_type(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
245 Msg->format_type = StrToi(HdrLine);
249 void examine_from(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
251 FreeStrBuf(&Msg->from);
252 Msg->from = NewStrBufPlain(NULL, StrLength(HdrLine));
253 StrBuf_RFC822_to_Utf8(Msg->from, HdrLine, WC->DefaultCharset, FoundCharset);
255 void tmplput_MAIL_SUMM_FROM(StrBuf *Target, WCTemplputParams *TP)
257 message_summary *Msg = (message_summary*) CTX;
258 StrBufAppendTemplate(Target, TP, Msg->from, 0);
263 void examine_subj(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
265 FreeStrBuf(&Msg->subj);
266 Msg->subj = NewStrBufPlain(NULL, StrLength(HdrLine));
267 StrBuf_RFC822_to_Utf8(Msg->subj, HdrLine, WC->DefaultCharset, FoundCharset);
269 void tmplput_MAIL_SUMM_SUBJECT(StrBuf *Target, WCTemplputParams *TP)
270 {/*////TODO: Fwd: and RE: filter!!*/
272 message_summary *Msg = (message_summary*) CTX;
273 StrBufAppendTemplate(Target, TP, Msg->subj, 0);
277 void examine_msgn(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
279 FreeStrBuf(&Msg->reply_inreplyto);
280 Msg->reply_inreplyto = NewStrBufPlain(NULL, StrLength(HdrLine));
281 StrBuf_RFC822_to_Utf8(Msg->reply_inreplyto, HdrLine, WC->DefaultCharset, FoundCharset);
283 void tmplput_MAIL_SUMM_INREPLYTO(StrBuf *Target, WCTemplputParams *TP)
285 message_summary *Msg = (message_summary*) CTX;
286 StrBufAppendTemplate(Target, TP, Msg->reply_inreplyto, 0);
289 int Conditional_MAIL_SUMM_UNREAD(StrBuf *Target, WCTemplputParams *TP)
291 message_summary *Msg = (message_summary*) CTX;
292 return Msg->is_new != 0;
295 void examine_wefw(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
297 FreeStrBuf(&Msg->reply_references);
298 Msg->reply_references = NewStrBufPlain(NULL, StrLength(HdrLine));
299 StrBuf_RFC822_to_Utf8(Msg->reply_references, HdrLine, WC->DefaultCharset, FoundCharset);
301 void tmplput_MAIL_SUMM_REFIDS(StrBuf *Target, WCTemplputParams *TP)
303 message_summary *Msg = (message_summary*) CTX;
304 StrBufAppendTemplate(Target, TP, Msg->reply_references, 0);
308 void examine_cccc(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
310 FreeStrBuf(&Msg->cccc);
311 Msg->cccc = NewStrBufPlain(NULL, StrLength(HdrLine));
312 StrBuf_RFC822_to_Utf8(Msg->cccc, HdrLine, WC->DefaultCharset, FoundCharset);
313 if (Msg->AllRcpt == NULL)
314 Msg->AllRcpt = NewStrBufPlain(NULL, StrLength(HdrLine));
315 if (StrLength(Msg->AllRcpt) > 0) {
316 StrBufAppendBufPlain(Msg->AllRcpt, HKEY(", "), 0);
318 StrBufAppendBuf(Msg->AllRcpt, Msg->cccc, 0);
320 void tmplput_MAIL_SUMM_CCCC(StrBuf *Target, WCTemplputParams *TP)
322 message_summary *Msg = (message_summary*) CTX;
323 StrBufAppendTemplate(Target, TP, Msg->cccc, 0);
327 void examine_room(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
329 if ((StrLength(HdrLine) > 0) &&
330 (strcasecmp(ChrPtr(HdrLine), ChrPtr(WC->wc_roomname)))) {
331 FreeStrBuf(&Msg->Room);
332 Msg->Room = NewStrBufDup(HdrLine);
335 void tmplput_MAIL_SUMM_ORGROOM(StrBuf *Target, WCTemplputParams *TP)
337 message_summary *Msg = (message_summary*) CTX;
338 StrBufAppendTemplate(Target, TP, Msg->Room, 0);
342 void examine_rfca(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
344 FreeStrBuf(&Msg->Rfca);
345 Msg->Rfca = NewStrBufDup(HdrLine);
347 void tmplput_MAIL_SUMM_RFCA(StrBuf *Target, WCTemplputParams *TP)
349 message_summary *Msg = (message_summary*) CTX;
350 StrBufAppendTemplate(Target, TP, Msg->Rfca, 0);
352 int Conditional_MAIL_SUMM_RFCA(StrBuf *Target, WCTemplputParams *TP)
354 message_summary *Msg = (message_summary*) CTX;
355 return StrLength(Msg->Rfca) > 0;
358 void examine_node(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
360 if ( (StrLength(HdrLine) > 0) &&
361 ((WC->room_flags & QR_NETWORK)
362 || ((strcasecmp(ChrPtr(HdrLine), ChrPtr(serv_info.serv_nodename))
363 && (strcasecmp(ChrPtr(HdrLine), ChrPtr(serv_info.serv_fqdn))))))) {
364 FreeStrBuf(&Msg->OtherNode);
365 Msg->OtherNode = NewStrBufDup(HdrLine);
368 void tmplput_MAIL_SUMM_OTHERNODE(StrBuf *Target, WCTemplputParams *TP)
370 message_summary *Msg = (message_summary*) CTX;
371 StrBufAppendTemplate(Target, TP, Msg->OtherNode, 0);
373 int Conditional_MAIL_SUMM_OTHERNODE(StrBuf *Target, WCTemplputParams *TP)
375 message_summary *Msg = (message_summary*) CTX;
376 return StrLength(Msg->OtherNode) > 0;
380 void examine_rcpt(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
382 FreeStrBuf(&Msg->to);
383 Msg->to = NewStrBufPlain(NULL, StrLength(HdrLine));
384 StrBuf_RFC822_to_Utf8(Msg->to, HdrLine, WC->DefaultCharset, FoundCharset);
385 if (Msg->AllRcpt == NULL)
386 Msg->AllRcpt = NewStrBufPlain(NULL, StrLength(HdrLine));
387 if (StrLength(Msg->AllRcpt) > 0) {
388 StrBufAppendBufPlain(Msg->AllRcpt, HKEY(", "), 0);
390 StrBufAppendBuf(Msg->AllRcpt, Msg->to, 0);
392 void tmplput_MAIL_SUMM_TO(StrBuf *Target, WCTemplputParams *TP)
394 message_summary *Msg = (message_summary*) CTX;
395 StrBufAppendTemplate(Target, TP, Msg->to, 0);
397 void tmplput_MAIL_SUMM_ALLRCPT(StrBuf *Target, WCTemplputParams *TP)
399 message_summary *Msg = (message_summary*) CTX;
400 StrBufAppendTemplate(Target, TP, Msg->AllRcpt, 0);
404 HashList *iterate_get_mailsumm_All(StrBuf *Target, WCTemplputParams *TP)
408 int Conditional_ROOM_DISPLAY_MSG(StrBuf *Target, WCTemplputParams *TP) {
413 GetHash(WCC->vars, HKEY("ITERATE:N"), (void *)&subst);
414 num_inset = subst->lvalue;
415 if ((num_inset >= WC->startmsg) && (WCC->num_displayed <= WCC->maxmsgs)) {
416 WCC->num_displayed = WCC->num_displayed+1;
417 return 1; /* Pass GO, collect $200 */
421 int Conditional_MAIL_SUMM_LASTMSG(StrBuf *Target, WCTemplputParams *TP) {
425 GetHash(WC->vars, HKEY("ITERATE:LASTN"), (void *)&nsubst);
426 is_last_n = nsubst->lvalue;
428 //GetHash(WC->vars, HKEY("ITERATE:N"), (void *)&n_dsubst);
429 //num_inset = n_dsubst->lvalue;
431 /* Is the num_displayed higher than maxmsgs? OR last in iterator */
432 return ((WC->num_displayed > WC->maxmsgs) || (is_last_n == 1));
434 void examine_time(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
436 Msg->date = StrTol(HdrLine);
439 void tmplput_MAIL_SUMM_DATE_BRIEF(StrBuf *Target, WCTemplputParams *TP)
442 message_summary *Msg = (message_summary*) CTX;
443 webcit_fmt_date(datebuf, Msg->date, 1);
444 StrBufAppendBufPlain(Target, datebuf, -1, 0);
447 void tmplput_MAIL_SUMM_DATE_FULL(StrBuf *Target, WCTemplputParams *TP)
450 message_summary *Msg = (message_summary*) CTX;
451 webcit_fmt_date(datebuf, Msg->date, 0);
452 StrBufAppendBufPlain(Target, datebuf, -1, 0);
454 void tmplput_MAIL_SUMM_DATE_NO(StrBuf *Target, WCTemplputParams *TP)
456 message_summary *Msg = (message_summary*) CTX;
457 StrBufAppendPrintf(Target, "%ld", Msg->date, 0);
462 void render_MAIL(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
464 Mime->Data = NewStrBufPlain(NULL, Mime->length);
465 read_message(Mime->Data, HKEY("view_submessage"), Mime->msgnum, 0, Mime->PartNum);
467 if ( (!IsEmptyStr(mime_submessages)) && (!section[0]) ) {
468 for (i=0; i<num_tokens(mime_submessages, '|'); ++i) {
469 extract_token(buf, mime_submessages, i, '|', sizeof buf);
470 / ** use printable_view to suppress buttons * /
471 wprintf("<blockquote>");
472 read_message(Mime->msgnum, 1, ChrPtr(Mime->Section));
473 wprintf("</blockquote>");
479 void render_MIME_VCard(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
483 if (StrLength(Mime->Data) > 0) {
486 /** If it's my vCard I can edit it */
487 if ( (!strcasecmp(ChrPtr(WCC->wc_roomname), USERCONFIGROOM))
488 || (!strcasecmp(&(ChrPtr(WCC->wc_roomname)[11]), USERCONFIGROOM))
489 || (WC->wc_view == VIEW_ADDRESSBOOK)
491 StrBufAppendPrintf(Buf, "<a href=\"edit_vcard?msgnum=%ld&partnum=%s\">",
492 Mime->msgnum, ChrPtr(Mime->PartNum));
493 StrBufAppendPrintf(Buf, "[%s]</a>", _("edit"));
496 /* In all cases, display the full card */
497 display_vcard(Buf, ChrPtr(Mime->Data), 0, 1, NULL, Mime->msgnum);
498 FreeStrBuf(&Mime->Data);
504 void render_MIME_ICS(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
506 if (StrLength(Mime->Data) == 0) {
509 if (StrLength(Mime->Data) > 0) {
510 cal_process_attachment(Mime);
516 void examine_mime_part(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
518 wc_mime_attachment *Mime;
521 Mime = (wc_mime_attachment*) malloc(sizeof(wc_mime_attachment));
522 memset(Mime, 0, sizeof(wc_mime_attachment));
523 Mime->msgnum = Msg->msgnum;
526 Mime->Name = NewStrBuf();
527 StrBufExtract_token(Buf, HdrLine, 0, '|');
528 StrBuf_RFC822_to_Utf8(Mime->Name, Buf, WC->DefaultCharset, FoundCharset);
529 StrBufTrim(Mime->Name);
531 StrBufExtract_token(Buf, HdrLine, 1, '|');
532 Mime->FileName = NewStrBuf();
533 StrBuf_RFC822_to_Utf8(Mime->FileName, Buf, WC->DefaultCharset, FoundCharset);
534 StrBufTrim(Mime->FileName);
536 Mime->PartNum = NewStrBuf();
537 StrBufExtract_token(Mime->PartNum, HdrLine, 2, '|');
538 StrBufTrim(Mime->PartNum);
539 if (strchr(ChrPtr(Mime->PartNum), '.') != NULL)
544 Mime->Disposition = NewStrBuf();
545 StrBufExtract_token(Mime->Disposition, HdrLine, 3, '|');
547 Mime->ContentType = NewStrBuf();
548 StrBufExtract_token(Mime->ContentType, HdrLine, 4, '|');
549 StrBufTrim(Mime->ContentType);
550 StrBufLowerCase(Mime->ContentType);
552 Mime->length = StrBufExtract_int(HdrLine, 5, '|');
554 if ( (StrLength(Mime->FileName) == 0) && (StrLength(Mime->Name) > 0) ) {
555 StrBufAppendBuf(Mime->FileName, Mime->Name, 0);
558 if (StrLength(Msg->PartNum) > 0) {
560 StrBufPrintf(Buf, "%s.%s", ChrPtr(Msg->PartNum), ChrPtr(Mime->PartNum));
566 if (Msg->AllAttach == NULL)
567 Msg->AllAttach = NewHash(1,NULL);
568 Put(Msg->AllAttach, SKEY(Mime->PartNum), Mime, DestroyMime);
573 void evaluate_mime_part(message_summary *Msg, wc_mime_attachment *Mime)
577 /* just print the root-node */
578 if ((Mime->level == 1) &&
579 GetHash(MimeRenderHandler, SKEY(Mime->ContentType), &vMimeRenderer) &&
580 vMimeRenderer != NULL)
582 Mime->Renderer = (RenderMimeFuncStruct*) vMimeRenderer;
583 if (Msg->Submessages == NULL)
584 Msg->Submessages = NewHash(1,NULL);
585 Put(Msg->Submessages, SKEY(Mime->PartNum), Mime, reference_free_handler);
587 else if ((Mime->level == 1) &&
588 (!strcasecmp(ChrPtr(Mime->Disposition), "inline"))
589 && (!strncasecmp(ChrPtr(Mime->ContentType), "image/", 6)) ){
590 if (Msg->AttachLinks == NULL)
591 Msg->AttachLinks = NewHash(1,NULL);
592 Put(Msg->AttachLinks, SKEY(Mime->PartNum), Mime, reference_free_handler);
594 else if ((Mime->level == 1) &&
595 (StrLength(Mime->ContentType) > 0) &&
596 ( (!strcasecmp(ChrPtr(Mime->Disposition), "attachment"))
597 || (!strcasecmp(ChrPtr(Mime->Disposition), "inline"))
598 || (!strcasecmp(ChrPtr(Mime->Disposition), ""))))
600 if (Msg->AttachLinks == NULL)
601 Msg->AttachLinks = NewHash(1,NULL);
602 Put(Msg->AttachLinks, SKEY(Mime->PartNum), Mime, reference_free_handler);
603 if ((strcasecmp(ChrPtr(Mime->ContentType), "application/octet-stream") == 0) &&
604 (StrLength(Mime->FileName) > 0)) {
605 FlushStrBuf(Mime->ContentType);
606 StrBufAppendBufPlain(Mime->ContentType,
607 GuessMimeByFilename(SKEY(Mime->FileName)),
613 void tmplput_MAIL_SUMM_NATTACH(StrBuf *Target, WCTemplputParams *TP)
615 message_summary *Msg = (message_summary*) CTX;
616 StrBufAppendPrintf(Target, "%ld", GetCount(Msg->Attachments));
620 void examine_hnod(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
622 FreeStrBuf(&Msg->hnod);
623 Msg->hnod = NewStrBufPlain(NULL, StrLength(HdrLine));
624 StrBuf_RFC822_to_Utf8(Msg->hnod, HdrLine, WC->DefaultCharset, FoundCharset);
626 void tmplput_MAIL_SUMM_H_NODE(StrBuf *Target, WCTemplputParams *TP)
628 message_summary *Msg = (message_summary*) CTX;
629 StrBufAppendTemplate(Target, TP, Msg->hnod, 0);
631 int Conditional_MAIL_SUMM_H_NODE(StrBuf *Target, WCTemplputParams *TP)
633 message_summary *Msg = (message_summary*) CTX;
634 return StrLength(Msg->hnod) > 0;
639 void examine_text(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
641 Msg->MsgBody->Data = NewStrBuf();
644 void examine_msg4_partnum(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
646 Msg->MsgBody->PartNum = NewStrBufDup(HdrLine);
647 StrBufTrim(Msg->MsgBody->PartNum);
650 void examine_content_lengh(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
652 Msg->MsgBody->length = StrTol(HdrLine);
653 Msg->MsgBody->size_known = 1;
656 void examine_content_type(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
666 Msg->MsgBody->ContentType = NewStrBufDup(HdrLine);
667 sem = strchr(ChrPtr(HdrLine), ';');
670 Token = NewStrBufPlain(NULL, StrLength(HdrLine));
671 Value = NewStrBufPlain(NULL, StrLength(HdrLine));
672 len = sem - ChrPtr(HdrLine);
673 StrBufCutAt(Msg->MsgBody->ContentType, len, NULL);
674 while (sem != NULL) {
675 while (isspace(*(sem + 1)))
677 StrBufCutLeft(HdrLine, sem - ChrPtr(HdrLine));
678 sem = strchr(ChrPtr(HdrLine), ';');
680 len = sem - ChrPtr(HdrLine);
682 len = StrLength(HdrLine);
685 StrBufAppendBufPlain(Token, ChrPtr(HdrLine), len, 0);
686 eq = strchr(ChrPtr(Token), '=');
688 len = eq - ChrPtr(Token);
689 StrBufAppendBufPlain(Value, eq + 1, StrLength(Token) - len - 1, 0);
690 StrBufCutAt(Token, len, NULL);
695 if (GetHash(MsgHeaderHandler, SKEY(Token), &vHdr) &&
697 Hdr = (headereval*)vHdr;
698 Hdr->evaluator(Msg, Value, FoundCharset);
700 else lprintf(1, "don't know how to handle content type sub-header[%s]\n", ChrPtr(Token));
707 void tmplput_MAIL_SUMM_N(StrBuf *Target, WCTemplputParams *TP)
709 message_summary *Msg = (message_summary*) CTX;
710 StrBufAppendPrintf(Target, "%ld", Msg->msgnum);
715 int Conditional_MAIL_MIME_ALL(StrBuf *Target, WCTemplputParams *TP)
717 message_summary *Msg = (message_summary*) CTX;
718 return GetCount(Msg->Attachments) > 0;
721 int Conditional_MAIL_MIME_SUBMESSAGES(StrBuf *Target, WCTemplputParams *TP)
723 message_summary *Msg = (message_summary*) CTX;
724 return GetCount(Msg->Submessages) > 0;
727 int Conditional_MAIL_MIME_ATTACHLINKS(StrBuf *Target, WCTemplputParams *TP)
729 message_summary *Msg = (message_summary*) CTX;
730 return GetCount(Msg->AttachLinks) > 0;
733 int Conditional_MAIL_MIME_ATTACH(StrBuf *Target, WCTemplputParams *TP)
735 message_summary *Msg = (message_summary*) CTX;
736 return GetCount(Msg->AllAttach) > 0;
741 /*----------------------------------------------------------------------------*/
742 void tmplput_QUOTED_MAIL_BODY(StrBuf *Target, WCTemplputParams *TP)
747 MsgNum = LBstr(TKEY(0));
749 read_message(Buf, HKEY("view_message_replyquote"), MsgNum, 0, NULL);
750 StrBufAppendTemplate(Target, TP, Buf, 1);
754 void tmplput_MAIL_BODY(StrBuf *Target, WCTemplputParams *TP)
756 message_summary *Msg = (message_summary*) CTX;
757 StrBufAppendTemplate(Target, TP, Msg->MsgBody->Data, 0);
761 void render_MAIL_variformat(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
763 /* Messages in legacy Citadel variformat get handled thusly... */
764 StrBuf *Target = NewStrBufPlain(NULL, StrLength(Mime->Data));
765 FmOut(Target, "JUSTIFY", Mime->Data);
766 FreeStrBuf(&Mime->Data);
770 void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
773 const char *ptr, *pte;
774 const char *BufPtr = NULL;
786 iconv_t ic = (iconv_t)(-1) ;
789 if ((StrLength(Mime->Data) == 0) && (Mime->length > 0)) {
790 FreeStrBuf(&Mime->Data);
791 Mime->Data = NewStrBufPlain(NULL, Mime->length);
792 if (!read_message(Mime->Data, HKEY("view_submessage"), Mime->msgnum, 0, Mime->PartNum))
796 /* Boring old 80-column fixed format text gets handled this way... */
797 if ((strcasecmp(ChrPtr(Mime->Charset), "us-ascii") == 0) &&
798 (strcasecmp(ChrPtr(Mime->Charset), "UTF-8") == 0))
803 if (StrLength(Mime->Charset) != 0)
805 else if (StrLength(FoundCharset) > 0)
807 else if (StrLength(WC->DefaultCharset) > 0)
808 cs = WC->DefaultCharset;
813 ctdl_iconv_open("UTF-8", ChrPtr(cs), &ic);
814 if (ic == (iconv_t)(-1) ) {
815 lprintf(5, "%s:%d iconv_open(UTF-8, %s) failed: %s\n",
816 __FILE__, __LINE__, ChrPtr(Mime->Charset), strerror(errno));
824 Target = NewStrBufPlain(NULL, StrLength(Mime->Data));
826 while ((n = StrBufSipLine(Line, Mime->Data, &BufPtr), n >= 0) && !done)
832 len = StrLength(Line);
835 while ((ptr < pte) &&
844 if (i > 0) StrBufCutLeft(Line, i);
846 if (StrLength(Line) == 0)
849 for (i = bn; i < bq; i++)
850 StrBufAppendBufPlain(Target, HKEY("<blockquote>"), 0);
851 for (i = bq; i < bn; i++)
852 StrBufAppendBufPlain(Target, HKEY("</blockquote>"), 0);
854 if (ConvertIt == 1) {
855 StrBufConvert(Line, Line1, &ic);
858 StrBufAppendBufPlain(Target, HKEY("<tt>"), 0);
859 UrlizeText(Line1, Line, Line2);
861 StrEscAppend(Target, Line1, NULL, 0, 0);
862 StrBufAppendBufPlain(Target, HKEY("</tt><br />\n"), 0);
866 for (i = 0; i < bn; i++)
867 StrBufAppendBufPlain(Target, HKEY("</blockquote>"), 0);
869 StrBufAppendBufPlain(Target, HKEY("</i><br />"), 0);
871 if (ic != (iconv_t)(-1) ) {
875 FreeStrBuf(&Mime->Data);
882 void render_MAIL_html(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
886 if (StrLength(Mime->Data) == 0)
889 Buf = NewStrBufPlain(NULL, StrLength(Mime->Data));
891 /* HTML is fun, but we've got to strip it first */
892 output_html(ChrPtr(Mime->Charset),
893 (WC->wc_view == VIEW_WIKI ? 1 : 0),
894 StrToi(Mime->PartNum),
896 FreeStrBuf(&Mime->Data);
900 void render_MAIL_UNKNOWN(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset)
902 /* Unknown weirdness */
903 FlushStrBuf(Mime->Data);
904 StrBufAppendBufPlain(Mime->Data, _("I don't know how to display "), -1, 0);
905 StrBufAppendBuf(Mime->Data, Mime->ContentType, 0);
906 StrBufAppendBufPlain(Mime->Data, HKEY("<br />\n"), 0);
914 HashList *iterate_get_mime_All(StrBuf *Target, WCTemplputParams *TP)
916 message_summary *Msg = (message_summary*) CTX;
917 return Msg->Attachments;
919 HashList *iterate_get_mime_Submessages(StrBuf *Target, WCTemplputParams *TP)
921 message_summary *Msg = (message_summary*) CTX;
922 return Msg->Submessages;
924 HashList *iterate_get_mime_AttachLinks(StrBuf *Target, WCTemplputParams *TP)
926 message_summary *Msg = (message_summary*) CTX;
927 return Msg->AttachLinks;
929 HashList *iterate_get_mime_Attachments(StrBuf *Target, WCTemplputParams *TP)
931 message_summary *Msg = (message_summary*) CTX;
932 return Msg->AllAttach;
935 void tmplput_MIME_Name(StrBuf *Target, WCTemplputParams *TP)
937 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
938 StrBufAppendTemplate(Target, TP, mime->Name, 0);
941 void tmplput_MIME_FileName(StrBuf *Target, WCTemplputParams *TP)
943 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
944 StrBufAppendTemplate(Target, TP, mime->FileName, 0);
947 void tmplput_MIME_PartNum(StrBuf *Target, WCTemplputParams *TP)
949 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
950 StrBufAppendTemplate(Target, TP, mime->PartNum, 0);
953 void tmplput_MIME_MsgNum(StrBuf *Target, WCTemplputParams *TP)
955 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
956 StrBufAppendPrintf(Target, "%ld", mime->msgnum);
959 void tmplput_MIME_Disposition(StrBuf *Target, WCTemplputParams *TP)
961 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
962 StrBufAppendTemplate(Target, TP, mime->Disposition, 0);
965 void tmplput_MIME_ContentType(StrBuf *Target, WCTemplputParams *TP)
967 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
968 StrBufAppendTemplate(Target, TP, mime->ContentType, 0);
971 void examine_charset(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset)
973 Msg->MsgBody->Charset = NewStrBufDup(HdrLine);
976 void tmplput_MIME_Charset(StrBuf *Target, WCTemplputParams *TP)
978 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
979 StrBufAppendTemplate(Target, TP, mime->Charset, 0);
982 void tmplput_MIME_Data(StrBuf *Target, WCTemplputParams *TP)
984 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
985 if (mime->Renderer != NULL)
986 mime->Renderer->f(mime, NULL, NULL);
987 StrBufAppendTemplate(Target, TP, mime->Data, 0);
988 /* TODO: check whether we need to load it now? */
991 void tmplput_MIME_LoadData(StrBuf *Target, WCTemplputParams *TP)
994 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
995 wc_mime_attachment *att;
997 if ( (!strcasecmp(ChrPtr(mime->Disposition), "inline"))||
998 (!strcasecmp(ChrPtr(mime->Disposition), "attachment")) )
1003 /* steal this mime part... */
1004 att = malloc(sizeof(wc_mime_attachment));
1005 memcpy(att, mime, sizeof(wc_mime_attachment));
1006 memset(mime, 0, sizeof(wc_mime_attachment));
1008 if (att->Data == NULL)
1011 if (WCC->attachments == NULL)
1012 WCC->attachments = NewHash(1, NULL);
1013 /* And add it to the list. */
1014 n = snprintf(N, sizeof N, "%d", GetCount(WCC->attachments) + 1);
1015 Put(WCC->attachments, N, n, att, DestroyMime);
1019 void tmplput_MIME_Length(StrBuf *Target, WCTemplputParams *TP)
1021 wc_mime_attachment *mime = (wc_mime_attachment*) CTX;
1022 StrBufAppendPrintf(Target, "%ld", mime->length);
1025 HashList *iterate_get_registered_Attachments(StrBuf *Target, WCTemplputParams *TP)
1027 return WC->attachments;
1030 void servcmd_do_search(char *buf, long bufsize)
1032 snprintf(buf, bufsize, "MSGS SEARCH|%s", bstr("query"));
1035 void servcmd_headers(char *buf, long bufsize)
1037 snprintf(buf, bufsize, "MSGS ALL");
1040 void servcmd_readfwd(char *buf, long bufsize)
1042 snprintf(buf, bufsize, "MSGS ALL");
1045 void servcmd_readnew(char *buf, long bufsize)
1047 snprintf(buf, bufsize, "MSGS NEW");
1050 void servcmd_readold(char *buf, long bufsize)
1052 snprintf(buf, bufsize, "MSGS OLD");
1056 readloop_struct rlid[] = {
1057 { {HKEY("do_search")}, servcmd_do_search},
1058 { {HKEY("headers")}, servcmd_headers},
1059 { {HKEY("readfwd")}, servcmd_readfwd},
1060 { {HKEY("readnew")}, servcmd_readnew},
1061 { {HKEY("readold")}, servcmd_readold}
1066 void SetAccessCommand(long Oper)
1068 wcsession *WCC = WC;
1070 if (WCC->UrlFragment1 != NULL ) {
1071 FlushStrBuf(WCC->UrlFragment1);
1072 StrBufAppendBufPlain(WCC->UrlFragment1,
1073 rlid[Oper].name.Key, rlid[Oper].name.len, 0);
1076 WCC->UrlFragment1 = NewStrBufPlain(rlid[Oper].name.Key, rlid[Oper].name.len);
1084 InitModule_MSGRENDERERS
1087 RegisterSortFunc(HKEY("date"),
1093 RegisterSortFunc(HKEY("subject"),
1099 RegisterSortFunc(HKEY("sender"),
1106 /* iterate over all known mails in WC->summ */
1107 RegisterIterator("MAIL:SUMM:MSGS", 0, NULL, iterate_get_mailsumm_All,
1108 NULL,NULL, CTX_MAILSUM, CTX_NONE, IT_NOFLAG);
1110 RegisterNamespace("MAIL:SUMM:DATEBRIEF", 0, 0, tmplput_MAIL_SUMM_DATE_BRIEF, CTX_MAILSUM);
1111 RegisterNamespace("MAIL:SUMM:DATEFULL", 0, 0, tmplput_MAIL_SUMM_DATE_FULL, CTX_MAILSUM);
1112 RegisterNamespace("MAIL:SUMM:DATENO", 0, 0, tmplput_MAIL_SUMM_DATE_NO, CTX_MAILSUM);
1113 RegisterNamespace("MAIL:SUMM:N", 0, 0, tmplput_MAIL_SUMM_N, CTX_MAILSUM);
1114 RegisterNamespace("MAIL:SUMM:FROM", 0, 2, tmplput_MAIL_SUMM_FROM, CTX_MAILSUM);
1115 RegisterNamespace("MAIL:SUMM:TO", 0, 2, tmplput_MAIL_SUMM_TO, CTX_MAILSUM);
1116 RegisterNamespace("MAIL:SUMM:SUBJECT", 0, 4, tmplput_MAIL_SUMM_SUBJECT, CTX_MAILSUM);
1117 RegisterNamespace("MAIL:SUMM:NTATACH", 0, 0, tmplput_MAIL_SUMM_NATTACH, CTX_MAILSUM);
1118 RegisterNamespace("MAIL:SUMM:CCCC", 0, 2, tmplput_MAIL_SUMM_CCCC, CTX_MAILSUM);
1119 RegisterNamespace("MAIL:SUMM:H_NODE", 0, 2, tmplput_MAIL_SUMM_H_NODE, CTX_MAILSUM);
1120 RegisterNamespace("MAIL:SUMM:ALLRCPT", 0, 2, tmplput_MAIL_SUMM_ALLRCPT, CTX_MAILSUM);
1121 RegisterNamespace("MAIL:SUMM:ORGROOM", 0, 2, tmplput_MAIL_SUMM_ORGROOM, CTX_MAILSUM);
1122 RegisterNamespace("MAIL:SUMM:RFCA", 0, 2, tmplput_MAIL_SUMM_RFCA, CTX_MAILSUM);
1123 RegisterNamespace("MAIL:SUMM:OTHERNODE", 2, 0, tmplput_MAIL_SUMM_OTHERNODE, CTX_MAILSUM);
1124 RegisterNamespace("MAIL:SUMM:REFIDS", 0, 0, tmplput_MAIL_SUMM_REFIDS, CTX_MAILSUM);
1125 RegisterNamespace("MAIL:SUMM:INREPLYTO", 0, 2, tmplput_MAIL_SUMM_INREPLYTO, CTX_MAILSUM);
1126 RegisterNamespace("MAIL:BODY", 0, 2, tmplput_MAIL_BODY, CTX_MAILSUM);
1127 RegisterNamespace("MAIL:QUOTETEXT", 1, 2, tmplput_QUOTED_MAIL_BODY, CTX_NONE);
1128 RegisterConditional(HKEY("COND:ROOM:DISPLAYMSG"), 0, Conditional_ROOM_DISPLAY_MSG, CTX_MAILSUM);
1129 RegisterConditional(HKEY("COND:MAIL:SUMM:LASTMSG"), 0, Conditional_MAIL_SUMM_LASTMSG, CTX_MAILSUM);
1130 RegisterConditional(HKEY("COND:MAIL:SUMM:RFCA"), 0, Conditional_MAIL_SUMM_RFCA, CTX_MAILSUM);
1131 RegisterConditional(HKEY("COND:MAIL:SUMM:UNREAD"), 0, Conditional_MAIL_SUMM_UNREAD, CTX_MAILSUM);
1132 RegisterConditional(HKEY("COND:MAIL:SUMM:H_NODE"), 0, Conditional_MAIL_SUMM_H_NODE, CTX_MAILSUM);
1133 RegisterConditional(HKEY("COND:MAIL:SUMM:OTHERNODE"), 0, Conditional_MAIL_SUMM_OTHERNODE, CTX_MAILSUM);
1134 RegisterConditional(HKEY("COND:MAIL:ANON"), 0, Conditional_ANONYMOUS_MESSAGE, CTX_MAILSUM);
1137 /* do we have mimetypes to iterate over? */
1138 RegisterConditional(HKEY("COND:MAIL:MIME:ATTACH"), 0, Conditional_MAIL_MIME_ALL, CTX_MAILSUM);
1139 RegisterConditional(HKEY("COND:MAIL:MIME:ATTACH:SUBMESSAGES"), 0, Conditional_MAIL_MIME_SUBMESSAGES, CTX_MAILSUM);
1140 RegisterConditional(HKEY("COND:MAIL:MIME:ATTACH:LINKS"), 0, Conditional_MAIL_MIME_ATTACHLINKS, CTX_MAILSUM);
1141 RegisterConditional(HKEY("COND:MAIL:MIME:ATTACH:ATT"), 0, Conditional_MAIL_MIME_ATTACH, CTX_MAILSUM);
1142 RegisterIterator("MAIL:MIME:ATTACH", 0, NULL, iterate_get_mime_All,
1143 NULL, NULL, CTX_MIME_ATACH, CTX_MAILSUM, IT_NOFLAG);
1144 RegisterIterator("MAIL:MIME:ATTACH:SUBMESSAGES", 0, NULL, iterate_get_mime_Submessages,
1145 NULL, NULL, CTX_MIME_ATACH, CTX_MAILSUM, IT_NOFLAG);
1146 RegisterIterator("MAIL:MIME:ATTACH:LINKS", 0, NULL, iterate_get_mime_AttachLinks,
1147 NULL, NULL, CTX_MIME_ATACH, CTX_MAILSUM, IT_NOFLAG);
1148 RegisterIterator("MAIL:MIME:ATTACH:ATT", 0, NULL, iterate_get_mime_Attachments,
1149 NULL, NULL, CTX_MIME_ATACH, CTX_MAILSUM, IT_NOFLAG);
1151 /* Parts of a mime attachent */
1152 RegisterNamespace("MAIL:MIME:NAME", 0, 2, tmplput_MIME_Name, CTX_MIME_ATACH);
1153 RegisterNamespace("MAIL:MIME:FILENAME", 0, 2, tmplput_MIME_FileName, CTX_MIME_ATACH);
1154 RegisterNamespace("MAIL:MIME:PARTNUM", 0, 2, tmplput_MIME_PartNum, CTX_MIME_ATACH);
1155 RegisterNamespace("MAIL:MIME:MSGNUM", 0, 2, tmplput_MIME_MsgNum, CTX_MIME_ATACH);
1156 RegisterNamespace("MAIL:MIME:DISPOSITION", 0, 2, tmplput_MIME_Disposition, CTX_MIME_ATACH);
1157 RegisterNamespace("MAIL:MIME:CONTENTTYPE", 0, 2, tmplput_MIME_ContentType, CTX_MIME_ATACH);
1158 RegisterNamespace("MAIL:MIME:CHARSET", 0, 2, tmplput_MIME_Charset, CTX_MIME_ATACH);
1159 RegisterNamespace("MAIL:MIME:LENGTH", 0, 2, tmplput_MIME_Length, CTX_MIME_ATACH);
1160 RegisterNamespace("MAIL:MIME:DATA", 0, 2, tmplput_MIME_Data, CTX_MIME_ATACH);
1161 /* load the actual attachment into WC->attachments; no output!!! */
1162 RegisterNamespace("MAIL:MIME:LOADDATA", 0, 0, tmplput_MIME_LoadData, CTX_MIME_ATACH);
1164 /* iterate the WC->attachments; use the above tokens for their contents */
1165 RegisterIterator("MSG:ATTACHNAMES", 0, NULL, iterate_get_registered_Attachments,
1166 NULL, NULL, CTX_MIME_ATACH, CTX_NONE, IT_NOFLAG);
1168 /* mime renderers translate an attachment into webcit viewable html text */
1169 RegisterMimeRenderer(HKEY("message/rfc822"), render_MAIL);
1170 RegisterMimeRenderer(HKEY("text/x-vcard"), render_MIME_VCard);
1171 RegisterMimeRenderer(HKEY("text/vcard"), render_MIME_VCard);
1172 RegisterMimeRenderer(HKEY("text/calendar"), render_MIME_ICS);
1173 RegisterMimeRenderer(HKEY("application/ics"), render_MIME_ICS);
1174 RegisterMimeRenderer(HKEY("text/x-citadel-variformat"), render_MAIL_variformat);
1175 RegisterMimeRenderer(HKEY("text/plain"), render_MAIL_text_plain);
1176 RegisterMimeRenderer(HKEY("text"), render_MAIL_text_plain);
1177 RegisterMimeRenderer(HKEY("text/html"), render_MAIL_html);
1178 RegisterMimeRenderer(HKEY(""), render_MAIL_UNKNOWN);
1180 /* these headers are citserver replies to MSG4 and friends. one evaluator for each */
1181 RegisterMsgHdr(HKEY("nhdr"), examine_nhdr, 0);
1182 RegisterMsgHdr(HKEY("type"), examine_type, 0);
1183 RegisterMsgHdr(HKEY("from"), examine_from, 0);
1184 RegisterMsgHdr(HKEY("subj"), examine_subj, 0);
1185 RegisterMsgHdr(HKEY("msgn"), examine_msgn, 0);
1186 RegisterMsgHdr(HKEY("wefw"), examine_wefw, 0);
1187 RegisterMsgHdr(HKEY("cccc"), examine_cccc, 0);
1188 RegisterMsgHdr(HKEY("hnod"), examine_hnod, 0);
1189 RegisterMsgHdr(HKEY("room"), examine_room, 0);
1190 RegisterMsgHdr(HKEY("rfca"), examine_rfca, 0);
1191 RegisterMsgHdr(HKEY("node"), examine_node, 0);
1192 RegisterMsgHdr(HKEY("rcpt"), examine_rcpt, 0);
1193 RegisterMsgHdr(HKEY("time"), examine_time, 0);
1194 RegisterMsgHdr(HKEY("part"), examine_mime_part, 0);
1195 RegisterMsgHdr(HKEY("text"), examine_text, 1);
1196 /* these are the content-type headers we get infront of a message; put it into the same hash since it doesn't clash. */
1197 RegisterMsgHdr(HKEY("X-Citadel-MSG4-Partnum"), examine_msg4_partnum, 0);
1198 RegisterMsgHdr(HKEY("Content-type"), examine_content_type, 0);
1199 RegisterMsgHdr(HKEY("Content-length"), examine_content_lengh, 0);
1200 RegisterMsgHdr(HKEY("Content-transfer-encoding"), examine_content_encoding, 0); /* do we care? */
1201 RegisterMsgHdr(HKEY("charset"), examine_charset, 0);
1203 /* Don't care about these... */
1204 RegisterMsgHdr(HKEY("pref"), examine_pref, 0);
1205 RegisterMsgHdr(HKEY("suff"), examine_suff, 0);
1206 RegisterMsgHdr(HKEY("path"), examine_path, 0);