source: zephyr.c @ 7d4fbcd

barnowl_perlaimdebianowlrelease-1.10release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since 7d4fbcd was 7d4fbcd, checked in by James M. Kretchmar <kretch@mit.edu>, 22 years ago
Initial check in
  • Property mode set to 100644
File size: 11.2 KB
Line 
1#include <stdlib.h>
2#include <unistd.h>
3#include <sys/types.h>
4#include <sys/wait.h>
5#include <string.h>
6#include <zephyr/zephyr.h>
7#include <com_err.h>
8#include <pwd.h>
9#include "owl.h"
10
11int owl_zephyr_loadsubs(char *filename) {
12  /* return 0  on success
13   *        -1 on file error
14   *        -2 on subscription error
15   */
16  FILE *file;
17  char *tmp, *start;
18  char buffer[1024], subsfile[1024];
19  ZSubscription_t subs[3001];
20  int count, ret, i;
21
22  ret=0;
23 
24  if (filename==NULL) {
25    sprintf(subsfile, "%s/%s", owl_global_get_homedir(&g), ".zephyr.subs");
26  } else {
27    strcpy(subsfile, filename);
28  }
29 
30  /* need to redo this to do chunks, not just bail after 3000 */
31  count=0;
32  file=fopen(subsfile, "r");
33  if (file) {
34    while ( fgets(buffer, 1024, file)!=NULL ) {
35      if (buffer[0]=='#' || buffer[0]=='\n' || buffer[0]=='\n') continue;
36
37      if (buffer[0]=='-') {
38        start=buffer+1;
39      } else {
40        start=buffer;
41      }
42     
43      if (count >= 3000) break; /* also tell the user */
44
45      /* add it to the list of subs */
46      if ((tmp=(char *) strtok(start, ",\n\r"))==NULL) continue;
47      subs[count].zsub_class=owl_strdup(tmp);
48      if ((tmp=(char *) strtok(NULL, ",\n\r"))==NULL) continue;
49      subs[count].zsub_classinst=owl_strdup(tmp);
50      if ((tmp=(char *) strtok(NULL, " \t\n\r"))==NULL) continue;
51      subs[count].zsub_recipient=owl_strdup(tmp);
52
53      /* if it started with '-' then add it to the global punt list */
54      if (buffer[0]=='-') {
55        owl_function_zpunt(subs[count].zsub_class, subs[count].zsub_classinst, subs[count].zsub_recipient, 0);
56      }
57     
58      count++;
59    }
60    fclose(file);
61  } else {
62    count=0;
63    ret=-1;
64  }
65
66  /* sub with defaults */
67  if (ZSubscribeTo(subs,count,0) != ZERR_NONE) {
68    fprintf(stderr, "Error subbing\n");
69    ret=-2;
70  }
71
72  /* free stuff */
73  for (i=0; i<count; i++) {
74    owl_free(subs[i].zsub_class);
75    owl_free(subs[i].zsub_classinst);
76    owl_free(subs[i].zsub_recipient);
77  }
78
79  return(ret);
80}
81
82int loadloginsubs(char *filename) {
83  FILE *file;
84  ZSubscription_t subs[3001];
85  char subsfile[1024], buffer[1024];
86  int count, ret, i;
87
88  ret=0;
89
90  if (filename==NULL) {
91    sprintf(subsfile, "%s/%s", owl_global_get_homedir(&g), ".anyone");
92  } else {
93    strcpy(subsfile, filename);
94  }
95
96  /* need to redo this to do chunks, not just bag out after 3000 */
97  count=0;
98  file=fopen(subsfile, "r");
99  if (file) {
100    while ( fgets(buffer, 1024, file)!=NULL ) {
101      if (buffer[0]=='#' || buffer[0]=='\n' || buffer[0]=='\n') continue;
102     
103      if (count >= 3000) break; /* also tell the user */
104
105      buffer[strlen(buffer)-1]='\0';
106      subs[count].zsub_class="login";
107      subs[count].zsub_recipient="*";
108      if (strchr(buffer, '@')) {
109        subs[count].zsub_classinst=owl_strdup(buffer);
110      } else {
111        subs[count].zsub_classinst=owl_malloc(1024);
112        sprintf(subs[count].zsub_classinst, "%s@%s", buffer, ZGetRealm());
113      }
114
115      count++;
116    }
117    fclose(file);
118  } else {
119    count=0;
120    ret=-1;
121  }
122
123  /* sub with defaults */
124  if (ZSubscribeTo(subs,count,0) != ZERR_NONE) {
125    fprintf(stderr, "Error subbing\n");
126    ret=-2;
127  }
128
129  /* free stuff */
130  for (i=0; i<count; i++) {
131    owl_free(subs[i].zsub_classinst);
132  }
133
134  return(ret);
135}
136
137void unsuball() {
138  int ret;
139 
140  ret=ZCancelSubscriptions(0);
141  if (ret != ZERR_NONE) {
142    com_err("owl",ret,"while unsubscribing");
143  }
144}
145
146int owl_zephyr_sub(char *class, char *inst, char *recip) {
147  ZSubscription_t subs[5];
148  int ret;
149
150  subs[0].zsub_class=class;
151  subs[0].zsub_classinst=inst;
152  subs[0].zsub_recipient=recip;
153
154  if (ZSubscribeTo(subs,1,0) != ZERR_NONE) {
155    fprintf(stderr, "Error subbing\n");
156    ret=-2;
157  }
158  return(0);
159}
160
161
162int owl_zephyr_unsub(char *class, char *inst, char *recip) {
163  ZSubscription_t subs[5];
164  int ret;
165
166  subs[0].zsub_class=class;
167  subs[0].zsub_classinst=inst;
168  subs[0].zsub_recipient=recip;
169
170  if (ZUnsubscribeTo(subs,1,0) != ZERR_NONE) {
171    fprintf(stderr, "Error unsubbing\n");
172    ret=-2;
173  }
174  return(0);
175}
176
177
178void zlog_in() {
179  int ret;
180
181  ret=ZSetLocation(EXPOSE_NETVIS);
182  /* do something on failure */
183}
184
185void zlog_out() {
186  int ret;
187
188  ret=ZUnsetLocation();
189  /* do something on failure */
190}
191
192
193char *owl_zephyr_get_field(ZNotice_t *n, int j, int *k) {
194  /* return a pointer to the Jth field, place the length in k.  If the
195     field doesn't exist return an emtpy string */
196  int i, count, save;
197
198  count=save=0;
199  for (i=0; i<n->z_message_len; i++) {
200    if (n->z_message[i]=='\0') {
201      count++;
202      if (count==j) {
203        /* just found the end of the field we're looking for */
204        *k=i-save;
205        return(n->z_message+save);
206      } else {
207        save=i+1;
208      }
209    }
210  }
211  /* catch the last field */
212  if (count==j-1) {
213    *k=n->z_message_len-save;
214    return(n->z_message+save);
215  }
216 
217  *k=0;
218  return("");
219}
220
221
222int owl_zephyr_get_num_fields(ZNotice_t *n) {
223  int i, fields;
224
225  fields=1;
226  for (i=0; i<n->z_message_len; i++) {
227    if (n->z_message[i]=='\0') fields++;
228  }
229 
230  return(fields);
231}
232
233
234char *owl_zephyr_get_message(ZNotice_t *n, int *k) {
235  /* return a pointer to the message, place the message length in k */
236  if (!strcasecmp(n->z_opcode, "ping")) {
237    *k=0;
238    return("");
239  }
240
241  return(owl_zephyr_get_field(n, 2, k));
242}
243
244
245char *owl_zephyr_get_zsig(ZNotice_t *n, int *k) {
246  /* return a pointer to the zsig if there is one */
247
248  if (n->z_message_len==0) {
249    *k=0;
250    return("");
251  }
252  *k=strlen(n->z_message);
253  return(n->z_message);
254}
255
256
257int send_zephyr(char *opcode, char *zsig, char *class, char *instance, char *recipient, char *message) {
258  int ret;
259  ZNotice_t notice;
260  char *ptr;
261  struct passwd *pw;
262  char zsigtmp[LINE];
263  char *zsigexec, *zsigowlvar, *zsigzvar;
264 
265  zsigexec = owl_global_get_zsig_exec(&g);
266  zsigowlvar = owl_global_get_zsig(&g);
267  zsigzvar = ZGetVariable("zwrite-signature");
268
269  if (zsig) {
270    strcpy(zsigtmp, zsig);
271  } else if (zsigowlvar && *zsigowlvar) {
272    strncpy(zsigtmp, zsigowlvar, LINE);
273  } else if (zsigexec && *zsigexec) {
274    FILE *file;
275    char buff[LINE];
276    strcpy(zsigtmp, "");
277    file=popen(zsigexec, "r");
278    if (!file) {
279      if (zsigzvar && *zsigzvar) {
280        strncpy(zsigtmp, zsigzvar, LINE);
281      }
282    } else {
283      while (fgets(buff, LINE, file)) { /* wrong sizing */
284        strcat(zsigtmp, buff);
285      }
286      pclose(file);
287      if (zsigtmp[strlen(zsigtmp)-1]=='\n') {
288        zsigtmp[strlen(zsigtmp)-1]='\0';
289      }
290    }
291  } else if (zsigzvar) {
292    strncpy(zsigtmp, zsigzvar, LINE);
293  } else if (((pw=getpwuid(getuid()))!=NULL) && (pw->pw_gecos)) {
294    strncpy(zsigtmp, pw->pw_gecos, LINE);
295    ptr=strchr(zsigtmp, ',');
296    if (ptr) {
297      ptr[0]='\0';
298    }
299  } else {
300    strcpy(zsigtmp, "");
301  }
302   
303  memset(&notice, 0, sizeof(notice));
304
305  notice.z_kind=ACKED;
306  notice.z_port=0;
307  notice.z_class=class;
308  notice.z_class_inst=instance;
309  if (!strcmp(recipient, "*") || !strcmp(recipient, "@")) {
310    notice.z_recipient="";
311  } else {
312    notice.z_recipient=recipient;
313  }
314  notice.z_default_format="Class $class, Instance $instance:\nTo: @bold($recipient) at $time $date\nFrom: @bold{$1 <$sender>}\n\n$2";
315  notice.z_sender=NULL;
316  if (opcode) notice.z_opcode=opcode;
317
318  notice.z_message_len=strlen(zsigtmp)+1+strlen(message);
319  notice.z_message=owl_malloc(notice.z_message_len+10);
320  strcpy(notice.z_message, zsigtmp);
321  memcpy(notice.z_message+strlen(zsigtmp)+1, message, strlen(message));
322
323  /* ret=ZSendNotice(&notice, ZAUTH); */
324  ret=ZSrvSendNotice(&notice, ZAUTH, send_zephyr_helper);
325 
326  /* free then check the return */
327  owl_free(notice.z_message);
328  ZFreeNotice(&notice);
329  if (ret!=ZERR_NONE) {
330    owl_function_makemsg("Error sending zephyr");
331    return(ret);
332  }
333
334  return(0);
335}
336
337Code_t send_zephyr_helper(ZNotice_t *notice, char *buf, int len, int wait) {
338  return(ZSendPacket(buf, len, 0));
339}
340
341void send_ping(char *to) {
342  send_zephyr("PING", "", "MESSAGE", "PERSONAL", to, "");
343}
344
345void owl_zephyr_handle_ack(ZNotice_t *retnotice) {
346  char buff[LINE];
347  char *tmp;
348 
349  /* if it's an HMACK ignore it */
350  if (retnotice->z_kind == HMACK) return;
351 
352  if (retnotice->z_kind == SERVNAK) {
353    owl_function_makemsg("Authorization failure sending zephyr");
354  } else if ((retnotice->z_kind != SERVACK) || !retnotice->z_message_len) {
355    owl_function_makemsg("Detected server failure while receiving acknowledgement");
356  } else if (!strcmp(retnotice->z_message, ZSRVACK_SENT)) {
357    if (!strcasecmp(retnotice->z_opcode, "ping")) {
358      return;
359    } else if (!strcasecmp(retnotice->z_class, "message") &&
360               !strcasecmp(retnotice->z_class_inst, "personal")) {
361      tmp=pretty_sender(retnotice->z_recipient);
362      sprintf(buff, "Message sent to %s.", tmp);
363      free(tmp);
364    } else {
365      sprintf(buff, "Message sent to -c %s -i %s\n", retnotice->z_class, retnotice->z_class_inst);
366    }
367    owl_function_makemsg(buff);
368  } else if (!strcmp(retnotice->z_message, ZSRVACK_NOTSENT)) {
369    if (strcasecmp(retnotice->z_class, "message")) {
370      sprintf(buff, "Not logged in or not subscribing to class %s, instance %s",
371              retnotice->z_class, retnotice->z_class_inst);
372      owl_function_makemsg(buff);
373    } else {
374      owl_function_makemsg("Not logged in or subscribing to messages.");
375    }
376  } else {
377    char buff[1024];
378    sprintf(buff, "Internal error on ack (%s)", retnotice->z_message);
379    owl_function_makemsg(buff);
380  }
381}
382
383int owl_zephyr_notice_is_ack(ZNotice_t *n) {
384  if (n->z_kind == SERVNAK || n->z_kind == SERVACK || n->z_kind == HMACK) {
385    if (!strcasecmp(n->z_class, LOGIN_CLASS)) return(0);
386    return(1);
387  }
388  return(0);
389}
390 
391void owl_zephyr_zaway(owl_message *m) {
392  char *tmpbuff;
393 
394  /* bail if it doesn't look like a message we should reply to */
395  if (strcasecmp(owl_message_get_class(m), "message")) return;
396  if (strcasecmp(owl_message_get_recipient(m), ZGetSender())) return;
397  if (!strcasecmp(owl_message_get_sender(m), "")) return;
398  if (!strcasecmp(owl_message_get_opcode(m), "ping")) return;
399  if (!strcasecmp(owl_message_get_opcode(m), "auto")) return;
400  if (!strcasecmp(owl_message_get_notice(m)->z_message, "Automated reply:")) return;
401  if (!strcasecmp(owl_message_get_sender(m), ZGetSender())) return;
402  /* add one to not send to ourselves */
403
404  send_zephyr("",
405              "Automated reply:",
406              owl_message_get_class(m),
407              owl_message_get_instance(m),
408              owl_message_get_sender(m),
409              owl_global_get_zaway_msg(&g));
410
411  /* display the message as an admin message in the receive window */
412  tmpbuff=owl_malloc(strlen(owl_global_get_zaway_msg(&g))+LINE);
413  sprintf(tmpbuff, "Message sent to %s", owl_message_get_sender(m));
414  owl_function_adminmsg(tmpbuff, owl_global_get_zaway_msg(&g));
415  owl_free(tmpbuff);
416}
417
418
419void owl_zephyr_hackaway_cr(ZNotice_t *n) {
420  /* replace \r's with ' '.  Gross-ish */
421  int i;
422
423  for (i=0; i<n->z_message_len; i++) {
424    if (n->z_message[i]=='\r') {
425      n->z_message[i]=' ';
426    }
427  }
428}
429
430void owl_zephyr_zlocate(char *user, char *out, int auth) {
431  int ret, numlocs;
432  int one = 1;
433  ZLocations_t locations;
434  char *myuser;
435 
436  strcpy(out, "");
437
438  ret=ZLocateUser(user,&numlocs,auth?ZAUTH:ZNOAUTH);
439  if (ret != ZERR_NONE) {
440    owl_function_makemsg("Error locating user");
441  }
442
443  if (numlocs==0) {
444    strcpy(out, "Hidden or not logged-in\n");
445    return;
446  }
447   
448  for (;numlocs;numlocs--) {
449    ZGetLocations(&locations,&one);
450    myuser=pretty_sender(user);
451    sprintf(out, "%s%s: %s\t%s\t%s\n", out, myuser, locations.host, locations.tty, locations.time);
452    owl_free(myuser);
453  }
454}
Note: See TracBrowser for help on using the repository browser.