#include "owl.h" CALLER_OWN owl_zwrite *owl_zwrite_new(const char *line) { owl_zwrite *z = g_new(owl_zwrite, 1); if (owl_zwrite_create_from_line(z, line) < 0) { owl_zwrite_delete(z); return NULL; } return z; } G_GNUC_WARN_UNUSED_RESULT int owl_zwrite_create_from_line(owl_zwrite *z, const char *line) { int argc, badargs, myargc; char **argv; const char *const *myargv; char *msg = NULL; badargs=0; /* start with null entries */ z->cmd=NULL; z->realm=NULL; z->class=NULL; z->inst=NULL; z->opcode=NULL; z->zsig=NULL; z->message=NULL; z->cc=0; z->noping=0; z->recips = g_ptr_array_new(); z->zwriteline = g_strdup(line); /* parse the command line for options */ argv=owl_parseline(line, &argc); myargv=strs(argv); if (argc<0) { owl_function_error("Unbalanced quotes in zwrite"); return(-1); } myargc=argc; if (myargc && *(myargv[0])!='-') { z->cmd=g_strdup(myargv[0]); myargc--; myargv++; } while (myargc) { if (!strcmp(myargv[0], "-c")) { if (myargc<2) { badargs=1; break; } z->class=owl_validate_utf8(myargv[1]); myargv+=2; myargc-=2; } else if (!strcmp(myargv[0], "-i")) { if (myargc<2) { badargs=1; break; } z->inst=owl_validate_utf8(myargv[1]); myargv+=2; myargc-=2; } else if (!strcmp(myargv[0], "-r")) { if (myargc<2) { badargs=1; break; } z->realm=owl_validate_utf8(myargv[1]); myargv+=2; myargc-=2; } else if (!strcmp(myargv[0], "-s")) { if (myargc<2) { badargs=1; break; } z->zsig=owl_validate_utf8(myargv[1]); myargv+=2; myargc-=2; } else if (!strcmp(myargv[0], "-O")) { if (myargc<2) { badargs=1; break; } z->opcode=owl_validate_utf8(myargv[1]); myargv+=2; myargc-=2; } else if (!strcmp(myargv[0], "-m")) { if (myargc<2) { badargs=1; break; } /* we must already have users or a class or an instance */ if (z->recips->len < 1 && (!z->class) && (!z->inst)) { badargs=1; break; } /* Once we have -m, gobble up everything else on the line */ myargv++; myargc--; msg = g_strjoinv(" ", (char**)myargv); break; } else if (!strcmp(myargv[0], "-C")) { z->cc=1; myargv++; myargc--; } else if (!strcmp(myargv[0], "-n")) { z->noping=1; myargv++; myargc--; } else { /* anything unattached is a recipient */ g_ptr_array_add(z->recips, owl_validate_utf8(myargv[0])); myargv++; myargc--; } } g_strfreev(argv); if (badargs) { return(-1); } if (z->class == NULL && z->inst == NULL && z->recips->len == 0) { owl_function_error("You must specify a recipient for zwrite"); return(-1); } /* now deal with defaults */ if (z->class==NULL) z->class=g_strdup("message"); if (z->inst==NULL) z->inst=g_strdup("personal"); if (z->realm==NULL) z->realm=g_strdup(""); if (z->opcode==NULL) z->opcode=g_strdup(""); /* z->message is allowed to stay NULL */ if(msg) { owl_zwrite_set_message(z, msg); g_free(msg); } return(0); } void owl_zwrite_populate_zsig(owl_zwrite *z) { /* get a zsig, if not given */ if (z->zsig != NULL) return; z->zsig = owl_perlconfig_execute(owl_global_get_zsigfunc(&g)); } void owl_zwrite_send_ping(const owl_zwrite *z) { int i; char *to; if (z->noping) return; if (strcasecmp(z->class, "message")) { return; } /* if there are no recipients we won't send a ping, which is what we want */ for (i = 0; i < z->recips->len; i++) { to = owl_zwrite_get_recip_n_with_realm(z, i); send_ping(to, z->class, z->inst); g_free(to); } } /* Set the message with no post-processing*/ void owl_zwrite_set_message_raw(owl_zwrite *z, const char *msg) { g_free(z->message); z->message = owl_validate_utf8(msg); } void owl_zwrite_set_message(owl_zwrite *z, const char *msg) { int i; GString *message; char *tmp = NULL, *tmp2; g_free(z->message); if (z->recips->len > 0 && z->cc) { message = g_string_new("CC: "); for (i = 0; i < z->recips->len; i++) { tmp = owl_zwrite_get_recip_n_with_realm(z, i); g_string_append_printf(message, "%s ", tmp); g_free(tmp); tmp = NULL; } tmp = owl_validate_utf8(msg); tmp2 = owl_text_expand_tabs(tmp); g_string_append_printf(message, "\n%s", tmp2); z->message = g_string_free(message, false); g_free(tmp); g_free(tmp2); } else { tmp=owl_validate_utf8(msg); z->message=owl_text_expand_tabs(tmp); g_free(tmp); } } const char *owl_zwrite_get_message(const owl_zwrite *z) { if (z->message) return(z->message); return(""); } int owl_zwrite_is_message_set(const owl_zwrite *z) { if (z->message) return(1); return(0); } int owl_zwrite_send_message(const owl_zwrite *z) { int i, ret = 0; char *to = NULL; if (z->message==NULL) return(-1); if (z->recips->len > 0) { for (i = 0; i < z->recips->len; i++) { to = owl_zwrite_get_recip_n_with_realm(z, i); ret = send_zephyr(z->opcode, z->zsig, z->class, z->inst, to, z->message); /* Abort on the first error, to match the zwrite binary. */ if (ret != 0) break; g_free(to); to = NULL; } } else { to = g_strdup_printf( "@%s", z->realm); ret = send_zephyr(z->opcode, z->zsig, z->class, z->inst, to, z->message); } g_free(to); return ret; } int owl_zwrite_create_and_send_from_line(const char *cmd, const char *msg) { owl_zwrite z; int rv; rv=owl_zwrite_create_from_line(&z, cmd); if (rv) return(rv); if (!owl_zwrite_is_message_set(&z)) { owl_zwrite_set_message(&z, msg); } owl_zwrite_populate_zsig(&z); owl_zwrite_send_message(&z); owl_zwrite_cleanup(&z); return(0); } const char *owl_zwrite_get_class(const owl_zwrite *z) { return(z->class); } const char *owl_zwrite_get_instance(const owl_zwrite *z) { return(z->inst); } const char *owl_zwrite_get_opcode(const owl_zwrite *z) { return(z->opcode); } void owl_zwrite_set_opcode(owl_zwrite *z, const char *opcode) { g_free(z->opcode); z->opcode=owl_validate_utf8(opcode); } const char *owl_zwrite_get_realm(const owl_zwrite *z) { return(z->realm); } const char *owl_zwrite_get_zsig(const owl_zwrite *z) { if (z->zsig) return(z->zsig); return(""); } void owl_zwrite_set_zsig(owl_zwrite *z, const char *zsig) { g_free(z->zsig); z->zsig = g_strdup(zsig); } int owl_zwrite_get_numrecips(const owl_zwrite *z) { return z->recips->len; } const char *owl_zwrite_get_recip_n(const owl_zwrite *z, int n) { return z->recips->pdata[n]; } /* Caller must free the result. */ CALLER_OWN char *owl_zwrite_get_recip_n_with_realm(const owl_zwrite *z, int n) { if (z->realm[0]) { return g_strdup_printf("%s@%s", owl_zwrite_get_recip_n(z, n), z->realm); } else { return g_strdup(owl_zwrite_get_recip_n(z, n)); } } int owl_zwrite_is_personal(const owl_zwrite *z) { /* return true if at least one of the recipients is personal */ int i; char *recip; for (i = 0; i < z->recips->len; i++) { recip = z->recips->pdata[i]; if (recip[0] != '@') return 1; } return(0); } void owl_zwrite_delete(owl_zwrite *z) { owl_zwrite_cleanup(z); g_free(z); } void owl_zwrite_cleanup(owl_zwrite *z) { owl_ptr_array_free(z->recips, g_free); g_free(z->cmd); g_free(z->zwriteline); g_free(z->class); g_free(z->inst); g_free(z->opcode); g_free(z->realm); g_free(z->message); g_free(z->zsig); } /* * Returns a zwrite line suitable for replying, specifically the * message field is stripped out. Result should be freed with * g_free. * * If not a CC, only the recip_index'th user will be replied to. */ CALLER_OWN char *owl_zwrite_get_replyline(const owl_zwrite *z, int recip_index) { /* Match ordering in zwrite help. */ GString *buf = g_string_new(""); int i; /* Disturbingly, it is apparently possible to z->cmd to be null if * owl_zwrite_create_from_line got something starting with -. And we * can't kill it because this is exported to perl. */ owl_string_append_quoted_arg(buf, z->cmd ? z->cmd : "zwrite"); if (z->noping) { g_string_append(buf, " -n"); } if (z->cc) { g_string_append(buf, " -C"); } if (strcmp(z->class, "message")) { g_string_append(buf, " -c "); owl_string_append_quoted_arg(buf, z->class); } if (strcmp(z->inst, "personal")) { g_string_append(buf, " -i "); owl_string_append_quoted_arg(buf, z->inst); } if (z->realm && z->realm[0] != '\0') { g_string_append(buf, " -r "); owl_string_append_quoted_arg(buf, z->realm); } if (z->opcode && z->opcode[0] != '\0') { g_string_append(buf, " -O "); owl_string_append_quoted_arg(buf, z->opcode); } if (z->cc) { for (i = 0; i < z->recips->len; i++) { g_string_append_c(buf, ' '); owl_string_append_quoted_arg(buf, z->recips->pdata[i]); } } else if (recip_index < z->recips->len) { g_string_append_c(buf, ' '); owl_string_append_quoted_arg(buf, z->recips->pdata[recip_index]); } return g_string_free(buf, false); }