Thanks to visit codestin.com
Credit goes to code.neomutt.org

NeoMutt  2025-12-11-189-gceedb6
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
string.c
Go to the documentation of this file.
1
28
34
35#include "config.h"
36#include <errno.h>
37#include <stdarg.h> // IWYU pragma: keep
38#include <stdbool.h>
39#include <stdio.h>
40#include <stdlib.h>
41#include <string.h>
42#include <strings.h>
43#include "array.h"
44#include "ctype2.h"
45#include "exit.h"
46#include "logging2.h"
47#include "memory.h"
48#include "string2.h"
49#ifdef HAVE_SYSEXITS_H
50#include <sysexits.h>
51#endif
52
53#ifndef HAVE_STRCASESTR
60static char *strcasestr(const char *haystack, const char *needle)
61{
62 size_t haystackn = strlen(haystack);
63 size_t needlen = strlen(needle);
64
65 const char *p = haystack;
66 while (haystackn >= needlen)
67 {
68 if (strncasecmp(p, needle, needlen) == 0)
69 return (char *) p;
70 p++;
71 haystackn--;
72 }
73 return NULL;
74}
75#endif /* HAVE_STRCASESTR */
76
77#ifndef HAVE_STRSEP
86static char *strsep(char **stringp, const char *delim)
87{
88 if (!*stringp)
89 return NULL;
90
91 char *start = *stringp;
92 for (char *p = *stringp; *p != '\0'; p++)
93 {
94 for (const char *s = delim; *s != '\0'; s++)
95 {
96 if (*p == *s)
97 {
98 *p = '\0';
99 *stringp = p + 1;
100 return start;
101 }
102 }
103 }
104 *stringp = NULL;
105 return start;
106}
107#endif /* HAVE_STRSEP */
108
113{
115 const char *err_str;
116};
117
119static const struct SysExits SysExits[] = {
120#ifdef EX_USAGE
121 { 0xff & EX_USAGE, "Bad usage." },
122#endif
123#ifdef EX_DATAERR
124 { 0xff & EX_DATAERR, "Data format error." },
125#endif
126#ifdef EX_NOINPUT
127 { 0xff & EX_NOINPUT, "Can't open input." },
128#endif
129#ifdef EX_NOUSER
130 { 0xff & EX_NOUSER, "User unknown." },
131#endif
132#ifdef EX_NOHOST
133 { 0xff & EX_NOHOST, "Host unknown." },
134#endif
135#ifdef EX_UNAVAILABLE
136 { 0xff & EX_UNAVAILABLE, "Service unavailable." },
137#endif
138#ifdef EX_SOFTWARE
139 { 0xff & EX_SOFTWARE, "Internal error." },
140#endif
141#ifdef EX_OSERR
142 { 0xff & EX_OSERR, "Operating system error." },
143#endif
144#ifdef EX_OSFILE
145 { 0xff & EX_OSFILE, "System file missing." },
146#endif
147#ifdef EX_CANTCREAT
148 { 0xff & EX_CANTCREAT, "Can't create output." },
149#endif
150#ifdef EX_IOERR
151 { 0xff & EX_IOERR, "I/O error." },
152#endif
153#ifdef EX_TEMPFAIL
154 { 0xff & EX_TEMPFAIL, "Deferred." },
155#endif
156#ifdef EX_PROTOCOL
157 { 0xff & EX_PROTOCOL, "Remote protocol error." },
158#endif
159#ifdef EX_NOPERM
160 { 0xff & EX_NOPERM, "Insufficient permission." },
161#endif
162#ifdef EX_CONFIG
163 { 0xff & EX_NOPERM, "Local configuration error." },
164#endif
165 { S_ERR, "Exec error." },
166};
167
173const char *mutt_str_sysexit(int err_num)
174{
175 for (size_t i = 0; i < countof(SysExits); i++)
176 {
177 if (err_num == SysExits[i].err_num)
178 return SysExits[i].err_str;
179 }
180
181 return NULL;
182}
183
190char *mutt_str_sep(char **stringp, const char *delim)
191{
192 if (!stringp || !*stringp || !delim)
193 return NULL;
194 return strsep(stringp, delim);
195}
196
205static size_t startswith(const char *str, const char *prefix, bool match_case)
206{
207 if (!str || (str[0] == '\0') || !prefix || (prefix[0] == '\0'))
208 {
209 return 0;
210 }
211
212 const char *saved_prefix = prefix;
213 for (; *str && *prefix; str++, prefix++)
214 {
215 if (*str == *prefix)
216 continue;
217
218 if (!match_case && mutt_tolower(*str) == mutt_tolower(*prefix))
219 continue;
220
221 return 0;
222 }
223
224 return (*prefix == '\0') ? (prefix - saved_prefix) : 0;
225}
226
234size_t mutt_str_startswith(const char *str, const char *prefix)
235{
236 return startswith(str, prefix, true);
237}
238
246size_t mutt_istr_startswith(const char *str, const char *prefix)
247{
248 return startswith(str, prefix, false);
249}
250
257char *mutt_str_dup(const char *str)
258{
259 if (!str || (*str == '\0'))
260 return NULL;
261
262 char *p = strdup(str);
263 if (!p)
264 {
265 mutt_error("%s", strerror(errno)); // LCOV_EXCL_LINE
266 mutt_exit(1); // LCOV_EXCL_LINE
267 }
268 return p;
269}
270
284char *mutt_str_replace(char **p, const char *s)
285{
286 if (!p)
287 return NULL;
288 const char *tmp = *p;
289 *p = mutt_str_dup(s);
290 FREE(&tmp);
291 return *p;
292}
293
303void mutt_str_adjust(char **ptr)
304{
305 if (!ptr || !*ptr)
306 return;
307 MUTT_MEM_REALLOC(ptr, strlen(*ptr) + 1, char);
308}
309
317char *mutt_str_lower(char *str)
318{
319 if (!str)
320 return NULL;
321
322 char *p = str;
323
324 while (*p)
325 {
326 *p = mutt_tolower(*p);
327 p++;
328 }
329
330 return str;
331}
332
340char *mutt_str_upper(char *str)
341{
342 if (!str)
343 return NULL;
344
345 char *p = str;
346
347 while (*p)
348 {
349 *p = mutt_toupper(*p);
350 p++;
351 }
352
353 return str;
354}
355
364char *mutt_strn_copy(char *dest, const char *src, size_t len, size_t dsize)
365{
366 if (!src || !dest || (len == 0) || (dsize == 0))
367 return dest;
368
369 if (len > (dsize - 1))
370 len = dsize - 1;
371 memcpy(dest, src, len);
372 dest[len] = '\0';
373 return dest;
374}
375
384char *mutt_strn_dup(const char *begin, size_t len)
385{
386 if (!begin)
387 return NULL;
388
389 char *p = MUTT_MEM_MALLOC(len + 1, char);
390 memcpy(p, begin, len);
391 p[len] = '\0';
392 return p;
393}
394
403int mutt_str_cmp(const char *a, const char *b)
404{
405 return strcmp(NONULL(a), NONULL(b));
406}
407
416int mutt_istr_cmp(const char *a, const char *b)
417{
418 return strcasecmp(NONULL(a), NONULL(b));
419}
420
429bool mutt_strn_equal(const char *a, const char *b, size_t num)
430{
431 return strncmp(NONULL(a), NONULL(b), num) == 0;
432}
433
443int mutt_istrn_cmp(const char *a, const char *b, size_t num)
444{
445 return strncasecmp(NONULL(a), NONULL(b), num);
446}
447
457bool mutt_istrn_equal(const char *a, const char *b, size_t num)
458{
459 return strncasecmp(NONULL(a), NONULL(b), num) == 0;
460}
461
473const char *mutt_istrn_rfind(const char *haystack, size_t haystack_length, const char *needle)
474{
475 if (!haystack || (haystack_length == 0) || !needle)
476 return NULL;
477
478 int needle_length = strlen(needle);
479 const char *haystack_end = haystack + haystack_length - needle_length;
480
481 for (const char *p = haystack_end; p >= haystack; p--)
482 {
483 for (size_t i = 0; i < needle_length; i++)
484 {
485 if ((mutt_tolower(p[i]) != mutt_tolower(needle[i])))
486 goto next;
487 }
488 return p;
489
490 next:;
491 }
492 return NULL;
493}
494
500size_t mutt_str_len(const char *a)
501{
502 return a ? strlen(a) : 0;
503}
504
513int mutt_str_coll(const char *a, const char *b)
514{
515 return strcoll(NONULL(a), NONULL(b));
516}
517
525const char *mutt_istr_find(const char *haystack, const char *needle)
526{
527 if (!haystack)
528 return NULL;
529 if (!needle)
530 return haystack;
531
532 const char *p = NULL, *q = NULL;
533
534 while (*(p = haystack))
535 {
536 for (q = needle; *p && *q && (mutt_tolower(*p) == mutt_tolower(*q)); p++, q++)
537 {
538 }
539 if ((*q == '\0'))
540 return haystack;
541 haystack++;
542 }
543 return NULL;
544}
545
553char *mutt_str_skip_whitespace(const char *p)
554{
555 if (!p)
556 return NULL;
557 SKIPWS(p);
558 return (char *) p;
559}
560
568{
569 if (!s)
570 return;
571
572 for (char *p = s + mutt_str_len(s) - 1; (p >= s) && mutt_isspace(*p); p--)
573 *p = '\0';
574}
575
583size_t mutt_str_copy(char *dest, const char *src, size_t dsize)
584{
585 if (!dest || (dsize == 0))
586 return 0;
587 if (!src)
588 {
589 dest[0] = '\0';
590 return 0;
591 }
592
593 char *dest0 = dest;
594 while ((--dsize > 0) && (*src != '\0'))
595 *dest++ = *src++;
596
597 *dest = '\0';
598 return dest - dest0;
599}
600
610char *mutt_str_skip_email_wsp(const char *s)
611{
612 if (!s)
613 return NULL;
614
615 for (; mutt_str_is_email_wsp(*s); s++)
616 ; // Do nothing
617
618 return (char *) s;
619}
620
630size_t mutt_str_lws_len(const char *s, size_t n)
631{
632 if (!s)
633 return 0;
634
635 const char *p = s;
636 size_t len = n;
637
638 if (n == 0)
639 return 0;
640
641 for (; p < (s + n); p++)
642 {
643 if (!strchr(" \t\r\n", *p))
644 {
645 len = p - s;
646 break;
647 }
648 }
649
650 if ((len != 0) && strchr("\r\n", *(p - 1))) /* LWS doesn't end with CRLF */
651 len = 0;
652 return len;
653}
654
662bool mutt_str_equal(const char *a, const char *b)
663{
664 return (a == b) || (mutt_str_cmp(a, b) == 0);
665}
666
674bool mutt_istr_equal(const char *a, const char *b)
675{
676 return (a == b) || (mutt_istr_cmp(a, b) == 0);
677}
678
685bool mutt_str_is_ascii(const char *str, size_t len)
686{
687 if (!str)
688 return true;
689
690 for (; (*str != '\0') && (len > 0); str++, len--)
691 if ((*str & 0x80) != 0)
692 return false;
693
694 return true;
695}
696
708const char *mutt_str_find_word(const char *src)
709{
710 if (!src)
711 return NULL;
712
713 while (*src && strchr(" \t\n", *src))
714 src++;
715 while (*src && !strchr(" \t\n", *src))
716 src++;
717 return src;
718}
719
728const char *mutt_str_getenv(const char *name)
729{
730 if (!name)
731 return NULL;
732
733 const char *val = getenv(name);
734 if (val && (val[0] != '\0'))
735 return val;
736
737 return NULL;
738}
739
747int mutt_istr_remall(char *str, const char *target)
748{
749 int rc = 1;
750 if (!str || !target)
751 return rc;
752
753 // Look through an ensure all instances of the substring are gone.
754 while ((str = (char *) strcasestr(str, target)))
755 {
756 size_t target_len = mutt_str_len(target);
757 memmove(str, str + target_len, 1 + strlen(str + target_len));
758 rc = 0; // If we got here, then a substring existed and has been removed.
759 }
760
761 return rc;
762}
763
764#ifdef HAVE_VASPRINTF
773int mutt_str_asprintf(char **strp, const char *fmt, ...)
774{
775 if (!strp || !fmt)
776 return -1;
777
778 va_list ap;
779 int n;
780
781 va_start(ap, fmt);
782 n = vasprintf(strp, fmt, ap);
783 va_end(ap);
784
785 /* GNU libc man page for vasprintf(3) states that the value of *strp
786 * is undefined when the return code is -1. */
787 if (n < 0)
788 {
789 mutt_error("%s", strerror(errno)); /* LCOV_EXCL_LINE */
790 mutt_exit(1); /* LCOV_EXCL_LINE */
791 }
792
793 if (n == 0)
794 {
795 /* NeoMutt convention is to use NULL for 0-length strings */
796 FREE(strp); /* LCOV_EXCL_LINE */
797 }
798
799 return n;
800}
801#else
802/* Allocate a C-string large enough to contain the formatted string.
803 * This is essentially malloc+sprintf in one.
804 */
805int mutt_str_asprintf(char **strp, const char *fmt, ...)
806{
807 if (!strp || !fmt)
808 return -1;
809
810 int rlen = 256;
811
812 *strp = MUTT_MEM_MALLOC(rlen, char);
813 while (true)
814 {
815 va_list ap;
816 va_start(ap, fmt);
817 const int n = vsnprintf(*strp, rlen, fmt, ap);
818 va_end(ap);
819 if (n < 0)
820 {
821 FREE(strp);
822 return n;
823 }
824
825 if (n < rlen)
826 {
827 /* reduce space to just that which was used. note that 'n' does not
828 * include the terminal nul char. */
829 if (n == 0) /* convention is to use NULL for zero-length strings. */
830 FREE(strp);
831 else if (n != rlen - 1)
832 MUTT_MEM_REALLOC(strp, n + 1, char);
833 return n;
834 }
835 /* increase size and try again */
836 rlen = n + 1;
837 MUTT_MEM_REALLOC(strp, rlen, char);
838 }
839 /* not reached */
840}
841#endif /* HAVE_ASPRINTF */
842
851void mutt_str_hyphenate(char *buf, size_t buflen, const char *str)
852{
853 if (!buf || (buflen == 0) || !str)
854 return;
855
856 mutt_str_copy(buf, str, buflen);
857 for (; *buf != '\0'; buf++)
858 {
859 if (*buf == '_')
860 *buf = '-';
861 }
862}
863
889int mutt_str_inbox_cmp(const char *a, const char *b)
890{
891#define IS_INBOX(s) (mutt_istrn_equal(s, "inbox", 5) && !mutt_isalnum((s)[5]))
892#define CMP_INBOX(a, b) (IS_INBOX(b) - IS_INBOX(a))
893
894 /* fast-track in case the paths have been mutt_pretty_mailbox'ified */
895 if ((a[0] == '+') && (b[0] == '+'))
896 {
897 return CMP_INBOX(a + 1, b + 1);
898 }
899
900 const char *a_end = strrchr(a, '/');
901 const char *b_end = strrchr(b, '/');
902
903 /* If one path contains a '/', but not the other */
904 if ((!a_end) ^ (!b_end))
905 return 0;
906
907 /* If neither path contains a '/' */
908 if (!a_end)
909 return 0;
910
911 /* Compare the subpaths */
912 size_t a_len = a_end - a;
913 size_t b_len = b_end - b;
914 size_t min = MIN(a_len, b_len);
915 int same = (a[min] == '/') && (b[min] == '/') && (a[min + 1] != '\0') &&
916 (b[min + 1] != '\0') && mutt_istrn_equal(a, b, min);
917
918 if (!same)
919 return 0;
920
921 return CMP_INBOX(a + 1 + min, b + 1 + min);
922
923#undef CMP_INBOX
924#undef IS_INBOX
925}
926
933void string_array_clear(struct StringArray *arr)
934{
935 const char **str = NULL;
936 ARRAY_FOREACH(str, arr)
937 {
938 FREE(str);
939 }
940
941 ARRAY_FREE(arr);
942}
Linear Array data structure.
#define ARRAY_FOREACH(elem, head)
Iterate over all elements of the array.
Definition array.h:223
#define ARRAY_FREE(head)
Release all memory.
Definition array.h:209
ctype(3) wrapper functions
bool mutt_isspace(int arg)
Wrapper for isspace(3)
Definition ctype.c:96
int mutt_toupper(int arg)
Wrapper for toupper(3)
Definition ctype.c:140
int mutt_tolower(int arg)
Wrapper for tolower(3)
Definition ctype.c:126
void mutt_exit(int code)
Leave NeoMutt NOW.
Definition exit.c:41
Leave the program NOW.
#define mutt_error(...)
Definition logging2.h:94
int mutt_str_inbox_cmp(const char *a, const char *b)
Do two folders share the same path and one is an inbox -.
Definition string.c:889
Logging Dispatcher.
Memory management wrappers.
#define countof(x)
Definition memory.h:45
#define FREE(x)
Definition memory.h:63
#define MIN(a, b)
Definition memory.h:38
#define MUTT_MEM_REALLOC(pptr, n, type)
Definition memory.h:51
#define MUTT_MEM_MALLOC(n, type)
Definition memory.h:49
char * mutt_strn_dup(const char *begin, size_t len)
Duplicate a sub-string.
Definition string.c:384
int mutt_str_cmp(const char *a, const char *b)
Compare two strings, safely.
Definition string.c:403
int mutt_istrn_cmp(const char *a, const char *b, size_t num)
Compare two strings ignoring case (to a maximum), safely.
Definition string.c:443
void mutt_str_remove_trailing_ws(char *s)
Trim trailing whitespace from a string.
Definition string.c:567
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition string.c:674
static size_t startswith(const char *str, const char *prefix, bool match_case)
Check whether a string starts with a prefix.
Definition string.c:205
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
int mutt_str_asprintf(char **strp, const char *fmt,...)
Definition string.c:805
char * mutt_str_upper(char *str)
Convert all characters in the string to uppercase.
Definition string.c:340
#define CMP_INBOX(a, b)
char * mutt_str_lower(char *str)
Convert all characters in the string to lowercase.
Definition string.c:317
char * mutt_str_skip_email_wsp(const char *s)
Skip over whitespace as defined by RFC5322.
Definition string.c:610
const char * mutt_istrn_rfind(const char *haystack, size_t haystack_length, const char *needle)
Find last instance of a substring, ignoring case.
Definition string.c:473
size_t mutt_str_lws_len(const char *s, size_t n)
Measure the linear-white-space at the beginning of a string.
Definition string.c:630
bool mutt_str_is_ascii(const char *str, size_t len)
Is a string ASCII (7-bit)?
Definition string.c:685
int mutt_istr_cmp(const char *a, const char *b)
Compare two strings ignoring case, safely.
Definition string.c:416
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
Definition string.c:662
bool mutt_strn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings (to a maximum), safely.
Definition string.c:429
static char * strsep(char **stringp, const char *delim)
Extract a token from a string.
Definition string.c:86
const char * mutt_str_find_word(const char *src)
Find the end of a word (non-space)
Definition string.c:708
char * mutt_strn_copy(char *dest, const char *src, size_t len, size_t dsize)
Copy a sub-string into a buffer.
Definition string.c:364
int mutt_istr_remall(char *str, const char *target)
Remove all occurrences of substring, ignoring case.
Definition string.c:747
const char * mutt_str_getenv(const char *name)
Get an environment variable.
Definition string.c:728
const char * mutt_istr_find(const char *haystack, const char *needle)
Find first occurrence of string (ignoring case)
Definition string.c:525
size_t mutt_str_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix.
Definition string.c:234
void mutt_str_hyphenate(char *buf, size_t buflen, const char *str)
Hyphenate a snake-case string.
Definition string.c:851
char * mutt_str_skip_whitespace(const char *p)
Find the first non-whitespace character in a string.
Definition string.c:553
void mutt_str_adjust(char **ptr)
Shrink-to-fit a string.
Definition string.c:303
void string_array_clear(struct StringArray *arr)
Free all memory of a StringArray.
Definition string.c:933
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
Definition string.c:500
size_t mutt_str_copy(char *dest, const char *src, size_t dsize)
Copy a string into a buffer (guaranteeing NUL-termination)
Definition string.c:583
static char * strcasestr(const char *haystack, const char *needle)
Find the first occurrence of needle in haystack, ignoring case.
Definition string.c:60
size_t mutt_istr_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix, ignoring case.
Definition string.c:246
bool mutt_istrn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings ignoring case (to a maximum), safely.
Definition string.c:457
const char * mutt_str_sysexit(int err_num)
Return a string matching an error code.
Definition string.c:173
char * mutt_str_replace(char **p, const char *s)
Replace one string with another.
Definition string.c:284
int mutt_str_coll(const char *a, const char *b)
Collate two strings (compare using locale), safely.
Definition string.c:513
char * mutt_str_sep(char **stringp, const char *delim)
Find first occurrence of any of delim characters in *stringp.
Definition string.c:190
String manipulation functions.
#define S_ERR
Definition string2.h:47
#define NONULL(x)
Definition string2.h:44
static bool mutt_str_is_email_wsp(char c)
Is this a whitespace character (for an email header)
Definition string2.h:111
#define SKIPWS(ch)
Definition string2.h:52
Lookup table of error messages.
Definition string.c:113
const char * err_str
Human-readable string for error.
Definition string.c:115
int err_num
Error number, see errno(3)
Definition string.c:114