source: message.c @ ee6b30f

Last change on this file since ee6b30f was ee6b30f, checked in by Anders Kaseorg <andersk@mit.edu>, 7 years ago
Avoid thread-unsafe functions ctime, localtime, strtok BarnOwl seems to be growing threads at an alarming rate, so let’s be sure these don’t turn into race conditions. Signed-off-by: Anders Kaseorg <andersk@mit.edu>
  • Property mode set to 100644
File size: 24.8 KB
Line 
1#include "owl.h"
2#include "filterproc.h"
3#include <sys/socket.h>
4#include <arpa/inet.h>
5
6static owl_fmtext_cache fmtext_cache[OWL_FMTEXT_CACHE_SIZE];
7static owl_fmtext_cache * fmtext_cache_next = fmtext_cache;
8
9void owl_message_init_fmtext_cache(void)
10{
11    int i;
12    for(i = 0; i < OWL_FMTEXT_CACHE_SIZE; i++) {
13        owl_fmtext_init_null(&(fmtext_cache[i].fmtext));
14        fmtext_cache[i].message = NULL;
15    }
16}
17
18static owl_fmtext_cache *owl_message_next_fmtext(void)
19{
20    owl_fmtext_cache * f = fmtext_cache_next;
21    if(fmtext_cache_next->message != NULL) {
22        owl_message_invalidate_format(fmtext_cache_next->message);
23    }
24    fmtext_cache_next++;
25    if(fmtext_cache_next - fmtext_cache == OWL_FMTEXT_CACHE_SIZE)
26        fmtext_cache_next = fmtext_cache;
27    return f;
28}
29
30void owl_message_init(owl_message *m)
31{
32  /* ctime_r requires a 26-byte buffer */
33  char timestr[26];
34
35  m->id=owl_global_get_nextmsgid(&g);
36  owl_message_set_direction_none(m);
37  m->delete=0;
38
39#ifdef HAVE_LIBZEPHYR
40  m->has_notice = false;
41#endif
42
43  owl_message_set_hostname(m, "");
44  m->attributes = g_ptr_array_new();
45 
46  /* save the time */
47  m->time = time(NULL);
48  ctime_r(&m->time, timestr);
49  m->timestr = g_strndup(timestr, strlen(timestr) - 1);
50
51  m->fmtext = NULL;
52}
53
54/* add the named attribute to the message.  If an attribute with the
55 * name already exists, replace the old value with the new value
56 */
57void owl_message_set_attribute(owl_message *m, const char *attrname, const char *attrvalue)
58{
59  int i;
60  owl_pair *p = NULL, *pair = NULL;
61
62  attrname = g_intern_string(attrname);
63
64  /* look for an existing pair with this key, */
65  for (i = 0; i < m->attributes->len; i++) {
66    p = m->attributes->pdata[i];
67    if (owl_pair_get_key(p) == attrname) {
68      g_free(owl_pair_get_value(p));
69      pair = p;
70      break;
71    }
72  }
73
74  if(pair ==  NULL) {
75    pair = g_slice_new(owl_pair);
76    owl_pair_create(pair, attrname, NULL);
77    g_ptr_array_add(m->attributes, pair);
78  }
79  owl_pair_set_value(pair, owl_validate_or_convert(attrvalue));
80}
81
82/* return the value associated with the named attribute, or NULL if
83 * the attribute does not exist
84 */
85const char *owl_message_get_attribute_value(const owl_message *m, const char *attrname)
86{
87  int i;
88  owl_pair *p;
89  GQuark quark;
90
91  quark = g_quark_try_string(attrname);
92  if (quark == 0)
93    /* don't bother inserting into string table */
94    return NULL;
95  attrname = g_quark_to_string(quark);
96
97  for (i = 0; i < m->attributes->len; i++) {
98    p = m->attributes->pdata[i];
99    if (owl_pair_get_key(p) == attrname) {
100      return(owl_pair_get_value(p));
101    }
102  }
103
104  /*
105  owl_function_debugmsg("No attribute %s found for message %i",
106                        attrname,
107                        owl_message_get_id(m));
108  */
109  return(NULL);
110}
111
112/* We cheat and indent it for now, since we really want this for
113 * the 'info' function.  Later there should just be a generic
114 * function to indent fmtext.
115 */
116void owl_message_attributes_tofmtext(const owl_message *m, owl_fmtext *fm) {
117  int i;
118  owl_pair *p;
119  char *buff, *tmpbuff;
120
121  owl_fmtext_init_null(fm);
122
123  for (i = 0; i < m->attributes->len; i++) {
124    p = m->attributes->pdata[i];
125
126    buff = g_strdup(owl_pair_get_value(p));
127    if (buff) {
128      tmpbuff = owl_text_indent(buff, 19, false);
129      g_free(buff);
130      buff = g_strdup_printf("  %-15.15s: %s\n", owl_pair_get_key(p), tmpbuff);
131      g_free(tmpbuff);
132    }
133
134    if(buff == NULL) {
135      buff = g_strdup_printf("  %-15.15s: %s\n", owl_pair_get_key(p), "<error>");
136      if(buff == NULL)
137        buff=g_strdup("   <error>\n");
138    }
139    owl_fmtext_append_normal(fm, buff);
140    g_free(buff);
141  }
142}
143
144void owl_message_invalidate_format(owl_message *m)
145{
146  if(m->fmtext) {
147    m->fmtext->message = NULL;
148    owl_fmtext_clear(&(m->fmtext->fmtext));
149    m->fmtext=NULL;
150  }
151}
152
153owl_fmtext *owl_message_get_fmtext(owl_message *m)
154{
155  owl_message_format(m);
156  return(&(m->fmtext->fmtext));
157}
158
159void owl_message_format(owl_message *m)
160{
161  const owl_style *s;
162  const owl_view *v;
163
164  if (!m->fmtext) {
165    m->fmtext = owl_message_next_fmtext();
166    m->fmtext->message = m;
167    /* for now we assume there's just the one view and use that style */
168    v=owl_global_get_current_view(&g);
169    s=owl_view_get_style(v);
170
171    owl_style_get_formattext(s, &(m->fmtext->fmtext), m);
172  }
173}
174
175void owl_message_set_class(owl_message *m, const char *class)
176{
177  owl_message_set_attribute(m, "class", class);
178}
179
180const char *owl_message_get_class(const owl_message *m)
181{
182  const char *class;
183
184  class=owl_message_get_attribute_value(m, "class");
185  if (!class) return("");
186  return(class);
187}
188
189void owl_message_set_instance(owl_message *m, const char *inst)
190{
191  owl_message_set_attribute(m, "instance", inst);
192}
193
194const char *owl_message_get_instance(const owl_message *m)
195{
196  const char *instance;
197
198  instance=owl_message_get_attribute_value(m, "instance");
199  if (!instance) return("");
200  return(instance);
201}
202
203void owl_message_set_sender(owl_message *m, const char *sender)
204{
205  owl_message_set_attribute(m, "sender", sender);
206}
207
208const char *owl_message_get_sender(const owl_message *m)
209{
210  const char *sender;
211
212  sender=owl_message_get_attribute_value(m, "sender");
213  if (!sender) return("");
214  return(sender);
215}
216
217void owl_message_set_zsig(owl_message *m, const char *zsig)
218{
219  owl_message_set_attribute(m, "zsig", zsig);
220}
221
222const char *owl_message_get_zsig(const owl_message *m)
223{
224  const char *zsig;
225
226  zsig=owl_message_get_attribute_value(m, "zsig");
227  if (!zsig) return("");
228  return(zsig);
229}
230
231void owl_message_set_recipient(owl_message *m, const char *recip)
232{
233  owl_message_set_attribute(m, "recipient", recip);
234}
235
236const char *owl_message_get_recipient(const owl_message *m)
237{
238  /* this is stupid for outgoing messages, we need to fix it. */
239
240  const char *recip;
241
242  recip=owl_message_get_attribute_value(m, "recipient");
243  if (!recip) return("");
244  return(recip);
245}
246
247void owl_message_set_realm(owl_message *m, const char *realm)
248{
249  owl_message_set_attribute(m, "realm", realm);
250}
251
252const char *owl_message_get_realm(const owl_message *m)
253{
254  const char *realm;
255 
256  realm=owl_message_get_attribute_value(m, "realm");
257  if (!realm) return("");
258  return(realm);
259}
260
261void owl_message_set_body(owl_message *m, const char *body)
262{
263  owl_message_set_attribute(m, "body", body);
264}
265
266const char *owl_message_get_body(const owl_message *m)
267{
268  const char *body;
269
270  body=owl_message_get_attribute_value(m, "body");
271  if (!body) return("");
272  return(body);
273}
274
275
276void owl_message_set_opcode(owl_message *m, const char *opcode)
277{
278  owl_message_set_attribute(m, "opcode", opcode);
279}
280
281const char *owl_message_get_opcode(const owl_message *m)
282{
283  const char *opcode;
284
285  opcode=owl_message_get_attribute_value(m, "opcode");
286  if (!opcode) return("");
287  return(opcode);
288}
289
290
291void owl_message_set_islogin(owl_message *m)
292{
293  owl_message_set_attribute(m, "loginout", "login");
294}
295
296
297void owl_message_set_islogout(owl_message *m)
298{
299  owl_message_set_attribute(m, "loginout", "logout");
300}
301
302int owl_message_is_loginout(const owl_message *m)
303{
304  const char *res;
305
306  res=owl_message_get_attribute_value(m, "loginout");
307  if (!res) return(0);
308  return(1);
309}
310
311int owl_message_is_login(const owl_message *m)
312{
313  const char *res;
314
315  res=owl_message_get_attribute_value(m, "loginout");
316  if (!res) return(0);
317  if (!strcmp(res, "login")) return(1);
318  return(0);
319}
320
321
322int owl_message_is_logout(const owl_message *m)
323{
324  const char *res;
325
326  res=owl_message_get_attribute_value(m, "loginout");
327  if (!res) return(0);
328  if (!strcmp(res, "logout")) return(1);
329  return(0);
330}
331
332void owl_message_set_isprivate(owl_message *m)
333{
334  owl_message_set_attribute(m, "isprivate", "true");
335}
336
337int owl_message_is_private(const owl_message *m)
338{
339  const char *res;
340
341  res=owl_message_get_attribute_value(m, "isprivate");
342  if (!res) return(0);
343  return !strcmp(res, "true");
344}
345
346const char *owl_message_get_timestr(const owl_message *m)
347{
348  if (m->timestr) return(m->timestr);
349  return("");
350}
351
352CALLER_OWN char *owl_message_format_time(const owl_message *m)
353{
354  struct tm tm;
355  return owl_util_format_time(localtime_r(&m->time, &tm));
356}
357
358void owl_message_set_type_admin(owl_message *m)
359{
360  owl_message_set_attribute(m, "type", "admin");
361}
362
363void owl_message_set_type_loopback(owl_message *m)
364{
365  owl_message_set_attribute(m, "type", "loopback");
366}
367
368void owl_message_set_type_zephyr(owl_message *m)
369{
370  owl_message_set_attribute(m, "type", "zephyr");
371}
372
373void owl_message_set_type_aim(owl_message *m)
374{
375  owl_message_set_attribute(m, "type", "AIM");
376}
377
378void owl_message_set_type(owl_message *m, const char* type)
379{
380  owl_message_set_attribute(m, "type", type);
381}
382
383int owl_message_is_type(const owl_message *m, const char *type) {
384  const char * t = owl_message_get_attribute_value(m, "type");
385  if(!t) return 0;
386  return !strcasecmp(t, type);
387}
388                                               
389int owl_message_is_type_admin(const owl_message *m)
390{
391  return owl_message_is_type(m, "admin");
392}
393
394int owl_message_is_type_zephyr(const owl_message *m)
395{
396  return owl_message_is_type(m, "zephyr");
397}
398
399int owl_message_is_type_aim(const owl_message *m)
400{
401  return owl_message_is_type(m, "aim");
402}
403
404int owl_message_is_type_loopback(const owl_message *m)
405{
406  return owl_message_is_type(m, "loopback");
407}
408
409int owl_message_is_pseudo(const owl_message *m)
410{
411  if (owl_message_get_attribute_value(m, "pseudo")) return(1);
412  return(0);
413}
414
415const char *owl_message_get_text(owl_message *m)
416{
417  owl_message_format(m);
418  return(owl_fmtext_get_text(&(m->fmtext->fmtext)));
419}
420
421void owl_message_set_direction_in(owl_message *m)
422{
423  m->direction=OWL_MESSAGE_DIRECTION_IN;
424}
425
426void owl_message_set_direction_out(owl_message *m)
427{
428  m->direction=OWL_MESSAGE_DIRECTION_OUT;
429}
430
431void owl_message_set_direction_none(owl_message *m)
432{
433  m->direction=OWL_MESSAGE_DIRECTION_NONE;
434}
435
436void owl_message_set_direction(owl_message *m, int direction)
437{
438  m->direction=direction;
439}
440
441int owl_message_is_direction_in(const owl_message *m)
442{
443  if (m->direction==OWL_MESSAGE_DIRECTION_IN) return(1);
444  return(0);
445}
446
447int owl_message_is_direction_out(const owl_message *m)
448{
449  if (m->direction==OWL_MESSAGE_DIRECTION_OUT) return(1);
450  return(0);
451}
452
453int owl_message_is_direction_none(const owl_message *m)
454{
455  if (m->direction==OWL_MESSAGE_DIRECTION_NONE) return(1);
456  return(0);
457}
458
459int owl_message_get_numlines(owl_message *m)
460{
461  if (m == NULL) return(0);
462  owl_message_format(m);
463  return(owl_fmtext_num_lines(&(m->fmtext->fmtext)));
464}
465
466void owl_message_mark_delete(owl_message *m)
467{
468  if (m == NULL) return;
469  m->delete=1;
470}
471
472void owl_message_unmark_delete(owl_message *m)
473{
474  if (m == NULL) return;
475  m->delete=0;
476}
477
478const char *owl_message_get_zwriteline(const owl_message *m)
479{
480  const char *z = owl_message_get_attribute_value(m, "zwriteline");
481  if (!z) return "";
482  return z;
483}
484
485void owl_message_set_zwriteline(owl_message *m, const char *line)
486{
487  owl_message_set_attribute(m, "zwriteline", line);
488}
489
490int owl_message_is_delete(const owl_message *m)
491{
492  if (m == NULL) return(0);
493  if (m->delete==1) return(1);
494  return(0);
495}
496
497#ifdef HAVE_LIBZEPHYR
498const ZNotice_t *owl_message_get_notice(const owl_message *m)
499{
500  return m->has_notice ? &m->notice : NULL;
501}
502#else
503void *owl_message_get_notice(const owl_message *m)
504{
505  return(NULL);
506}
507#endif
508
509void owl_message_set_hostname(owl_message *m, const char *hostname)
510{
511  m->hostname = g_intern_string(hostname);
512}
513
514const char *owl_message_get_hostname(const owl_message *m)
515{
516  return(m->hostname);
517}
518
519void owl_message_curs_waddstr(owl_message *m, WINDOW *win, int aline, int bline, int acol, int bcol, int fgcolor, int bgcolor)
520{
521  owl_fmtext a, b;
522
523  /* this will ensure that our cached copy is up to date */
524  owl_message_format(m);
525
526  owl_fmtext_init_null(&a);
527  owl_fmtext_init_null(&b);
528 
529  owl_fmtext_truncate_lines(&(m->fmtext->fmtext), aline, bline-aline, &a);
530  owl_fmtext_truncate_cols(&a, acol, bcol, &b);
531
532  owl_fmtext_curs_waddstr(&b, win, OWL_FMTEXT_ATTR_NONE, fgcolor, bgcolor);
533
534  owl_fmtext_cleanup(&a);
535  owl_fmtext_cleanup(&b);
536}
537
538int owl_message_is_personal(const owl_message *m)
539{
540  const owl_filter * f = owl_global_get_filter(&g, "personal");
541  if(!f) {
542      owl_function_error("personal filter is not defined");
543      return (0);
544  }
545  return owl_filter_message_match(f, m);
546}
547
548int owl_message_is_question(const owl_message *m)
549{
550  if(!owl_message_is_type_admin(m)) return 0;
551  if(owl_message_get_attribute_value(m, "question") != NULL) return 1;
552  return 0;
553}
554
555int owl_message_is_answered(const owl_message *m) {
556  const char *q;
557  if(!owl_message_is_question(m)) return 0;
558  q = owl_message_get_attribute_value(m, "question");
559  if(!q) return 0;
560  return !strcmp(q, "answered");
561}
562
563void owl_message_set_isanswered(owl_message *m) {
564  owl_message_set_attribute(m, "question", "answered");
565}
566
567int owl_message_is_mail(const owl_message *m)
568{
569  if (owl_message_is_type_zephyr(m)) {
570    if (!strcasecmp(owl_message_get_class(m), "mail") && owl_message_is_private(m)) {
571      return(1);
572    } else {
573      return(0);
574    }
575  }
576  return(0);
577}
578
579/* caller must free return value. */
580CALLER_OWN char *owl_message_get_cc(const owl_message *m)
581{
582  const char *cur;
583  char *out, *end;
584
585  cur = owl_message_get_body(m);
586  while (*cur && *cur==' ') cur++;
587  if (strncasecmp(cur, "cc:", 3)) return(NULL);
588  cur+=3;
589  while (*cur && *cur==' ') cur++;
590  out = g_strdup(cur);
591  end = strchr(out, '\n');
592  if (end) end[0] = '\0';
593  return(out);
594}
595
596/* caller must free return value */
597CALLER_OWN GList *owl_message_get_cc_without_recipient(const owl_message *m)
598{
599  char *cc, *shortuser, *recip, *saveptr;
600  const char *user;
601  GList *out = NULL;
602
603  cc = owl_message_get_cc(m);
604  if (cc == NULL)
605    return NULL;
606
607  recip = short_zuser(owl_message_get_recipient(m));
608
609  user = strtok_r(cc, " ", &saveptr);
610  while (user != NULL) {
611    shortuser = short_zuser(user);
612    if (strcasecmp(shortuser, recip) != 0) {
613      out = g_list_prepend(out, g_strdup(user));
614    }
615    g_free(shortuser);
616    user = strtok_r(NULL, " ", &saveptr);
617  }
618
619  g_free(recip);
620  g_free(cc);
621
622  return(out);
623}
624
625int owl_message_get_id(const owl_message *m)
626{
627  return(m->id);
628}
629
630const char *owl_message_get_type(const owl_message *m) {
631  const char * type = owl_message_get_attribute_value(m, "type");
632  if(!type) {
633    return "generic";
634  }
635  return type;
636}
637
638const char *owl_message_get_direction(const owl_message *m) {
639  switch (m->direction) {
640  case OWL_MESSAGE_DIRECTION_IN:
641    return("in");
642  case OWL_MESSAGE_DIRECTION_OUT:
643    return("out");
644  case OWL_MESSAGE_DIRECTION_NONE:
645    return("none");
646  default:
647    return("unknown");
648  }
649}
650
651int owl_message_parse_direction(const char *d) {
652  if(!strcmp(d, "in")) {
653    return OWL_MESSAGE_DIRECTION_IN;
654  } else if(!strcmp(d, "out")) {
655    return OWL_MESSAGE_DIRECTION_OUT;
656  } else {
657    return OWL_MESSAGE_DIRECTION_NONE;
658  }
659}
660
661
662const char *owl_message_get_login(const owl_message *m) {
663  if (owl_message_is_login(m)) {
664    return "login";
665  } else if (owl_message_is_logout(m)) {
666    return "logout";
667  } else {
668    return "none";
669  }
670}
671
672
673const char *owl_message_get_header(const owl_message *m) {
674  return owl_message_get_attribute_value(m, "adminheader");
675}
676
677/* return 1 if the message contains "string", 0 otherwise.  This is
678 * case insensitive because the functions it uses are
679 */
680int owl_message_search(owl_message *m, const owl_regex *re)
681{
682
683  owl_message_format(m); /* is this necessary? */
684 
685  return owl_fmtext_search(&(m->fmtext->fmtext), re, 0) >= 0;
686}
687
688
689/* if loginout == -1 it's a logout message
690 *                 0 it's not a login/logout message
691 *                 1 it's a login message
692 */
693void owl_message_create_aim(owl_message *m, const char *sender, const char *recipient, const char *text, int direction, int loginout)
694{
695  owl_message_init(m);
696  owl_message_set_body(m, text);
697  owl_message_set_sender(m, sender);
698  owl_message_set_recipient(m, recipient);
699  owl_message_set_type_aim(m);
700
701  if (direction==OWL_MESSAGE_DIRECTION_IN) {
702    owl_message_set_direction_in(m);
703  } else if (direction==OWL_MESSAGE_DIRECTION_OUT) {
704    owl_message_set_direction_out(m);
705  }
706
707  /* for now all messages that aren't loginout are private */
708  if (!loginout) {
709    owl_message_set_isprivate(m);
710  }
711
712  if (loginout==-1) {
713    owl_message_set_islogout(m);
714  } else if (loginout==1) {
715    owl_message_set_islogin(m);
716  }
717}
718
719void owl_message_create_admin(owl_message *m, const char *header, const char *text)
720{
721  owl_message_init(m);
722  owl_message_set_type_admin(m);
723  owl_message_set_body(m, text);
724  owl_message_set_attribute(m, "adminheader", header); /* just a hack for now */
725}
726
727/* caller should set the direction */
728void owl_message_create_loopback(owl_message *m, const char *text)
729{
730  owl_message_init(m);
731  owl_message_set_type_loopback(m);
732  owl_message_set_body(m, text);
733  owl_message_set_sender(m, "loopsender");
734  owl_message_set_recipient(m, "looprecip");
735  owl_message_set_isprivate(m);
736}
737
738void owl_message_save_ccs(owl_message *m) {
739  GList *cc;
740  char *tmp;
741
742  cc = owl_message_get_cc_without_recipient(m);
743
744  if (cc != NULL) {
745    GString *recips = g_string_new("");
746    cc = g_list_prepend(cc, short_zuser(owl_message_get_sender(m)));
747    cc = g_list_prepend(cc, short_zuser(owl_message_get_recipient(m)));
748    cc = g_list_sort(cc, (GCompareFunc)strcasecmp);
749
750    while(cc != NULL) {
751      /* Collapse any identical entries */
752      while (cc->next && strcasecmp(cc->data, cc->next->data) == 0) {
753        g_free(cc->data);
754        cc = g_list_delete_link(cc, cc);
755      }
756
757      tmp = short_zuser(cc->data);
758      g_string_append(recips, tmp);
759
760      g_free(tmp);
761      g_free(cc->data);
762      cc = g_list_delete_link(cc, cc);
763
764      if (cc)
765        g_string_append_c(recips, ' ');
766    }
767
768    owl_message_set_attribute(m, "zephyr_ccs", recips->str);
769    g_string_free(recips, true);
770  }
771}
772
773#ifdef HAVE_LIBZEPHYR
774void owl_message_create_from_znotice(owl_message *m, const ZNotice_t *n)
775{
776#ifdef ZNOTICE_SOCKADDR
777  char hbuf[NI_MAXHOST];
778#else /* !ZNOTICE_SOCKADDR */
779  struct hostent *hent;
780#endif /* ZNOTICE_SOCKADDR */
781  /* ctime_r requires a 26-byte buffer */
782  char timestr[26], *tmp, *tmp2;
783  int len;
784
785  owl_message_init(m);
786 
787  owl_message_set_type_zephyr(m);
788  owl_message_set_direction_in(m);
789 
790  /* first save the full notice */
791  m->notice = *n;
792  m->has_notice = true;
793
794  /* a little gross, we'll replace \r's with ' ' for now */
795  owl_zephyr_hackaway_cr(&(m->notice));
796 
797  /* save the time, we need to nuke the string saved by message_init */
798  if (m->timestr) g_free(m->timestr);
799  m->time = n->z_time.tv_sec;
800  ctime_r(&m->time, timestr);
801  m->timestr = g_strndup(timestr, strlen(timestr) - 1);
802
803  /* set other info */
804  owl_message_set_sender(m, n->z_sender);
805  owl_message_set_class(m, n->z_class);
806  owl_message_set_instance(m, n->z_class_inst);
807  owl_message_set_recipient(m, n->z_recipient);
808  if (n->z_opcode) {
809    owl_message_set_opcode(m, n->z_opcode);
810  } else {
811    owl_message_set_opcode(m, "");
812  }
813  owl_message_set_zsig(m, owl_zephyr_get_zsig(n, &len));
814
815  owl_message_set_realm(m, zuser_realm(n->z_recipient));
816
817  /* Set the "isloginout" attribute if it's a login message */
818  if (!strcasecmp(n->z_class, "login") || !strcasecmp(n->z_class, OWL_WEBZEPHYR_CLASS)) {
819    if (!strcasecmp(n->z_opcode, "user_login") || !strcasecmp(n->z_opcode, "user_logout")) {
820      tmp=owl_zephyr_get_field(n, 1);
821      owl_message_set_attribute(m, "loginhost", tmp);
822      g_free(tmp);
823
824      tmp=owl_zephyr_get_field(n, 3);
825      owl_message_set_attribute(m, "logintty", tmp);
826      g_free(tmp);
827    }
828
829    if (!strcasecmp(n->z_opcode, "user_login")) {
830      owl_message_set_islogin(m);
831    } else if (!strcasecmp(n->z_opcode, "user_logout")) {
832      owl_message_set_islogout(m);
833    }
834  }
835
836 
837  /* set the "isprivate" attribute if it's a private zephyr. */
838  if (owl_zwrite_recip_is_personal(n->z_recipient)) {
839    owl_message_set_isprivate(m);
840  }
841
842  /* set the "isauto" attribute if it's an autoreply */
843  if (!strcasecmp(n->z_message, "Automated reply:") ||
844      !strcasecmp(n->z_opcode, "auto")) {
845    owl_message_set_attribute(m, "isauto", "");
846  }
847
848  /* save the hostname */
849#ifdef ZNOTICE_SOCKADDR
850  owl_function_debugmsg("About to do getnameinfo");
851  if (getnameinfo(&n->z_sender_sockaddr.sa, sizeof(n->z_sender_sockaddr), hbuf, sizeof(hbuf), NULL, 0, 0) == 0)
852    owl_message_set_hostname(m, hbuf);
853#else /* !ZNOTICE_SOCKADDR */
854  owl_function_debugmsg("About to do gethostbyaddr");
855  hent = gethostbyaddr(&n->z_uid.zuid_addr, sizeof(n->z_uid.zuid_addr), AF_INET);
856  if (hent && hent->h_name)
857    owl_message_set_hostname(m, hent->h_name);
858  else
859    owl_message_set_hostname(m, inet_ntoa(n->z_sender_addr));
860#endif /* ZNOTICE_SOCKADDR */
861
862  /* set the body */
863  tmp=owl_zephyr_get_message(n, m);
864  if (owl_global_is_newlinestrip(&g)) {
865    tmp2=owl_util_stripnewlines(tmp);
866    owl_message_set_body(m, tmp2);
867    g_free(tmp2);
868  } else {
869    owl_message_set_body(m, tmp);
870  }
871  g_free(tmp);
872
873  /* if zcrypt is enabled try to decrypt the message */
874  if (owl_global_is_zcrypt(&g) && !strcasecmp(n->z_opcode, "crypt")) {
875    const char *argv[] = {
876      NULL,
877      "-D",
878      "-c", owl_message_get_class(m),
879      "-i", owl_message_get_instance(m),
880      NULL
881    };
882    char *out = NULL;
883    int rv;
884    int status;
885    char *zcrypt;
886
887    zcrypt = g_build_filename(owl_get_bindir(), "zcrypt", NULL);
888    argv[0] = zcrypt;
889
890    rv = call_filter(argv, owl_message_get_body(m), &out, &status);
891    g_free(zcrypt);
892
893    if(!rv && !status) {
894      int len = strlen(out);
895      if(len >= 8 && !strcmp(out + len - 8, "**END**\n")) {
896        out[len - 8] = 0;
897      }
898      owl_message_set_body(m, out);
899    } else {
900      /* Replace the opcode. Otherwise the UI and other bits of code think the
901       * message was encrypted. */
902      owl_message_set_opcode(m, "failed-decrypt");
903    }
904    g_free(out);
905  }
906
907  owl_message_save_ccs(m);
908}
909#else
910void owl_message_create_from_znotice(owl_message *m, const void *n)
911{
912}
913#endif
914
915/* If 'direction' is '0' it is a login message, '1' is a logout message. */
916void owl_message_create_pseudo_zlogin(owl_message *m, int direction, const char *user, const char *host, const char *time, const char *tty)
917{
918  char *longuser;
919
920#ifdef HAVE_LIBZEPHYR
921  memset(&(m->notice), 0, sizeof(ZNotice_t));
922#endif
923 
924  longuser=long_zuser(user);
925 
926  owl_message_init(m);
927 
928  owl_message_set_type_zephyr(m);
929  owl_message_set_direction_in(m);
930
931  owl_message_set_attribute(m, "pseudo", "");
932  owl_message_set_attribute(m, "loginhost", host ? host : "");
933  owl_message_set_attribute(m, "logintty", tty ? tty : "");
934
935  owl_message_set_sender(m, longuser);
936  owl_message_set_class(m, "LOGIN");
937  owl_message_set_instance(m, longuser);
938  owl_message_set_recipient(m, "");
939  if (direction==0) {
940    owl_message_set_opcode(m, "USER_LOGIN");
941    owl_message_set_islogin(m);
942  } else if (direction==1) {
943    owl_message_set_opcode(m, "USER_LOGOUT");
944    owl_message_set_islogout(m);
945  }
946
947  owl_message_set_realm(m, zuser_realm(longuser));
948
949  owl_message_set_body(m, "<uninitialized>");
950
951  /* save the hostname */
952  owl_function_debugmsg("create_pseudo_login: host is %s", host ? host : "");
953  owl_message_set_hostname(m, host ? host : "");
954  g_free(longuser);
955}
956
957void owl_message_create_from_zwrite(owl_message *m, const owl_zwrite *z, const char *body, int recip_index)
958{
959  char *replyline;
960 
961  owl_message_init(m);
962
963  /* set things */
964  owl_message_set_direction_out(m);
965  owl_message_set_type_zephyr(m);
966  owl_message_set_sender(m, owl_zephyr_get_sender());
967  owl_message_set_class(m, owl_zwrite_get_class(z));
968  owl_message_set_instance(m, owl_zwrite_get_instance(z));
969  if (recip_index < owl_zwrite_get_numrecips(z)) {
970    char *zuser = owl_zwrite_get_recip_n_with_realm(z, recip_index);
971    char *longzuser = long_zuser(zuser);
972    owl_message_set_recipient(m, longzuser);
973    owl_message_set_realm(m, zuser_realm(longzuser));
974    g_free(longzuser);
975    g_free(zuser);
976  } else {
977    /* TODO: We should maybe munge this into the case above, as that comparison
978     * is a little overly clever. It's also not clear this codepath ever runs
979     * anyway. */
980    const char *realm = owl_zwrite_get_realm(z);
981    owl_message_set_realm(m, realm[0] ? realm : owl_zephyr_get_realm());
982  }
983  owl_message_set_opcode(m, owl_zwrite_get_opcode(z));
984
985  /* Although not strictly the zwriteline, anyone using the unsantized version
986   * of it probably has a bug. */
987  replyline = owl_zwrite_get_replyline(z, recip_index);
988  owl_message_set_zwriteline(m, replyline);
989  g_free(replyline);
990
991  owl_message_set_body(m, body);
992  owl_message_set_zsig(m, owl_zwrite_get_zsig(z));
993 
994  /* save the hostname */
995  owl_message_set_hostname(m, g_get_host_name());
996
997  /* set the "isprivate" attribute if it's a private zephyr. */
998  if (owl_zwrite_is_personal(z)) {
999    owl_message_set_isprivate(m);
1000  }
1001
1002  owl_message_save_ccs(m);
1003}
1004
1005void owl_message_cleanup(owl_message *m)
1006{
1007  int i;
1008  owl_pair *p;
1009#ifdef HAVE_LIBZEPHYR   
1010  if (m->has_notice) {
1011    ZFreeNotice(&(m->notice));
1012  }
1013#endif
1014  if (m->timestr) g_free(m->timestr);
1015
1016  /* free all the attributes */
1017  for (i = 0; i < m->attributes->len; i++) {
1018    p = m->attributes->pdata[i];
1019    g_free(owl_pair_get_value(p));
1020    g_slice_free(owl_pair, p);
1021  }
1022
1023  g_ptr_array_free(m->attributes, true);
1024 
1025  owl_message_invalidate_format(m);
1026}
1027
1028void owl_message_delete(owl_message *m)
1029{
1030  owl_message_cleanup(m);
1031  g_slice_free(owl_message, m);
1032}
Note: See TracBrowser for help on using the repository browser.