source: util.c @ 65b2173

release-1.4release-1.5release-1.6release-1.7release-1.8release-1.9
Last change on this file since 65b2173 was 65b2173, checked in by Anders Kaseorg <andersk@mit.edu>, 9 years ago
Split up declarations for adding const qualifiers for char * and void *. Signed-off-by: Anders Kaseorg <andersk@mit.edu>
  • Property mode set to 100644
File size: 18.8 KB
Line 
1#include "owl.h"
2#include <stdlib.h>
3#include <string.h>
4#include <unistd.h>
5#include <ctype.h>
6#include <pwd.h>
7
8void sepbar(char *in)
9{
10  char buff[1024];
11  WINDOW *sepwin;
12  owl_messagelist *ml;
13  owl_view *v;
14  int x, y, i;
15  char *foo, *appendtosepbar;
16
17  sepwin=owl_global_get_curs_sepwin(&g);
18  ml=owl_global_get_msglist(&g);
19  v=owl_global_get_current_view(&g);
20
21  werase(sepwin);
22  wattron(sepwin, A_REVERSE);
23  if (owl_global_is_fancylines(&g)) {
24    whline(sepwin, ACS_HLINE, owl_global_get_cols(&g));
25  } else {
26    whline(sepwin, '-', owl_global_get_cols(&g));
27  }
28
29  if (owl_global_is_sepbar_disable(&g)) {
30    getyx(sepwin, y, x);
31    wmove(sepwin, y, owl_global_get_cols(&g)-1);
32    return;
33  }
34
35  wmove(sepwin, 0, 2); 
36
37  if (owl_messagelist_get_size(ml)==0) {
38    strcpy(buff, " (-/-) ");
39  } else {
40    snprintf(buff, 1024, " (%i/%i/%i) ", owl_global_get_curmsg(&g)+1,
41            owl_view_get_size(v),
42            owl_messagelist_get_size(ml));
43  }
44  waddstr(sepwin, buff);
45
46  foo=owl_view_get_filtname(v);
47  if (strcmp(foo, owl_global_get_view_home(&g))) wattroff(sepwin, A_REVERSE);
48  waddstr(sepwin, " ");
49  waddstr(sepwin, owl_view_get_filtname(v));
50  waddstr(sepwin, " ");
51  if (strcmp(foo, owl_global_get_view_home(&g))) wattron(sepwin, A_REVERSE);
52
53  if (owl_mainwin_is_curmsg_truncated(owl_global_get_mainwin(&g))) {
54    getyx(sepwin, y, x);
55    wmove(sepwin, y, x+2);
56    wattron(sepwin, A_BOLD);
57    waddstr(sepwin, " <truncated> ");
58    wattroff(sepwin, A_BOLD);
59  }
60
61  i=owl_mainwin_get_last_msg(owl_global_get_mainwin(&g));
62  if ((i != -1) &&
63      (i < owl_view_get_size(v)-1)) {
64    getyx(sepwin, y, x);
65    wmove(sepwin, y, x+2);
66    wattron(sepwin, A_BOLD);
67    waddstr(sepwin, " <more> ");
68    wattroff(sepwin, A_BOLD);
69  }
70
71  if (owl_global_get_rightshift(&g)>0) {
72    getyx(sepwin, y, x);
73    wmove(sepwin, y, x+2);
74    snprintf(buff, 1024, " right: %i ", owl_global_get_rightshift(&g));
75    waddstr(sepwin, buff);
76  }
77
78  if (owl_global_is_zaway(&g) || owl_global_is_aaway(&g)) {
79    getyx(sepwin, y, x);
80    wmove(sepwin, y, x+2);
81    wattron(sepwin, A_BOLD);
82    wattroff(sepwin, A_REVERSE);
83    if (owl_global_is_zaway(&g) && owl_global_is_aaway(&g)) {
84      waddstr(sepwin, " AWAY ");
85    } else if (owl_global_is_zaway(&g)) {
86      waddstr(sepwin, " Z-AWAY ");
87    } else if (owl_global_is_aaway(&g)) {
88      waddstr(sepwin, " A-AWAY ");
89    }
90    wattron(sepwin, A_REVERSE);
91    wattroff(sepwin, A_BOLD);
92  }
93
94  if (owl_global_get_curmsg_vert_offset(&g)) {
95    getyx(sepwin, y, x);
96    wmove(sepwin, y, x+2);
97    wattron(sepwin, A_BOLD);
98    wattroff(sepwin, A_REVERSE);
99    waddstr(sepwin, " SCROLL ");
100    wattron(sepwin, A_REVERSE);
101    wattroff(sepwin, A_BOLD);
102  }
103 
104  if (in) {
105    getyx(sepwin, y, x);
106    wmove(sepwin, y, x+2);
107    waddstr(sepwin, in);
108  }
109
110  appendtosepbar = owl_global_get_appendtosepbar(&g);
111  if (appendtosepbar && *appendtosepbar) {
112    getyx(sepwin, y, x);
113    wmove(sepwin, y, x+2);
114    waddstr(sepwin, " ");
115    waddstr(sepwin, owl_global_get_appendtosepbar(&g));
116    waddstr(sepwin, " ");
117  }
118
119  getyx(sepwin, y, x);
120  wmove(sepwin, y, owl_global_get_cols(&g)-1);
121   
122  wattroff(sepwin, A_BOLD);
123  wattroff(sepwin, A_REVERSE);
124  wnoutrefresh(sepwin);
125}
126
127char **atokenize(char *buffer, char *sep, int *i)
128{
129  /* each element of return must be freed by user */
130  char **args;
131  char *workbuff, *foo;
132  int done=0, first=1, count=0;
133
134  workbuff=owl_malloc(strlen(buffer)+1);
135  memcpy(workbuff, buffer, strlen(buffer)+1);
136
137  args=NULL;
138  while (!done) {
139    if (first) {
140      first=0;
141      foo=strtok(workbuff, sep);
142    } else {
143      foo=strtok(NULL, sep);
144    }
145    if (foo==NULL) {
146      done=1;
147    } else {
148      args=owl_realloc(args, sizeof(char *) * (count+1));
149      args[count]=owl_malloc(strlen(foo)+1);
150      strcpy(args[count], foo);
151      count++;
152    }
153  }
154  *i=count;
155  owl_free(workbuff);
156  return(args);
157}
158
159char *skiptokens(char *buff, int n) {
160  /* skips n tokens and returns where that would be.
161   * TODO: handle quotes more sanely. */
162 
163  int inquotes=0;
164  while (*buff && n>0) {
165      while (*buff == ' ') buff++;
166      while (*buff && (inquotes || *buff != ' ')) { 
167        if (*buff == '"' || *buff == '\'') inquotes=!inquotes;
168        buff++;
169      }
170      while (*buff == ' ') buff++;
171      n--;
172  }
173  return buff;
174}
175
176/* Return a "nice" version of the path.  Tilde expansion is done, and
177 * duplicate slashes are removed.  Caller must free the return.
178 */
179char *owl_util_makepath(char *in)
180{
181  int i, j, x;
182  char *out, user[MAXPATHLEN];
183  struct passwd *pw;
184
185  out=owl_malloc(MAXPATHLEN+1);
186  out[0]='\0';
187  j=strlen(in);
188  x=0;
189  for (i=0; i<j; i++) {
190    if (in[i]=='~') {
191      if ( (i==(j-1)) ||          /* last character */
192           (in[i+1]=='/') ) {     /* ~/ */
193        /* use my homedir */
194        pw=getpwuid(getuid());
195        if (!pw) {
196          out[x]=in[i];
197        } else {
198          out[x]='\0';
199          strcat(out, pw->pw_dir);
200          x+=strlen(pw->pw_dir);
201        }
202      } else {
203        /* another user homedir */
204        int a, b;
205        b=0;
206        for (a=i+1; i<j; a++) {
207          if (in[a]==' ' || in[a]=='/') {
208            break;
209          } else {
210            user[b]=in[a];
211            i++;
212            b++;
213          }
214        }
215        user[b]='\0';
216        pw=getpwnam(user);
217        if (!pw) {
218          out[x]=in[i];
219        } else {
220          out[x]='\0';
221          strcat(out, pw->pw_dir);
222          x+=strlen(pw->pw_dir);
223        }
224      }
225    } else if (in[i]=='/') {
226      /* check for a double / */
227      if (i<(j-1) && (in[i+1]=='/')) {
228        /* do nothing */
229      } else {
230        out[x]=in[i];
231        x++;
232      }
233    } else {
234      out[x]=in[i];
235      x++;
236    }
237  }
238  out[x]='\0';
239  return(out);
240}
241
242void atokenize_free(char **tok, int nels)
243{
244  int i;
245  for (i=0; i<nels; i++) {
246    owl_free(tok[i]);
247  }
248  owl_free(tok);
249}
250
251
252void owl_parsefree(char **argv, int argc)
253{
254  int i;
255
256  if (!argv) return;
257 
258  for (i=0; i<argc; i++) {
259    if (argv[i]) owl_free(argv[i]);
260  }
261  owl_free(argv);
262}
263
264char **owl_parseline(char *line, int *argc)
265{
266  /* break a command line up into argv, argc.  The caller must free
267     the returned values.  If there is an error argc will be set to
268     -1, argv will be NULL and the caller does not need to free
269     anything */
270
271  char **argv;
272  int i, len, between=1;
273  char *curarg;
274  char quote;
275
276  argv=owl_malloc(sizeof(char *));
277  len=strlen(line);
278  curarg=owl_malloc(len+10);
279  strcpy(curarg, "");
280  quote='\0';
281  *argc=0;
282  for (i=0; i<len+1; i++) {
283    /* find the first real character */
284    if (between) {
285      if (line[i]==' ' || line[i]=='\t' || line[i]=='\0') {
286        continue;
287      } else {
288        between=0;
289        i--;
290        continue;
291      }
292    }
293
294    /* deal with a quote character */
295    if (line[i]=='"' || line[i]=="'"[0]) {
296      /* if this type of quote is open, close it */
297      if (quote==line[i]) {
298        quote='\0';
299        continue;
300      }
301
302      /* if no quoting is open then open with this */
303      if (quote=='\0') {
304        quote=line[i];
305        continue;
306      }
307
308      /* if another type of quote is open then treat this as a literal */
309      curarg[strlen(curarg)+1]='\0';
310      curarg[strlen(curarg)]=line[i];
311      continue;
312    }
313
314    /* if it's not a space or end of command, then use it */
315    if (line[i]!=' ' && line[i]!='\t' && line[i]!='\n' && line[i]!='\0') {
316      curarg[strlen(curarg)+1]='\0';
317      curarg[strlen(curarg)]=line[i];
318      continue;
319    }
320
321    /* otherwise, if we're not in quotes, add the whole argument */
322    if (quote=='\0') {
323      /* add the argument */
324      argv=owl_realloc(argv, sizeof(char *)*((*argc)+1));
325      argv[*argc]=owl_malloc(strlen(curarg)+2);
326      strcpy(argv[*argc], curarg);
327      *argc=*argc+1;
328      strcpy(curarg, "");
329      between=1;
330      continue;
331    }
332
333    /* if it is a space and we're in quotes, then use it */
334    curarg[strlen(curarg)+1]='\0';
335    curarg[strlen(curarg)]=line[i];
336  }
337
338  owl_free(curarg);
339
340  /* check for unbalanced quotes */
341  if (quote!='\0') {
342    owl_parsefree(argv, *argc);
343    *argc=-1;
344    return(NULL);
345  }
346
347  return(argv);
348}
349
350/* caller must free the return */
351char *owl_util_minutes_to_timestr(int in)
352{
353  int days, hours;
354  long run;
355  char *out;
356
357  run=in;
358
359  days=run/1440;
360  run-=days*1440;
361  hours=run/60;
362  run-=hours*60;
363
364  if (days>0) {
365    out=owl_sprintf("%i d %2.2i:%2.2li", days, hours, run);
366  } else {
367    out=owl_sprintf("    %2.2i:%2.2li", hours, run);
368  }
369  return(out);
370}
371
372/* hooks for doing memory allocation et. al. in owl */
373
374void *owl_malloc(size_t size)
375{
376  return(g_malloc(size));
377}
378
379void owl_free(void *ptr)
380{
381  g_free(ptr);
382}
383
384char *owl_strdup(const char *s1)
385{
386  return(g_strdup(s1));
387}
388
389void *owl_realloc(void *ptr, size_t size)
390{
391  return(g_realloc(ptr, size));
392}
393
394/* allocates memory and returns the string or null.
395 * caller must free the string.
396 */
397char *owl_sprintf(const char *fmt, ...)
398{
399  va_list ap;
400  char *ret = NULL;
401  va_start(ap, fmt);
402  ret = g_strdup_vprintf(fmt, ap);
403  va_end(ap);
404  return ret;
405}
406
407
408/* Return the owl color associated with the named color.  Return -1
409 * if the named color is not available
410 */
411int owl_util_string_to_color(char *color)
412{
413  int c;
414  if (!strcasecmp(color, "black")) {
415    return(OWL_COLOR_BLACK);
416  } else if (!strcasecmp(color, "red")) {
417    return(OWL_COLOR_RED);
418  } else if (!strcasecmp(color, "green")) {
419    return(OWL_COLOR_GREEN);
420  } else if (!strcasecmp(color, "yellow")) {
421    return(OWL_COLOR_YELLOW);
422  } else if (!strcasecmp(color, "blue")) {
423    return(OWL_COLOR_BLUE);
424  } else if (!strcasecmp(color, "magenta")) {
425    return(OWL_COLOR_MAGENTA);
426  } else if (!strcasecmp(color, "cyan")) {
427    return(OWL_COLOR_CYAN);
428  } else if (!strcasecmp(color, "white")) {
429    return(OWL_COLOR_WHITE);
430  } else if (!strcasecmp(color, "default")) {
431    return(OWL_COLOR_DEFAULT);
432  }
433  c = atoi(color);
434  if (c >= -1 && c < COLORS) {
435    return(c);
436  }
437  return(OWL_COLOR_INVALID);
438}
439
440/* Return a string name of the given owl color */
441char *owl_util_color_to_string(int color)
442{
443  if (color==OWL_COLOR_BLACK)   return("black");
444  if (color==OWL_COLOR_RED)     return("red");
445  if (color==OWL_COLOR_GREEN)   return("green");
446  if (color==OWL_COLOR_YELLOW)  return("yellow");
447  if (color==OWL_COLOR_BLUE)    return("blue");
448  if (color==OWL_COLOR_MAGENTA) return("magenta");
449  if (color==OWL_COLOR_CYAN)    return("cyan");
450  if (color==OWL_COLOR_WHITE)   return("white");
451  if (color==OWL_COLOR_DEFAULT) return("default");
452  return("Unknown color");
453}
454
455/* Get the default tty name.  Caller must free the return */
456char *owl_util_get_default_tty()
457{
458  char *tmp;
459  char *out;
460
461  if (getenv("DISPLAY")) {
462    out=owl_strdup(getenv("DISPLAY"));
463  } else if ((tmp=ttyname(fileno(stdout)))!=NULL) {
464    out=owl_strdup(tmp);
465    if (!strncmp(out, "/dev/", 5)) {
466      owl_free(out);
467      out=owl_strdup(tmp+5);
468    }
469  } else {
470    out=owl_strdup("unknown");
471  }
472  return(out);
473}
474
475
476/* Animation hack */
477void owl_hack_animate()
478{
479  owl_messagelist *ml;
480  owl_message *m;
481  owl_fmtext *fm;
482  char *text, *ptr;
483  int place;
484
485  /* grab the first message and make sure its id is 0 */
486  ml=owl_global_get_msglist(&g);
487  m=owl_messagelist_get_element(ml, 0);
488  if (!m) return;
489  if (owl_message_get_id(m)!=0) return;
490
491  fm=owl_message_get_fmtext(m);
492  text=owl_fmtext_get_text(fm);
493
494  ptr=strstr(text, "OvO");
495  if (ptr) {
496    place=ptr-text;
497    owl_fmtext_set_char(fm, place, '-');
498    owl_fmtext_set_char(fm, place+2, '-');
499
500    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
501    if (owl_popwin_is_active(owl_global_get_popwin(&g))) {
502      owl_popwin_refresh(owl_global_get_popwin(&g));
503      /* TODO: this is a broken kludge */
504      if (owl_global_get_viewwin(&g)) {
505        owl_viewwin_redisplay(owl_global_get_viewwin(&g), 0);
506      }
507    }
508    owl_global_set_needrefresh(&g);
509    return;
510  }
511
512  ptr=strstr(text, "-v-");
513  if (ptr) {
514    place=ptr-text;
515    owl_fmtext_set_char(fm, place, 'O');
516    owl_fmtext_set_char(fm, place+2, 'O');
517
518    owl_mainwin_redisplay(owl_global_get_mainwin(&g));
519    if (owl_popwin_is_active(owl_global_get_popwin(&g))) {
520      owl_popwin_refresh(owl_global_get_popwin(&g));
521      /* TODO: this is a broken kludge */
522      if (owl_global_get_viewwin(&g)) {
523        owl_viewwin_redisplay(owl_global_get_viewwin(&g), 0);
524      }
525    }
526    owl_global_set_needrefresh(&g);
527    return;
528  }
529}
530
531/* strip leading and trailing new lines.  Caller must free the
532 * return.
533 */
534char *owl_util_stripnewlines(char *in)
535{
536 
537  char  *tmp, *ptr1, *ptr2, *out;
538
539  ptr1=tmp=owl_strdup(in);
540  while (ptr1[0]=='\n') {
541    ptr1++;
542  }
543  ptr2=ptr1+strlen(ptr1)-1;
544  while (ptr2>ptr1 && ptr2[0]=='\n') {
545    ptr2[0]='\0';
546    ptr2--;
547  }
548
549  out=owl_strdup(ptr1);
550  owl_free(tmp);
551  return(out);
552}
553
554/* Delete the line matching "line" from the named file.  If no such
555 * line is found the file is left intact.  If backup==1 then create a
556 * backupfile containing the original contents.  This is an
557 * inefficient impelementation which reads the entire file into
558 * memory.
559 */
560void owl_util_file_deleteline(char *filename, char *line, int backup)
561{
562  char buff[LINE], *text;
563  char *backupfilename="";
564  FILE *file, *backupfile=NULL;
565  int size, newline;
566
567  /* open the file for reading */
568  file=fopen(filename, "r");
569  if (!file) {
570    owl_function_error("Error opening file %s", filename);
571    return;
572  }
573
574  /* open the backup file for writing */
575  if (backup) {
576    backupfilename=owl_sprintf("%s.backup", filename);
577    backupfile=fopen(backupfilename, "w");
578    if (!backupfile) {
579      owl_function_error("Error opening file %s for writing", backupfilename);
580      owl_free(backupfilename);
581      fclose(file);
582      return;
583    }
584  }
585
586  /* we'll read the entire file into memory, minus the line we don't want and
587   * and at the same time create the backup file if necessary
588   */
589  text=owl_malloc(LINE);
590  strcpy(text, "");
591  size=LINE;
592  while (fgets(buff, LINE, file)!=NULL) {
593    /* strip the newline */
594    newline=0;
595    if (buff[strlen(buff)-1]=='\n') {
596      buff[strlen(buff)-1]='\0';
597      newline=1;
598    }
599   
600    /* if we don't match the line, add to saved text in memory */
601    if (strcasecmp(buff, line)) {
602      size+=LINE;
603      text=owl_realloc(text, size);
604      strcat(text, buff);
605      if (newline) strcat(text, "\n");
606    }
607
608    /* write to backupfile if necessary */
609    if (backup) {
610      fputs(buff, backupfile);
611      if (newline) fputs("\n", backupfile);
612    }
613  }
614  if (backup) fclose(backupfile);
615  fclose(file);
616
617  /* now rewrite the original file from memory */
618  file=fopen(filename, "w");
619  if (!file) {
620    owl_function_error("WARNING: Error opening %s for writing.  Use %s to restore.", filename, backupfilename);
621    owl_function_beep();
622  } else {
623    fputs(text, file);
624    fclose(file);
625  }
626
627  if (backup)
628    owl_free(backupfilename);
629  owl_free(text);
630}
631
632int owl_util_max(int a, int b)
633{
634  if (a>b) return(a);
635  return(b);
636}
637
638int owl_util_min(int a, int b)
639{
640  if (a<b) return(a);
641  return(b);
642}
643
644/* Return the base class or instance from a zephyr class, by removing
645   leading `un' or trailing `.d'.
646   The caller is responsible for freeing the allocated string.
647*/
648char * owl_util_baseclass(char * class)
649{
650  char *start, *end;
651
652  start = class;
653  while(!strncmp(start, "un", 2)) {
654    start += 2;
655  }
656
657  start = owl_strdup(start);
658  end = start + strlen(start) - 1;
659  while(end > start && *end == 'd' && *(end-1) == '.') {
660    end -= 2;
661  }
662  *(end + 1) = 0;
663
664  return start;
665}
666
667char * owl_get_datadir()
668{
669  char * datadir = getenv("BARNOWL_DATA_DIR");
670  if(datadir != NULL)
671    return datadir;
672  return DATADIR;
673}
674
675/* Strips format characters from a valid utf-8 string. Returns the
676   empty string if 'in' does not validate. */
677char * owl_strip_format_chars(const char *in)
678{
679  char *r;
680  if (g_utf8_validate(in, -1, NULL)) {
681    const char *s, *p;
682    r = owl_malloc(strlen(in)+1);
683    r[0] = '\0';
684    s = in;
685    p = strchr(s, OWL_FMTEXT_UC_STARTBYTE_UTF8);
686    while(p) {
687      /* If it's a format character, copy up to it, and skip all
688         immediately following format characters. */
689      if (owl_fmtext_is_format_char(g_utf8_get_char(p))) {
690        strncat(r, s, p-s);
691        p = g_utf8_next_char(p);
692        while (p && owl_fmtext_is_format_char(g_utf8_get_char(p))) {
693          p = g_utf8_next_char(p);
694        }
695        s = p;
696        p = strchr(s, OWL_FMTEXT_UC_STARTBYTE_UTF8);
697      }
698      else {
699        p = strchr(p+1, OWL_FMTEXT_UC_STARTBYTE_UTF8);
700      }
701    }
702    if (s) strcat(r,s);
703  }
704  else {
705    r = owl_strdup("");
706  }
707  return r;
708}
709
710/* If in is not UTF-8, convert from ISO-8859-1. We may want to allow
711 * the caller to specify an alternative in the future. We also strip
712 * out characters in Unicode Plane 16, as we use that plane internally
713 * for formatting.
714 */
715char * owl_validate_or_convert(const char *in)
716{
717  if (g_utf8_validate(in, -1, NULL)) {
718    return owl_strip_format_chars(in);
719  }
720  else {
721    return g_convert(in, -1,
722                     "UTF-8", "ISO-8859-1",
723                     NULL, NULL, NULL);
724  }
725}
726/*
727 * Validate 'in' as UTF-8, and either return a copy of it, or an empty
728 * string if it is invalid utf-8.
729 */
730char * owl_validate_utf8(const char *in)
731{
732  char *out;
733  if (g_utf8_validate(in, -1, NULL)) {
734    out = owl_strdup(in);
735  } else {
736    out = owl_strdup("");
737  }
738  return out;
739}
740
741/* This is based on _extract() and _isCJ() from perl's Text::WrapI18N */
742int owl_util_can_break_after(gunichar c)
743{
744 
745  if (c == ' ') return 1;
746  if (c >= 0x3000 && c <= 0x312f) {
747    /* CJK punctuations, Hiragana, Katakana, Bopomofo */
748    if (c == 0x300a || c == 0x300c || c == 0x300e ||
749        c == 0x3010 || c == 0x3014 || c == 0x3016 ||
750        c == 0x3018 || c == 0x301a)
751      return 0;
752    return 1;
753  }
754  if (c >= 0x31a0 && c <= 0x31bf) {return 1;}  /* Bopomofo */
755  if (c >= 0x31f0 && c <= 0x31ff) {return 1;}  /* Katakana extension */
756  if (c >= 0x3400 && c <= 0x9fff) {return 1;}  /* Han Ideogram */
757  if (c >= 0xf900 && c <= 0xfaff) {return 1;}  /* Han Ideogram */
758  if (c >= 0x20000 && c <= 0x2ffff) {return 1;}  /* Han Ideogram */
759  return 0;
760}
761
762/**************************************************************************/
763/************************* REGRESSION TESTS *******************************/
764/**************************************************************************/
765
766#ifdef OWL_INCLUDE_REG_TESTS
767
768#include "test.h"
769
770int owl_util_regtest(void)
771{
772  int numfailed=0;
773
774  printf("# BEGIN testing owl_util\n");
775
776  FAIL_UNLESS("owl_util_substitute 1",
777              !strcmp("foo", owl_text_substitute("foo", "", "Y")));
778  FAIL_UNLESS("owl_text_substitute 2",
779              !strcmp("fYZYZ", owl_text_substitute("foo", "o", "YZ")));
780  FAIL_UNLESS("owl_text_substitute 3",
781              !strcmp("foo", owl_text_substitute("fYZYZ", "YZ", "o")));
782  FAIL_UNLESS("owl_text_substitute 4",
783              !strcmp("/u/foo/meep", owl_text_substitute("~/meep", "~", "/u/foo")));
784
785  FAIL_UNLESS("skiptokens 1", 
786              !strcmp("bar quux", skiptokens("foo bar quux", 1)));
787  FAIL_UNLESS("skiptokens 2", 
788              !strcmp("meep", skiptokens("foo 'bar quux' meep", 2)));
789
790  FAIL_UNLESS("expand_tabs 1",
791              !strcmp("        hi", owl_text_expand_tabs("\thi")));
792
793  FAIL_UNLESS("expand_tabs 2",
794              !strcmp("        hi\nword    tab", owl_text_expand_tabs("\thi\nword\ttab")));
795
796  FAIL_UNLESS("expand_tabs 3",
797              !strcmp("                2 tabs", owl_text_expand_tabs("\t\t2 tabs")));
798
799  /* if (numfailed) printf("*** WARNING: failures encountered with owl_util\n"); */
800  printf("# END testing owl_util (%d failures)\n", numfailed);
801  return(numfailed);
802}
803
804#endif /* OWL_INCLUDE_REG_TESTS */
Note: See TracBrowser for help on using the repository browser.