2 * Claws Mail -- a GTK based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2021 The Claws Mail Team and Hiroyuki Yamamoto
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 * The code of the g_utf8_substring function below is owned by
19 * Matthias Clasen <matthiasc@src.gnome.org>/<mclasen@redhat.com>
20 * and is got from GLIB 2.30: https://git.gnome.org/browse/glib/commit/
21 * ?h=glib-2-30&id=9eb65dd3ed5e1a9638595cbe10699c7606376511
23 * GLib 2.30 is licensed under GPL v2 or later and:
24 * Copyright (C) 1999 Tom Tromey
25 * Copyright (C) 2000 Red Hat, Inc.
27 * https://git.gnome.org/browse/glib/tree/glib/gutf8.c
28 * ?h=glib-2-30&id=9eb65dd3ed5e1a9638595cbe10699c7606376511
33 #include "claws-features.h"
41 #include <glib/gi18n.h>
51 #include <sys/param.h>
53 # include <ws2tcpip.h>
55 # include <sys/socket.h>
58 #if (HAVE_WCTYPE_H && HAVE_WCHAR_H)
66 #include <sys/types.h>
68 # include <sys/wait.h>
75 #include <sys/utsname.h>
90 #include "file-utils.h"
94 static gboolean debug_mode
= FALSE
;
96 void list_free_strings_full(GList
*list
)
98 g_list_free_full(list
, (GDestroyNotify
)g_free
);
101 void slist_free_strings_full(GSList
*list
)
103 g_slist_free_full(list
, (GDestroyNotify
)g_free
);
106 static void hash_free_strings_func(gpointer key
, gpointer value
, gpointer data
)
111 void hash_free_strings(GHashTable
*table
)
113 g_hash_table_foreach(table
, hash_free_strings_func
, NULL
);
116 gint
str_case_equal(gconstpointer v
, gconstpointer v2
)
118 return g_ascii_strcasecmp((const gchar
*)v
, (const gchar
*)v2
) == 0;
121 guint
str_case_hash(gconstpointer key
)
123 const gchar
*p
= key
;
127 h
= g_ascii_tolower(h
);
128 for (p
+= 1; *p
!= '\0'; p
++)
129 h
= (h
<< 5) - h
+ g_ascii_tolower(*p
);
135 gint
to_number(const gchar
*nstr
)
137 register const gchar
*p
;
139 if (*nstr
== '\0') return -1;
141 for (p
= nstr
; *p
!= '\0'; p
++)
142 if (!g_ascii_isdigit(*p
)) return -1;
147 /* convert integer into string,
148 nstr must be not lower than 11 characters length */
149 gchar
*itos_buf(gchar
*nstr
, gint n
)
151 g_snprintf(nstr
, 11, "%d", n
);
155 /* convert integer into string */
158 static gchar nstr
[11];
160 return itos_buf(nstr
, n
);
163 #define divide(num,divisor,i,d) \
165 i = num >> divisor; \
166 d = num & ((1<<divisor)-1); \
167 d = (d*100) >> divisor; \
172 * \brief Convert a given size in bytes in a human-readable string
174 * \param size The size expressed in bytes to convert in string
175 * \return The string that respresents the size in an human-readable way
177 gchar
*to_human_readable(goffset size
)
179 static gchar str
[14];
180 static gchar
*b_format
= NULL
, *kb_format
= NULL
,
181 *mb_format
= NULL
, *gb_format
= NULL
;
182 register int t
= 0, r
= 0;
183 if (b_format
== NULL
) {
185 kb_format
= _("%d.%02dKiB");
186 mb_format
= _("%d.%02dMiB");
187 gb_format
= _("%.2fGiB");
190 if (size
< (goffset
)1024) {
191 g_snprintf(str
, sizeof(str
), b_format
, (gint
)size
);
193 } else if (size
>> 10 < (goffset
)1024) {
194 divide(size
, 10, t
, r
);
195 g_snprintf(str
, sizeof(str
), kb_format
, t
, r
);
197 } else if (size
>> 20 < (goffset
)1024) {
198 divide(size
, 20, t
, r
);
199 g_snprintf(str
, sizeof(str
), mb_format
, t
, r
);
202 g_snprintf(str
, sizeof(str
), gb_format
, (gfloat
)(size
>> 30));
208 gint
path_cmp(const gchar
*s1
, const gchar
*s2
)
213 gchar
*s1buf
, *s2buf
;
216 if (s1
== NULL
|| s2
== NULL
) return -1;
217 if (*s1
== '\0' || *s2
== '\0') return -1;
220 s1buf
= g_strdup (s1
);
221 s2buf
= g_strdup (s2
);
222 subst_char (s1buf
, '/', G_DIR_SEPARATOR
);
223 subst_char (s2buf
, '/', G_DIR_SEPARATOR
);
226 #endif /* !G_OS_WIN32 */
231 if (s1
[len1
- 1] == G_DIR_SEPARATOR
) len1
--;
232 if (s2
[len2
- 1] == G_DIR_SEPARATOR
) len2
--;
234 rc
= strncmp(s1
, s2
, MAX(len1
, len2
));
238 #endif /* !G_OS_WIN32 */
242 /* remove trailing return code */
243 gchar
*strretchomp(gchar
*str
)
247 if (!*str
) return str
;
249 for (s
= str
+ strlen(str
) - 1;
250 s
>= str
&& (*s
== '\n' || *s
== '\r');
257 /* remove trailing character */
258 gchar
*strtailchomp(gchar
*str
, gchar tail_char
)
262 if (!*str
) return str
;
263 if (tail_char
== '\0') return str
;
265 for (s
= str
+ strlen(str
) - 1; s
>= str
&& *s
== tail_char
; s
--)
271 /* remove CR (carriage return) */
272 gchar
*strcrchomp(gchar
*str
)
276 if (!*str
) return str
;
278 s
= str
+ strlen(str
) - 1;
279 if (*s
== '\n' && s
> str
&& *(s
- 1) == '\r') {
287 /* truncates string at first CR (carriage return) or LF (line feed) */
288 gchar
*strcrlftrunc(gchar
*str
)
292 if ((str
== NULL
) || (!*str
)) return str
;
294 if ((p
= strstr(str
, "\r")) != NULL
)
296 if ((p
= strstr(str
, "\n")) != NULL
)
302 #ifndef HAVE_STRCASESTR
303 /* Similar to `strstr' but this function ignores the case of both strings. */
304 gchar
*strcasestr(const gchar
*haystack
, const gchar
*needle
)
306 size_t haystack_len
= strlen(haystack
);
308 return strncasestr(haystack
, haystack_len
, needle
);
310 #endif /* HAVE_STRCASESTR */
312 gchar
*strncasestr(const gchar
*haystack
, gint haystack_len
, const gchar
*needle
)
314 register size_t needle_len
;
316 needle_len
= strlen(needle
);
318 if (haystack_len
< needle_len
|| needle_len
== 0)
321 while (haystack_len
>= needle_len
) {
322 if (!g_ascii_strncasecmp(haystack
, needle
, needle_len
))
323 return (gchar
*)haystack
;
333 gpointer
my_memmem(gconstpointer haystack
, size_t haystacklen
,
334 gconstpointer needle
, size_t needlelen
)
336 const gchar
*haystack_
= (const gchar
*)haystack
;
337 const gchar
*needle_
= (const gchar
*)needle
;
338 const gchar
*haystack_cur
= (const gchar
*)haystack
;
339 size_t haystack_left
= haystacklen
;
342 return memchr(haystack_
, *needle_
, haystacklen
);
344 while ((haystack_cur
= memchr(haystack_cur
, *needle_
, haystack_left
))
346 if (haystacklen
- (haystack_cur
- haystack_
) < needlelen
)
348 if (memcmp(haystack_cur
+ 1, needle_
+ 1, needlelen
- 1) == 0)
349 return (gpointer
)haystack_cur
;
352 haystack_left
= haystacklen
- (haystack_cur
- haystack_
);
359 /* Copy no more than N characters of SRC to DEST, with NULL terminating. */
360 gchar
*strncpy2(gchar
*dest
, const gchar
*src
, size_t n
)
362 register const gchar
*s
= src
;
363 register gchar
*d
= dest
;
373 /* Examine if next block is non-ASCII string */
374 gboolean
is_next_nonascii(const gchar
*s
)
378 /* skip head space */
379 for (p
= s
; *p
!= '\0' && g_ascii_isspace(*p
); p
++)
381 for (; *p
!= '\0' && !g_ascii_isspace(*p
); p
++) {
382 if (*(guchar
*)p
> 127 || *(guchar
*)p
< 32)
389 gint
get_next_word_len(const gchar
*s
)
393 for (; *s
!= '\0' && !g_ascii_isspace(*s
); s
++, len
++)
399 static void trim_subject_for_compare(gchar
*str
)
403 eliminate_parenthesis(str
, '[', ']');
404 eliminate_parenthesis(str
, '(', ')');
407 srcp
= str
+ subject_get_prefix_length(str
);
409 memmove(str
, srcp
, strlen(srcp
) + 1);
412 static void trim_subject_for_sort(gchar
*str
)
418 srcp
= str
+ subject_get_prefix_length(str
);
420 memmove(str
, srcp
, strlen(srcp
) + 1);
423 /* compare subjects */
424 gint
subject_compare(const gchar
*s1
, const gchar
*s2
)
428 if (!s1
|| !s2
) return -1;
429 if (!*s1
|| !*s2
) return -1;
431 Xstrdup_a(str1
, s1
, return -1);
432 Xstrdup_a(str2
, s2
, return -1);
434 trim_subject_for_compare(str1
);
435 trim_subject_for_compare(str2
);
437 if (!*str1
|| !*str2
) return -1;
439 return strcmp(str1
, str2
);
442 gint
subject_compare_for_sort(const gchar
*s1
, const gchar
*s2
)
446 if (!s1
|| !s2
) return -1;
448 Xstrdup_a(str1
, s1
, return -1);
449 Xstrdup_a(str2
, s2
, return -1);
451 trim_subject_for_sort(str1
);
452 trim_subject_for_sort(str2
);
454 if (!g_utf8_validate(str1
, -1, NULL
)) {
455 g_warning("message subject \"%s\" failed UTF-8 validation", str1
);
457 } else if (!g_utf8_validate(str2
, -1, NULL
)) {
458 g_warning("message subject \"%s\" failed UTF-8 validation", str2
);
462 return g_utf8_collate(str1
, str2
);
465 void trim_subject(gchar
*str
)
467 register gchar
*srcp
;
473 srcp
= str
+ subject_get_prefix_length(str
);
478 } else if (*srcp
== '(') {
490 else if (*srcp
== cl
)
497 while (g_ascii_isspace(*srcp
)) srcp
++;
498 memmove(str
, srcp
, strlen(srcp
) + 1);
501 void eliminate_parenthesis(gchar
*str
, gchar op
, gchar cl
)
503 register gchar
*srcp
, *destp
;
508 while ((destp
= strchr(destp
, op
))) {
514 else if (*srcp
== cl
)
520 while (g_ascii_isspace(*srcp
)) srcp
++;
521 memmove(destp
, srcp
, strlen(srcp
) + 1);
525 void extract_parenthesis(gchar
*str
, gchar op
, gchar cl
)
527 register gchar
*srcp
, *destp
;
532 while ((srcp
= strchr(destp
, op
))) {
535 memmove(destp
, srcp
+ 1, strlen(srcp
));
540 else if (*destp
== cl
)
552 static void extract_parenthesis_with_skip_quote(gchar
*str
, gchar quote_chr
,
555 register gchar
*srcp
, *destp
;
557 gboolean in_quote
= FALSE
;
561 while ((srcp
= strchr_with_skip_quote(destp
, quote_chr
, op
))) {
564 memmove(destp
, srcp
+ 1, strlen(srcp
));
567 if (*destp
== op
&& !in_quote
)
569 else if (*destp
== cl
&& !in_quote
)
571 else if (*destp
== quote_chr
)
583 void extract_quote(gchar
*str
, gchar quote_chr
)
587 if ((str
= strchr(str
, quote_chr
))) {
589 while ((p
= strchr(p
+ 1, quote_chr
)) && (p
[-1] == '\\')) {
590 memmove(p
- 1, p
, strlen(p
) + 1);
595 memmove(str
, str
+ 1, p
- str
);
600 /* Returns a newly allocated string with all quote_chr not at the beginning
601 or the end of str escaped with '\' or the given str if not required. */
602 gchar
*escape_internal_quotes(gchar
*str
, gchar quote_chr
)
604 register gchar
*p
, *q
;
608 if (str
== NULL
|| *str
== '\0')
614 /* search for unescaped quote_chr */
619 if (*p
== quote_chr
&& *(p
- 1) != '\\' && *(p
+ 1) != '\0')
623 if (!k
) /* nothing to escape */
626 /* unescaped quote_chr found */
627 qstr
= g_malloc(l
+ k
+ 1);
630 if (*p
== quote_chr
) {
635 if (*p
== quote_chr
&& *(p
- 1) != '\\' && *(p
+ 1) != '\0')
644 void eliminate_address_comment(gchar
*str
)
646 register gchar
*srcp
, *destp
;
651 while ((destp
= strchr(destp
, '"'))) {
652 if ((srcp
= strchr(destp
+ 1, '"'))) {
657 while (g_ascii_isspace(*srcp
)) srcp
++;
658 memmove(destp
, srcp
, strlen(srcp
) + 1);
668 while ((destp
= strchr_with_skip_quote(destp
, '"', '('))) {
674 else if (*srcp
== ')')
680 while (g_ascii_isspace(*srcp
)) srcp
++;
681 memmove(destp
, srcp
, strlen(srcp
) + 1);
685 gchar
*strchr_with_skip_quote(const gchar
*str
, gint quote_chr
, gint c
)
687 gboolean in_quote
= FALSE
;
690 if (*str
== c
&& !in_quote
)
692 if (*str
== quote_chr
)
700 void extract_address(gchar
*str
)
702 cm_return_if_fail(str
!= NULL
);
703 eliminate_address_comment(str
);
704 if (strchr_with_skip_quote(str
, '"', '<'))
705 extract_parenthesis_with_skip_quote(str
, '"', '<', '>');
709 void extract_list_id_str(gchar
*str
)
711 if (strchr_with_skip_quote(str
, '"', '<'))
712 extract_parenthesis_with_skip_quote(str
, '"', '<', '>');
716 static GSList
*address_list_append_real(GSList
*addr_list
, const gchar
*str
, gboolean removecomments
)
721 if (!str
) return addr_list
;
723 Xstrdup_a(work
, str
, return addr_list
);
726 eliminate_address_comment(work
);
729 while (workp
&& *workp
) {
732 if ((p
= strchr_with_skip_quote(workp
, '"', ','))) {
738 if (removecomments
&& strchr_with_skip_quote(workp
, '"', '<'))
739 extract_parenthesis_with_skip_quote
740 (workp
, '"', '<', '>');
744 addr_list
= g_slist_append(addr_list
, g_strdup(workp
));
752 GSList
*address_list_append(GSList
*addr_list
, const gchar
*str
)
754 return address_list_append_real(addr_list
, str
, TRUE
);
757 GSList
*address_list_append_with_comments(GSList
*addr_list
, const gchar
*str
)
759 return address_list_append_real(addr_list
, str
, FALSE
);
762 GSList
*references_list_prepend(GSList
*msgid_list
, const gchar
*str
)
766 if (!str
) return msgid_list
;
769 while (strp
&& *strp
) {
770 const gchar
*start
, *end
;
773 if ((start
= strchr(strp
, '<')) != NULL
) {
774 end
= strchr(start
+ 1, '>');
779 msgid
= g_strndup(start
+ 1, end
- start
- 1);
782 msgid_list
= g_slist_prepend(msgid_list
, msgid
);
792 GSList
*references_list_append(GSList
*msgid_list
, const gchar
*str
)
796 list
= references_list_prepend(NULL
, str
);
797 list
= g_slist_reverse(list
);
798 msgid_list
= g_slist_concat(msgid_list
, list
);
803 GSList
*newsgroup_list_append(GSList
*group_list
, const gchar
*str
)
808 if (!str
) return group_list
;
810 Xstrdup_a(work
, str
, return group_list
);
814 while (workp
&& *workp
) {
817 if ((p
= strchr_with_skip_quote(workp
, '"', ','))) {
825 group_list
= g_slist_append(group_list
,
834 GList
*add_history(GList
*list
, const gchar
*str
)
839 cm_return_val_if_fail(str
!= NULL
, list
);
841 old
= g_list_find_custom(list
, (gpointer
)str
, (GCompareFunc
)g_strcmp0
);
844 list
= g_list_remove(list
, old
->data
);
846 } else if (g_list_length(list
) >= MAX_HISTORY_SIZE
) {
849 last
= g_list_last(list
);
852 list
= g_list_remove(list
, last
->data
);
857 list
= g_list_prepend(list
, g_strdup(str
));
862 void remove_return(gchar
*str
)
864 register gchar
*p
= str
;
867 if (*p
== '\n' || *p
== '\r')
868 memmove(p
, p
+ 1, strlen(p
));
874 void remove_space(gchar
*str
)
876 register gchar
*p
= str
;
881 while (g_ascii_isspace(*(p
+ spc
)))
884 memmove(p
, p
+ spc
, strlen(p
+ spc
) + 1);
890 void unfold_line(gchar
*str
)
896 ch
= str
; /* iterator for source string */
899 c
= g_utf8_get_char_validated(ch
, -1);
901 if (c
== (gunichar
)-1 || c
== (gunichar
)-2) {
902 /* non-unicode byte, move past it */
907 len
= g_unichar_to_utf8(c
, NULL
);
909 if ((!g_unichar_isdefined(c
) || !g_unichar_isprint(c
) ||
910 g_unichar_isspace(c
)) && c
!= 173) {
911 /* replace anything bad or whitespacey with a single space */
915 /* move rest of the string forwards, since we just replaced
916 * a multi-byte sequence with one byte */
917 memmove(ch
, ch
+ len
-1, strlen(ch
+ len
-1) + 1);
920 /* A valid unicode character, copy it. */
926 void subst_char(gchar
*str
, gchar orig
, gchar subst
)
928 register gchar
*p
= str
;
937 void subst_chars(gchar
*str
, gchar
*orig
, gchar subst
)
939 register gchar
*p
= str
;
942 if (strchr(orig
, *p
) != NULL
)
948 void subst_for_filename(gchar
*str
)
953 subst_chars(str
, "\t\r\n\\/*?:", '_');
955 subst_chars(str
, "\t\r\n\\/*", '_');
959 void subst_for_shellsafe_filename(gchar
*str
)
963 subst_for_filename(str
);
964 subst_chars(str
, " \"'|&;()<>'!{}[]",'_');
967 gboolean
is_ascii_str(const gchar
*str
)
969 const guchar
*p
= (const guchar
*)str
;
972 if (*p
!= '\t' && *p
!= ' ' &&
973 *p
!= '\r' && *p
!= '\n' &&
974 (*p
< 32 || *p
>= 127))
982 static const gchar
* line_has_quote_char_last(const gchar
* str
, const gchar
*quote_chars
)
984 gchar
* position
= NULL
;
985 gchar
* tmp_pos
= NULL
;
988 if (str
== NULL
|| quote_chars
== NULL
)
991 for (i
= 0; i
< strlen(quote_chars
); i
++) {
992 tmp_pos
= strrchr (str
, quote_chars
[i
]);
994 || (tmp_pos
!= NULL
&& position
<= tmp_pos
) )
1000 gint
get_quote_level(const gchar
*str
, const gchar
*quote_chars
)
1002 const gchar
*first_pos
;
1003 const gchar
*last_pos
;
1004 const gchar
*p
= str
;
1005 gint quote_level
= -1;
1007 /* speed up line processing by only searching to the last '>' */
1008 if ((first_pos
= line_has_quote_char(str
, quote_chars
)) != NULL
) {
1009 /* skip a line if it contains a '<' before the initial '>' */
1010 if (memchr(str
, '<', first_pos
- str
) != NULL
)
1012 last_pos
= line_has_quote_char_last(first_pos
, quote_chars
);
1016 while (p
<= last_pos
) {
1017 while (p
< last_pos
) {
1018 if (g_ascii_isspace(*p
))
1024 if (strchr(quote_chars
, *p
))
1026 else if (*p
!= '-' && !g_ascii_isspace(*p
) && p
<= last_pos
) {
1027 /* any characters are allowed except '-','<' and space */
1028 while (*p
!= '-' && *p
!= '<'
1029 && !strchr(quote_chars
, *p
)
1030 && !g_ascii_isspace(*p
)
1033 if (strchr(quote_chars
, *p
))
1045 gint
check_line_length(const gchar
*str
, gint max_chars
, gint
*line
)
1047 const gchar
*p
= str
, *q
;
1048 gint cur_line
= 0, len
;
1050 while ((q
= strchr(p
, '\n')) != NULL
) {
1052 if (len
> max_chars
) {
1062 if (len
> max_chars
) {
1071 const gchar
* line_has_quote_char(const gchar
* str
, const gchar
*quote_chars
)
1073 gchar
* position
= NULL
;
1074 gchar
* tmp_pos
= NULL
;
1077 if (str
== NULL
|| quote_chars
== NULL
)
1080 for (i
= 0; i
< strlen(quote_chars
); i
++) {
1081 tmp_pos
= strchr (str
, quote_chars
[i
]);
1083 || (tmp_pos
!= NULL
&& position
>= tmp_pos
) )
1089 static gchar
*strstr_with_skip_quote(const gchar
*haystack
, const gchar
*needle
)
1091 register guint haystack_len
, needle_len
;
1092 gboolean in_squote
= FALSE
, in_dquote
= FALSE
;
1094 haystack_len
= strlen(haystack
);
1095 needle_len
= strlen(needle
);
1097 if (haystack_len
< needle_len
|| needle_len
== 0)
1100 while (haystack_len
>= needle_len
) {
1101 if (!in_squote
&& !in_dquote
&&
1102 !strncmp(haystack
, needle
, needle_len
))
1103 return (gchar
*)haystack
;
1105 /* 'foo"bar"' -> foo"bar"
1106 "foo'bar'" -> foo'bar' */
1107 if (*haystack
== '\'') {
1110 else if (!in_dquote
)
1112 } else if (*haystack
== '\"') {
1115 else if (!in_squote
)
1117 } else if (*haystack
== '\\') {
1129 gchar
**strsplit_with_quote(const gchar
*str
, const gchar
*delim
,
1132 GSList
*string_list
= NULL
, *slist
;
1133 gchar
**str_array
, *s
, *new_str
;
1134 guint i
, n
= 1, len
;
1136 cm_return_val_if_fail(str
!= NULL
, NULL
);
1137 cm_return_val_if_fail(delim
!= NULL
, NULL
);
1140 max_tokens
= G_MAXINT
;
1142 s
= strstr_with_skip_quote(str
, delim
);
1144 guint delimiter_len
= strlen(delim
);
1148 new_str
= g_strndup(str
, len
);
1150 if (new_str
[0] == '\'' || new_str
[0] == '\"') {
1151 if (new_str
[len
- 1] == new_str
[0]) {
1152 new_str
[len
- 1] = '\0';
1153 memmove(new_str
, new_str
+ 1, len
- 1);
1156 string_list
= g_slist_prepend(string_list
, new_str
);
1158 str
= s
+ delimiter_len
;
1159 s
= strstr_with_skip_quote(str
, delim
);
1160 } while (--max_tokens
&& s
);
1164 new_str
= g_strdup(str
);
1165 if (new_str
[0] == '\'' || new_str
[0] == '\"') {
1167 if (new_str
[len
- 1] == new_str
[0]) {
1168 new_str
[len
- 1] = '\0';
1169 memmove(new_str
, new_str
+ 1, len
- 1);
1172 string_list
= g_slist_prepend(string_list
, new_str
);
1176 str_array
= g_new(gchar
*, n
);
1180 str_array
[i
--] = NULL
;
1181 for (slist
= string_list
; slist
; slist
= slist
->next
)
1182 str_array
[i
--] = slist
->data
;
1184 g_slist_free(string_list
);
1189 gchar
*get_abbrev_newsgroup_name(const gchar
*group
, gint len
)
1191 gchar
*abbrev_group
;
1193 const gchar
*p
= group
;
1196 cm_return_val_if_fail(group
!= NULL
, NULL
);
1198 last
= group
+ strlen(group
);
1199 abbrev_group
= ap
= g_malloc(strlen(group
) + 1);
1204 if ((ap
- abbrev_group
) + (last
- p
) > len
&& strchr(p
, '.')) {
1206 while (*p
!= '.') p
++;
1209 return abbrev_group
;
1214 return abbrev_group
;
1217 gchar
*trim_string(const gchar
*str
, gint len
)
1219 const gchar
*p
= str
;
1224 if (!str
) return NULL
;
1225 if (strlen(str
) <= len
)
1226 return g_strdup(str
);
1227 if (g_utf8_validate(str
, -1, NULL
) == FALSE
)
1228 return g_strdup(str
);
1230 while (*p
!= '\0') {
1231 mb_len
= g_utf8_skip
[*(guchar
*)p
];
1234 else if (new_len
+ mb_len
> len
)
1241 Xstrndup_a(new_str
, str
, new_len
, return g_strdup(str
));
1242 return g_strconcat(new_str
, "...", NULL
);
1245 GList
*uri_list_extract_filenames(const gchar
*uri_list
)
1247 GList
*result
= NULL
;
1249 gchar
*escaped_utf8uri
;
1255 while (g_ascii_isspace(*p
)) p
++;
1256 if (!strncmp(p
, "file:", 5)) {
1259 while (*q
&& *q
!= '\n' && *q
!= '\r') q
++;
1262 gchar
*file
, *locale_file
= NULL
;
1264 while (q
> p
&& g_ascii_isspace(*q
))
1266 Xalloca(escaped_utf8uri
, q
- p
+ 2,
1268 Xalloca(file
, q
- p
+ 2,
1271 strncpy(escaped_utf8uri
, p
, q
- p
+ 1);
1272 escaped_utf8uri
[q
- p
+ 1] = '\0';
1273 decode_uri_with_plus(file
, escaped_utf8uri
, FALSE
);
1275 * g_filename_from_uri() rejects escaped/locale encoded uri
1276 * string which come from Nautilus.
1279 if (g_utf8_validate(file
, -1, NULL
))
1281 = conv_codeset_strdup(
1284 conv_get_locale_charset_str());
1286 locale_file
= g_strdup(file
+ 5);
1288 locale_file
= g_filename_from_uri(escaped_utf8uri
, NULL
, NULL
);
1290 result
= g_list_append(result
, locale_file
);
1294 p
= strchr(p
, '\n');
1301 /* Converts two-digit hexadecimal to decimal. Used for unescaping escaped
1304 static gint
axtoi(const gchar
*hexstr
)
1306 gint hi
, lo
, result
;
1309 if ('0' <= hi
&& hi
<= '9') {
1312 if ('a' <= hi
&& hi
<= 'f') {
1315 if ('A' <= hi
&& hi
<= 'F') {
1320 if ('0' <= lo
&& lo
<= '9') {
1323 if ('a' <= lo
&& lo
<= 'f') {
1326 if ('A' <= lo
&& lo
<= 'F') {
1329 result
= lo
+ (16 * hi
);
1333 gboolean
is_uri_string(const gchar
*str
)
1335 while (str
&& *str
&& g_ascii_isspace(*str
))
1337 return (g_ascii_strncasecmp(str
, "http://", 7) == 0 ||
1338 g_ascii_strncasecmp(str
, "https://", 8) == 0 ||
1339 g_ascii_strncasecmp(str
, "ftp://", 6) == 0 ||
1340 g_ascii_strncasecmp(str
, "ftps://", 7) == 0 ||
1341 g_ascii_strncasecmp(str
, "sftp://", 7) == 0 ||
1342 g_ascii_strncasecmp(str
, "ftp.", 4) == 0 ||
1343 g_ascii_strncasecmp(str
, "webcal://", 9) == 0 ||
1344 g_ascii_strncasecmp(str
, "webcals://", 10) == 0 ||
1345 g_ascii_strncasecmp(str
, "www.", 4) == 0);
1348 gchar
*get_uri_path(const gchar
*uri
)
1350 while (uri
&& *uri
&& g_ascii_isspace(*uri
))
1352 if (g_ascii_strncasecmp(uri
, "http://", 7) == 0)
1353 return (gchar
*)(uri
+ 7);
1354 else if (g_ascii_strncasecmp(uri
, "https://", 8) == 0)
1355 return (gchar
*)(uri
+ 8);
1356 else if (g_ascii_strncasecmp(uri
, "ftp://", 6) == 0)
1357 return (gchar
*)(uri
+ 6);
1358 else if (g_ascii_strncasecmp(uri
, "ftps://", 7) == 0)
1359 return (gchar
*)(uri
+ 7);
1360 else if (g_ascii_strncasecmp(uri
, "sftp://", 7) == 0)
1361 return (gchar
*)(uri
+ 7);
1362 else if (g_ascii_strncasecmp(uri
, "webcal://", 9) == 0)
1363 return (gchar
*)(uri
+ 7);
1364 else if (g_ascii_strncasecmp(uri
, "webcals://", 10) == 0)
1365 return (gchar
*)(uri
+ 7);
1367 return (gchar
*)uri
;
1370 gint
get_uri_len(const gchar
*str
)
1374 if (is_uri_string(str
)) {
1375 for (p
= str
; *p
!= '\0'; p
++) {
1376 if (!g_ascii_isgraph(*p
) || strchr("<>\"", *p
))
1385 /* Decodes URL-Encoded strings (i.e. strings in which spaces are replaced by
1386 * plusses, and escape characters are used)
1388 void decode_uri_with_plus(gchar
*decoded_uri
, const gchar
*encoded_uri
, gboolean with_plus
)
1390 gchar
*dec
= decoded_uri
;
1391 const gchar
*enc
= encoded_uri
;
1396 if (isxdigit((guchar
)enc
[0]) &&
1397 isxdigit((guchar
)enc
[1])) {
1403 if (with_plus
&& *enc
== '+')
1415 void decode_uri(gchar
*decoded_uri
, const gchar
*encoded_uri
)
1417 decode_uri_with_plus(decoded_uri
, encoded_uri
, TRUE
);
1420 static gchar
*decode_uri_gdup(const gchar
*encoded_uri
)
1422 gchar
*buffer
= g_malloc(strlen(encoded_uri
)+1);
1423 decode_uri_with_plus(buffer
, encoded_uri
, FALSE
);
1427 gint
scan_mailto_url(const gchar
*mailto
, gchar
**from
, gchar
**to
, gchar
**cc
, gchar
**bcc
,
1428 gchar
**subject
, gchar
**body
, gchar
***attach
, gchar
**inreplyto
)
1432 const gchar
*forbidden_uris
[] = { ".gnupg/",
1438 gint num_attach
= 0;
1440 cm_return_val_if_fail(mailto
!= NULL
, -1);
1442 Xstrdup_a(tmp_mailto
, mailto
, return -1);
1444 if (!strncmp(tmp_mailto
, "mailto:", 7))
1447 p
= strchr(tmp_mailto
, '?');
1454 *to
= decode_uri_gdup(tmp_mailto
);
1457 gchar
*field
, *value
;
1474 if (*value
== '\0') continue;
1476 if (from
&& !g_ascii_strcasecmp(field
, "from")) {
1478 *from
= decode_uri_gdup(value
);
1480 gchar
*tmp
= decode_uri_gdup(value
);
1481 gchar
*new_from
= g_strdup_printf("%s, %s", *from
, tmp
);
1486 } else if (cc
&& !g_ascii_strcasecmp(field
, "cc")) {
1488 *cc
= decode_uri_gdup(value
);
1490 gchar
*tmp
= decode_uri_gdup(value
);
1491 gchar
*new_cc
= g_strdup_printf("%s, %s", *cc
, tmp
);
1496 } else if (bcc
&& !g_ascii_strcasecmp(field
, "bcc")) {
1498 *bcc
= decode_uri_gdup(value
);
1500 gchar
*tmp
= decode_uri_gdup(value
);
1501 gchar
*new_bcc
= g_strdup_printf("%s, %s", *bcc
, tmp
);
1506 } else if (subject
&& !*subject
&&
1507 !g_ascii_strcasecmp(field
, "subject")) {
1508 *subject
= decode_uri_gdup(value
);
1509 } else if (body
&& !*body
&& !g_ascii_strcasecmp(field
, "body")) {
1510 *body
= decode_uri_gdup(value
);
1511 } else if (body
&& !*body
&& !g_ascii_strcasecmp(field
, "insert")) {
1513 gchar
*tmp
= decode_uri_gdup(value
);
1515 for (; forbidden_uris
[i
]; i
++) {
1516 if (strstr(tmp
, forbidden_uris
[i
])) {
1517 g_print("Refusing to insert '%s', potential private data leak\n",
1526 if (!is_file_entry_regular(tmp
)) {
1527 g_warning("refusing to insert '%s', not a regular file", tmp
);
1528 } else if (!g_file_get_contents(tmp
, body
, NULL
, NULL
)) {
1529 g_warning("couldn't set insert file '%s' in body", value
);
1534 } else if (attach
&& !g_ascii_strcasecmp(field
, "attach")) {
1536 gchar
*tmp
= decode_uri_gdup(value
);
1537 gchar
**my_att
= g_malloc(sizeof(char *));
1541 for (; forbidden_uris
[i
]; i
++) {
1542 if (strstr(tmp
, forbidden_uris
[i
])) {
1543 g_print("Refusing to attach '%s', potential private data leak\n",
1551 /* attach is correct */
1553 my_att
= g_realloc(my_att
, (sizeof(char *))*(num_attach
+1));
1554 my_att
[num_attach
-1] = tmp
;
1555 my_att
[num_attach
] = NULL
;
1560 } else if (inreplyto
&& !*inreplyto
&&
1561 !g_ascii_strcasecmp(field
, "in-reply-to")) {
1562 *inreplyto
= decode_uri_gdup(value
);
1571 #include <windows.h>
1572 #ifndef CSIDL_APPDATA
1573 #define CSIDL_APPDATA 0x001a
1575 #ifndef CSIDL_LOCAL_APPDATA
1576 #define CSIDL_LOCAL_APPDATA 0x001c
1578 #ifndef CSIDL_FLAG_CREATE
1579 #define CSIDL_FLAG_CREATE 0x8000
1581 #define DIM(v) (sizeof(v)/sizeof((v)[0]))
1585 w32_strerror (int w32_errno
)
1587 static char strerr
[256];
1588 int ec
= (int)GetLastError ();
1592 FormatMessage (FORMAT_MESSAGE_FROM_SYSTEM
, NULL
, w32_errno
,
1593 MAKELANGID (LANG_NEUTRAL
, SUBLANG_DEFAULT
),
1594 strerr
, DIM (strerr
)-1, NULL
);
1598 static __inline__
void *
1599 dlopen (const char * name
, int flag
)
1601 void * hd
= LoadLibrary (name
);
1605 static __inline__
void *
1606 dlsym (void * hd
, const char * sym
)
1610 void * fnc
= GetProcAddress (hd
, sym
);
1619 static __inline__
const char *
1622 return w32_strerror (0);
1626 static __inline__
int
1638 w32_shgetfolderpath (HWND a
, int b
, HANDLE c
, DWORD d
, LPSTR e
)
1640 static int initialized
;
1641 static HRESULT (WINAPI
* func
)(HWND
,int,HANDLE
,DWORD
,LPSTR
);
1645 static char *dllnames
[] = { "shell32.dll", "shfolder.dll", NULL
};
1651 for (i
=0, handle
= NULL
; !handle
&& dllnames
[i
]; i
++)
1653 handle
= dlopen (dllnames
[i
], RTLD_LAZY
);
1656 func
= dlsym (handle
, "SHGetFolderPathW");
1667 return func (a
,b
,c
,d
,e
);
1672 /* Returns a static string with the directroy from which the module
1673 has been loaded. Returns an empty string on error. */
1674 static char *w32_get_module_dir(void)
1676 static char *moddir
;
1679 char name
[MAX_PATH
+10];
1682 if ( !GetModuleFileNameA (0, name
, sizeof (name
)-10) )
1685 p
= strrchr (name
, '\\');
1691 moddir
= g_strdup (name
);
1695 #endif /* G_OS_WIN32 */
1697 /* Return a static string with the locale dir. */
1698 const gchar
*get_locale_dir(void)
1700 static gchar
*loc_dir
;
1704 loc_dir
= g_strconcat(w32_get_module_dir(), G_DIR_SEPARATOR_S
,
1705 "\\share\\locale", NULL
);
1708 loc_dir
= LOCALEDIR
;
1714 const gchar
*get_home_dir(void)
1717 static char home_dir_utf16
[MAX_PATH
] = "";
1718 static gchar
*home_dir_utf8
= NULL
;
1719 if (home_dir_utf16
[0] == '\0') {
1720 if (w32_shgetfolderpath
1721 (NULL
, CSIDL_APPDATA
|CSIDL_FLAG_CREATE
,
1722 NULL
, 0, home_dir_utf16
) < 0)
1723 strcpy (home_dir_utf16
, "C:\\Claws Mail");
1724 home_dir_utf8
= g_utf16_to_utf8 ((const gunichar2
*)home_dir_utf16
, -1, NULL
, NULL
, NULL
);
1726 return home_dir_utf8
;
1728 static const gchar
*homeenv
= NULL
;
1733 if (!homeenv
&& g_getenv("HOME") != NULL
)
1734 homeenv
= g_strdup(g_getenv("HOME"));
1736 homeenv
= g_get_home_dir();
1742 static gchar
*claws_rc_dir
= NULL
;
1743 static gboolean rc_dir_alt
= FALSE
;
1744 const gchar
*get_rc_dir(void)
1747 if (!claws_rc_dir
) {
1748 claws_rc_dir
= g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S
,
1750 debug_print("using default rc_dir %s\n", claws_rc_dir
);
1752 return claws_rc_dir
;
1755 void set_rc_dir(const gchar
*dir
)
1757 gchar
*canonical_dir
;
1758 if (claws_rc_dir
!= NULL
) {
1759 g_print("Error: rc_dir already set\n");
1761 int err
= cm_canonicalize_filename(dir
, &canonical_dir
);
1765 g_print("Error looking for %s: %d(%s)\n",
1766 dir
, -err
, g_strerror(-err
));
1771 claws_rc_dir
= canonical_dir
;
1773 len
= strlen(claws_rc_dir
);
1774 if (claws_rc_dir
[len
- 1] == G_DIR_SEPARATOR
)
1775 claws_rc_dir
[len
- 1] = '\0';
1777 debug_print("set rc_dir to %s\n", claws_rc_dir
);
1778 if (!is_dir_exist(claws_rc_dir
)) {
1779 if (make_dir_hier(claws_rc_dir
) != 0) {
1780 g_print("Error: can't create %s\n",
1788 gboolean
rc_dir_is_alt(void) {
1792 const gchar
*get_mail_base_dir(void)
1794 return get_home_dir();
1797 const gchar
*get_news_cache_dir(void)
1799 static gchar
*news_cache_dir
= NULL
;
1800 if (!news_cache_dir
)
1801 news_cache_dir
= g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S
,
1802 NEWS_CACHE_DIR
, NULL
);
1804 return news_cache_dir
;
1807 const gchar
*get_imap_cache_dir(void)
1809 static gchar
*imap_cache_dir
= NULL
;
1811 if (!imap_cache_dir
)
1812 imap_cache_dir
= g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S
,
1813 IMAP_CACHE_DIR
, NULL
);
1815 return imap_cache_dir
;
1818 const gchar
*get_mime_tmp_dir(void)
1820 static gchar
*mime_tmp_dir
= NULL
;
1823 mime_tmp_dir
= g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S
,
1824 MIME_TMP_DIR
, NULL
);
1826 return mime_tmp_dir
;
1829 const gchar
*get_template_dir(void)
1831 static gchar
*template_dir
= NULL
;
1834 template_dir
= g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S
,
1835 TEMPLATE_DIR
, NULL
);
1837 return template_dir
;
1841 const gchar
*w32_get_cert_file(void)
1843 const gchar
*cert_file
= NULL
;
1845 cert_file
= g_strconcat(w32_get_module_dir(),
1846 "\\share\\claws-mail\\",
1847 "ca-certificates.crt",
1853 /* Return the filepath of the claws-mail.desktop file */
1854 const gchar
*get_desktop_file(void)
1856 #ifdef DESKTOPFILEPATH
1857 return DESKTOPFILEPATH
;
1863 /* Return the default directory for Plugins. */
1864 const gchar
*get_plugin_dir(void)
1867 static gchar
*plugin_dir
= NULL
;
1870 plugin_dir
= g_strconcat(w32_get_module_dir(),
1871 "\\lib\\claws-mail\\plugins\\",
1875 if (is_dir_exist(PLUGINDIR
))
1878 static gchar
*plugin_dir
= NULL
;
1880 plugin_dir
= g_strconcat(get_rc_dir(),
1881 G_DIR_SEPARATOR_S
, "plugins",
1882 G_DIR_SEPARATOR_S
, NULL
);
1890 /* Return the default directory for Themes. */
1891 const gchar
*w32_get_themes_dir(void)
1893 static gchar
*themes_dir
= NULL
;
1896 themes_dir
= g_strconcat(w32_get_module_dir(),
1897 "\\share\\claws-mail\\themes",
1903 const gchar
*get_tmp_dir(void)
1905 static gchar
*tmp_dir
= NULL
;
1908 tmp_dir
= g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S
,
1914 gchar
*get_tmp_file(void)
1917 static guint32 id
= 0;
1919 tmp_file
= g_strdup_printf("%s%ctmpfile.%08x",
1920 get_tmp_dir(), G_DIR_SEPARATOR
, id
++);
1925 const gchar
*get_domain_name(void)
1928 static gchar
*domain_name
= NULL
;
1929 struct addrinfo hints
, *res
;
1934 if (gethostname(hostname
, sizeof(hostname
)) != 0) {
1935 perror("gethostname");
1936 domain_name
= "localhost";
1938 memset(&hints
, 0, sizeof(struct addrinfo
));
1939 hints
.ai_family
= AF_UNSPEC
;
1940 hints
.ai_socktype
= 0;
1941 hints
.ai_flags
= AI_CANONNAME
;
1942 hints
.ai_protocol
= 0;
1944 s
= getaddrinfo(hostname
, NULL
, &hints
, &res
);
1946 fprintf(stderr
, "getaddrinfo: %s\n", gai_strerror(s
));
1947 domain_name
= g_strdup(hostname
);
1949 domain_name
= g_strdup(res
->ai_canonname
);
1953 debug_print("domain name = %s\n", domain_name
);
1962 /* Tells whether the given host address string is a valid representation of a
1963 * numerical IP (v4 or, if supported, v6) address.
1965 gboolean
is_numeric_host_address(const gchar
*hostaddress
)
1967 struct addrinfo hints
, *res
;
1970 /* See what getaddrinfo makes of the string when told that it is a
1971 * numeric IP address representation. */
1972 memset(&hints
, 0, sizeof(struct addrinfo
));
1973 hints
.ai_family
= AF_UNSPEC
;
1974 hints
.ai_socktype
= 0;
1975 hints
.ai_flags
= AI_NUMERICHOST
;
1976 hints
.ai_protocol
= 0;
1978 err
= getaddrinfo(hostaddress
, NULL
, &hints
, &res
);
1985 off_t
get_file_size(const gchar
*file
)
1990 GError
*error
= NULL
;
1993 f
= g_file_new_for_path(file
);
1994 fi
= g_file_query_info(f
, "standard::size",
1995 G_FILE_QUERY_INFO_NONE
, NULL
, &error
);
1996 if (error
!= NULL
) {
1997 debug_print("get_file_size error: %s\n", error
->message
);
1998 g_error_free(error
);
2002 size
= g_file_info_get_size(fi
);
2010 if (g_stat(file
, &s
) < 0) {
2011 FILE_OP_ERROR(file
, "stat");
2019 time_t get_file_mtime(const gchar
*file
)
2023 if (g_stat(file
, &s
) < 0) {
2024 FILE_OP_ERROR(file
, "stat");
2031 gboolean
file_exist(const gchar
*file
, gboolean allow_fifo
)
2038 if (g_stat(file
, &s
) < 0) {
2039 if (ENOENT
!= errno
) FILE_OP_ERROR(file
, "stat");
2043 if (S_ISREG(s
.st_mode
) || (allow_fifo
&& S_ISFIFO(s
.st_mode
)))
2050 /* Test on whether FILE is a relative file name. This is
2051 * straightforward for Unix but more complex for Windows. */
2052 gboolean
is_relative_filename(const gchar
*file
)
2057 if ( *file
== '\\' && file
[1] == '\\' && strchr (file
+2, '\\') )
2058 return FALSE
; /* Prefixed with a hostname - this can't
2059 * be a relative name. */
2061 if ( ((*file
>= 'a' && *file
<= 'z')
2062 || (*file
>= 'A' && *file
<= 'Z'))
2064 file
+= 2; /* Skip drive letter. */
2066 return !(*file
== '\\' || *file
== '/');
2068 return !(*file
== G_DIR_SEPARATOR
);
2073 gboolean
is_dir_exist(const gchar
*dir
)
2078 return g_file_test(dir
, G_FILE_TEST_IS_DIR
);
2081 gboolean
is_file_entry_exist(const gchar
*file
)
2086 return g_file_test(file
, G_FILE_TEST_EXISTS
);
2089 gboolean
is_file_entry_regular(const gchar
*file
)
2094 return g_file_test(file
, G_FILE_TEST_IS_REGULAR
);
2097 gboolean
dirent_is_regular_file(struct dirent
*d
)
2099 #if !defined(G_OS_WIN32) && defined(HAVE_DIRENT_D_TYPE)
2100 if (d
->d_type
== DT_REG
)
2102 else if (d
->d_type
!= DT_UNKNOWN
)
2106 return g_file_test(d
->d_name
, G_FILE_TEST_IS_REGULAR
);
2109 gint
change_dir(const gchar
*dir
)
2111 gchar
*prevdir
= NULL
;
2114 prevdir
= g_get_current_dir();
2116 if (g_chdir(dir
) < 0) {
2117 FILE_OP_ERROR(dir
, "chdir");
2118 if (debug_mode
) g_free(prevdir
);
2120 } else if (debug_mode
) {
2123 cwd
= g_get_current_dir();
2124 if (strcmp(prevdir
, cwd
) != 0)
2125 g_print("current dir: %s\n", cwd
);
2133 gint
make_dir(const gchar
*dir
)
2135 if (g_mkdir(dir
, S_IRWXU
) < 0) {
2136 FILE_OP_ERROR(dir
, "mkdir");
2139 if (g_chmod(dir
, S_IRWXU
) < 0)
2140 FILE_OP_ERROR(dir
, "chmod");
2145 gint
make_dir_hier(const gchar
*dir
)
2150 for (p
= dir
; (p
= strchr(p
, G_DIR_SEPARATOR
)) != NULL
; p
++) {
2151 parent_dir
= g_strndup(dir
, p
- dir
);
2152 if (*parent_dir
!= '\0') {
2153 if (!is_dir_exist(parent_dir
)) {
2154 if (make_dir(parent_dir
) < 0) {
2163 if (!is_dir_exist(dir
)) {
2164 if (make_dir(dir
) < 0)
2171 gint
remove_all_files(const gchar
*dir
)
2174 const gchar
*file_name
;
2177 if ((dp
= g_dir_open(dir
, 0, NULL
)) == NULL
) {
2178 g_warning("failed to open directory: %s", dir
);
2182 while ((file_name
= g_dir_read_name(dp
)) != NULL
) {
2183 tmp
= g_strconcat(dir
, G_DIR_SEPARATOR_S
, file_name
, NULL
);
2184 if (claws_unlink(tmp
) < 0)
2185 FILE_OP_ERROR(tmp
, "unlink");
2194 gint
remove_numbered_files(const gchar
*dir
, guint first
, guint last
)
2197 const gchar
*dir_name
;
2201 if (first
== last
) {
2202 /* Skip all the dir reading part. */
2203 gchar
*filename
= g_strdup_printf("%s%s%u", dir
, G_DIR_SEPARATOR_S
, first
);
2204 if (is_dir_exist(filename
)) {
2205 /* a numbered directory with this name exists,
2206 * remove the dot-file instead */
2208 filename
= g_strdup_printf("%s%s.%u", dir
, G_DIR_SEPARATOR_S
, first
);
2210 if (claws_unlink(filename
) < 0) {
2211 FILE_OP_ERROR(filename
, "unlink");
2219 prev_dir
= g_get_current_dir();
2221 if (g_chdir(dir
) < 0) {
2222 FILE_OP_ERROR(dir
, "chdir");
2227 if ((dp
= g_dir_open(".", 0, NULL
)) == NULL
) {
2228 g_warning("failed to open directory: %s", dir
);
2233 while ((dir_name
= g_dir_read_name(dp
)) != NULL
) {
2234 file_no
= to_number(dir_name
);
2235 if (file_no
> 0 && first
<= file_no
&& file_no
<= last
) {
2236 if (is_dir_exist(dir_name
)) {
2237 gchar
*dot_file
= g_strdup_printf(".%s", dir_name
);
2238 if (is_file_exist(dot_file
) && claws_unlink(dot_file
) < 0) {
2239 FILE_OP_ERROR(dot_file
, "unlink");
2244 if (claws_unlink(dir_name
) < 0)
2245 FILE_OP_ERROR(dir_name
, "unlink");
2251 if (g_chdir(prev_dir
) < 0) {
2252 FILE_OP_ERROR(prev_dir
, "chdir");
2262 gint
remove_numbered_files_not_in_list(const gchar
*dir
, GSList
*numberlist
)
2265 const gchar
*dir_name
;
2268 GHashTable
*wanted_files
;
2270 GError
*error
= NULL
;
2272 if (numberlist
== NULL
)
2275 prev_dir
= g_get_current_dir();
2277 if (g_chdir(dir
) < 0) {
2278 FILE_OP_ERROR(dir
, "chdir");
2283 if ((dp
= g_dir_open(".", 0, &error
)) == NULL
) {
2284 g_message("Couldn't open current directory: %s (%d).\n",
2285 error
->message
, error
->code
);
2286 g_error_free(error
);
2291 wanted_files
= g_hash_table_new(g_direct_hash
, g_direct_equal
);
2292 for (cur
= numberlist
; cur
!= NULL
; cur
= cur
->next
) {
2293 /* numberlist->data is expected to be GINT_TO_POINTER */
2294 g_hash_table_insert(wanted_files
, cur
->data
, GINT_TO_POINTER(1));
2297 while ((dir_name
= g_dir_read_name(dp
)) != NULL
) {
2298 file_no
= to_number(dir_name
);
2299 if (is_dir_exist(dir_name
))
2301 if (file_no
> 0 && g_hash_table_lookup(wanted_files
, GINT_TO_POINTER(file_no
)) == NULL
) {
2302 debug_print("removing unwanted file %d from %s\n", file_no
, dir
);
2303 if (is_dir_exist(dir_name
)) {
2304 gchar
*dot_file
= g_strdup_printf(".%s", dir_name
);
2305 if (is_file_exist(dot_file
) && claws_unlink(dot_file
) < 0) {
2306 FILE_OP_ERROR(dot_file
, "unlink");
2311 if (claws_unlink(dir_name
) < 0)
2312 FILE_OP_ERROR(dir_name
, "unlink");
2317 g_hash_table_destroy(wanted_files
);
2319 if (g_chdir(prev_dir
) < 0) {
2320 FILE_OP_ERROR(prev_dir
, "chdir");
2330 gint
remove_all_numbered_files(const gchar
*dir
)
2332 return remove_numbered_files(dir
, 0, UINT_MAX
);
2335 gint
remove_dir_recursive(const gchar
*dir
)
2339 const gchar
*dir_name
;
2342 if (g_stat(dir
, &s
) < 0) {
2343 FILE_OP_ERROR(dir
, "stat");
2344 if (ENOENT
== errno
) return 0;
2348 if (!S_ISDIR(s
.st_mode
)) {
2349 if (claws_unlink(dir
) < 0) {
2350 FILE_OP_ERROR(dir
, "unlink");
2357 prev_dir
= g_get_current_dir();
2358 /* g_print("prev_dir = %s\n", prev_dir); */
2360 if (!path_cmp(prev_dir
, dir
)) {
2362 if (g_chdir("..") < 0) {
2363 FILE_OP_ERROR(dir
, "chdir");
2366 prev_dir
= g_get_current_dir();
2369 if (g_chdir(dir
) < 0) {
2370 FILE_OP_ERROR(dir
, "chdir");
2375 if ((dp
= g_dir_open(".", 0, NULL
)) == NULL
) {
2376 g_warning("failed to open directory: %s", dir
);
2382 /* remove all files in the directory */
2383 while ((dir_name
= g_dir_read_name(dp
)) != NULL
) {
2384 /* g_print("removing %s\n", dir_name); */
2386 if (is_dir_exist(dir_name
)) {
2389 if ((ret
= remove_dir_recursive(dir_name
)) < 0) {
2390 g_warning("can't remove directory: %s", dir_name
);
2395 if (claws_unlink(dir_name
) < 0)
2396 FILE_OP_ERROR(dir_name
, "unlink");
2402 if (g_chdir(prev_dir
) < 0) {
2403 FILE_OP_ERROR(prev_dir
, "chdir");
2410 if (g_rmdir(dir
) < 0) {
2411 FILE_OP_ERROR(dir
, "rmdir");
2418 /* convert line endings into CRLF. If the last line doesn't end with
2419 * linebreak, add it.
2421 gchar
*canonicalize_str(const gchar
*str
)
2427 for (p
= str
; *p
!= '\0'; ++p
) {
2434 if (p
== str
|| *(p
- 1) != '\n')
2437 out
= outp
= g_malloc(new_len
+ 1);
2438 for (p
= str
; *p
!= '\0'; ++p
) {
2445 if (p
== str
|| *(p
- 1) != '\n') {
2454 gchar
*normalize_newlines(const gchar
*str
)
2459 out
= outp
= g_malloc(strlen(str
) + 1);
2460 for (p
= str
; *p
!= '\0'; ++p
) {
2462 if (*(p
+ 1) != '\n')
2473 gchar
*get_outgoing_rfc2822_str(FILE *fp
)
2475 gchar buf
[BUFFSIZE
];
2478 str
= g_string_new(NULL
);
2480 /* output header part */
2481 while (claws_fgets(buf
, sizeof(buf
), fp
) != NULL
) {
2483 if (!g_ascii_strncasecmp(buf
, "Bcc:", 4)) {
2490 else if (next
!= ' ' && next
!= '\t') {
2494 if (claws_fgets(buf
, sizeof(buf
), fp
) == NULL
)
2498 g_string_append(str
, buf
);
2499 g_string_append(str
, "\r\n");
2505 /* output body part */
2506 while (claws_fgets(buf
, sizeof(buf
), fp
) != NULL
) {
2509 g_string_append_c(str
, '.');
2510 g_string_append(str
, buf
);
2511 g_string_append(str
, "\r\n");
2514 return g_string_free(str
, FALSE
);
2518 * Create a new boundary in a way that it is very unlikely that this
2519 * will occur in the following text. It would be easy to ensure
2520 * uniqueness if everything is either quoted-printable or base64
2521 * encoded (note that conversion is allowed), but because MIME bodies
2522 * may be nested, it may happen that the same boundary has already
2525 * boundary := 0*69<bchars> bcharsnospace
2526 * bchars := bcharsnospace / " "
2527 * bcharsnospace := DIGIT / ALPHA / "'" / "(" / ")" /
2528 * "+" / "_" / "," / "-" / "." /
2529 * "/" / ":" / "=" / "?"
2531 * some special characters removed because of buggy MTAs
2534 gchar
*generate_mime_boundary(const gchar
*prefix
)
2536 static gchar tbl
[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
2537 "abcdefghijklmnopqrstuvwxyz"
2542 for (i
= 0; i
< sizeof(buf_uniq
) - 1; i
++)
2543 buf_uniq
[i
] = tbl
[g_random_int_range(0, sizeof(tbl
) - 1)];
2546 return g_strdup_printf("%s_/%s", prefix
? prefix
: "MP",
2550 char *fgets_crlf(char *buf
, int size
, FILE *stream
)
2552 gboolean is_cr
= FALSE
;
2553 gboolean last_was_cr
= FALSE
;
2558 while (--size
> 0 && (c
= getc(stream
)) != EOF
)
2561 is_cr
= (c
== '\r');
2571 last_was_cr
= is_cr
;
2573 if (c
== EOF
&& cs
== buf
)
2581 static gint
execute_async(gchar
*const argv
[], const gchar
*working_directory
)
2583 cm_return_val_if_fail(argv
!= NULL
&& argv
[0] != NULL
, -1);
2585 if (g_spawn_async(working_directory
, (gchar
**)argv
, NULL
, G_SPAWN_SEARCH_PATH
,
2586 NULL
, NULL
, NULL
, FALSE
) == FALSE
) {
2587 g_warning("couldn't execute command: %s", argv
[0]);
2594 static gint
execute_sync(gchar
*const argv
[], const gchar
*working_directory
)
2598 cm_return_val_if_fail(argv
!= NULL
&& argv
[0] != NULL
, -1);
2601 if (g_spawn_sync(working_directory
, (gchar
**)argv
, NULL
, G_SPAWN_SEARCH_PATH
,
2602 NULL
, NULL
, NULL
, NULL
, &status
, NULL
) == FALSE
) {
2603 g_warning("couldn't execute command: %s", argv
[0]);
2607 if (WIFEXITED(status
))
2608 return WEXITSTATUS(status
);
2612 if (g_spawn_sync(working_directory
, (gchar
**)argv
, NULL
,
2613 G_SPAWN_SEARCH_PATH
|
2614 G_SPAWN_CHILD_INHERITS_STDIN
|
2615 G_SPAWN_LEAVE_DESCRIPTORS_OPEN
,
2616 NULL
, NULL
, NULL
, NULL
, &status
, NULL
) == FALSE
) {
2617 g_warning("couldn't execute command: %s", argv
[0]);
2625 gint
execute_command_line(const gchar
*cmdline
, gboolean async
,
2626 const gchar
*working_directory
)
2631 cm_return_val_if_fail(cmdline
!= NULL
, -1);
2633 debug_print("execute_command_line(): executing: %s\n", cmdline
);
2635 argv
= strsplit_with_quote(cmdline
, " ", 0);
2638 ret
= execute_async(argv
, working_directory
);
2640 ret
= execute_sync(argv
, working_directory
);
2647 gchar
*get_command_output(const gchar
*cmdline
)
2649 gchar
*child_stdout
;
2652 cm_return_val_if_fail(cmdline
!= NULL
, NULL
);
2654 debug_print("get_command_output(): executing: %s\n", cmdline
);
2656 if (g_spawn_command_line_sync(cmdline
, &child_stdout
, NULL
, &status
,
2658 g_warning("couldn't execute command: %s", cmdline
);
2662 return child_stdout
;
2665 FILE *get_command_output_stream(const char* cmdline
)
2669 gchar
**argv
= NULL
;
2672 cm_return_val_if_fail(cmdline
!= NULL
, NULL
);
2674 debug_print("get_command_output_stream(): executing: %s\n", cmdline
);
2676 /* turn the command-line string into an array */
2677 if (!g_shell_parse_argv(cmdline
, NULL
, &argv
, &err
)) {
2678 g_warning("could not parse command line from '%s': %s", cmdline
, err
->message
);
2683 if (!g_spawn_async_with_pipes(NULL
, argv
, NULL
, G_SPAWN_SEARCH_PATH
,
2684 NULL
, NULL
, &pid
, NULL
, &fd
, NULL
, &err
)
2687 g_warning("could not spawn '%s': %s", cmdline
, err
->message
);
2694 return fdopen(fd
, "r");
2698 static gint
is_unchanged_uri_char(char c
)
2709 static void encode_uri(gchar
*encoded_uri
, gint bufsize
, const gchar
*uri
)
2715 for(i
= 0; i
< strlen(uri
) ; i
++) {
2716 if (is_unchanged_uri_char(uri
[i
])) {
2717 if (k
+ 2 >= bufsize
)
2719 encoded_uri
[k
++] = uri
[i
];
2722 char * hexa
= "0123456789ABCDEF";
2724 if (k
+ 4 >= bufsize
)
2726 encoded_uri
[k
++] = '%';
2727 encoded_uri
[k
++] = hexa
[uri
[i
] / 16];
2728 encoded_uri
[k
++] = hexa
[uri
[i
] % 16];
2735 gint
open_uri(const gchar
*uri
, const gchar
*cmdline
)
2739 gchar buf
[BUFFSIZE
];
2741 gchar encoded_uri
[BUFFSIZE
];
2742 cm_return_val_if_fail(uri
!= NULL
, -1);
2744 /* an option to choose whether to use encode_uri or not ? */
2745 encode_uri(encoded_uri
, BUFFSIZE
, uri
);
2748 (p
= strchr(cmdline
, '%')) && *(p
+ 1) == 's' &&
2749 !strchr(p
+ 2, '%'))
2750 g_snprintf(buf
, sizeof(buf
), cmdline
, encoded_uri
);
2753 g_warning("Open URI command-line is invalid "
2754 "(there must be only one '%%s'): %s",
2756 g_snprintf(buf
, sizeof(buf
), DEFAULT_BROWSER_CMD
, encoded_uri
);
2759 execute_command_line(buf
, TRUE
, NULL
);
2761 ShellExecute(NULL
, "open", uri
, NULL
, NULL
, SW_SHOW
);
2766 gint
open_txt_editor(const gchar
*filepath
, const gchar
*cmdline
)
2768 gchar buf
[BUFFSIZE
];
2771 cm_return_val_if_fail(filepath
!= NULL
, -1);
2774 (p
= strchr(cmdline
, '%')) && *(p
+ 1) == 's' &&
2775 !strchr(p
+ 2, '%'))
2776 g_snprintf(buf
, sizeof(buf
), cmdline
, filepath
);
2779 g_warning("Open Text Editor command-line is invalid "
2780 "(there must be only one '%%s'): %s",
2782 g_snprintf(buf
, sizeof(buf
), DEFAULT_EDITOR_CMD
, filepath
);
2785 execute_command_line(buf
, TRUE
, NULL
);
2790 time_t remote_tzoffset_sec(const gchar
*zone
)
2792 static gchar ustzstr
[] = "PSTPDTMSTMDTCSTCDTESTEDT";
2798 time_t remoteoffset
;
2800 strncpy(zone3
, zone
, 3);
2804 if (sscanf(zone
, "%c%d", &c
, &offset
) == 2 &&
2805 (c
== '+' || c
== '-')) {
2806 remoteoffset
= ((offset
/ 100) * 60 + (offset
% 100)) * 60;
2808 remoteoffset
= -remoteoffset
;
2809 } else if (!strncmp(zone
, "UT" , 2) ||
2810 !strncmp(zone
, "GMT", 3)) {
2812 } else if (strlen(zone3
) == 3) {
2813 for (p
= ustzstr
; *p
!= '\0'; p
+= 3) {
2814 if (!g_ascii_strncasecmp(p
, zone3
, 3)) {
2815 iustz
= ((gint
)(p
- ustzstr
) / 3 + 1) / 2 - 8;
2816 remoteoffset
= iustz
* 3600;
2822 } else if (strlen(zone3
) == 1) {
2824 case 'Z': remoteoffset
= 0; break;
2825 case 'A': remoteoffset
= -1; break;
2826 case 'B': remoteoffset
= -2; break;
2827 case 'C': remoteoffset
= -3; break;
2828 case 'D': remoteoffset
= -4; break;
2829 case 'E': remoteoffset
= -5; break;
2830 case 'F': remoteoffset
= -6; break;
2831 case 'G': remoteoffset
= -7; break;
2832 case 'H': remoteoffset
= -8; break;
2833 case 'I': remoteoffset
= -9; break;
2834 case 'K': remoteoffset
= -10; break; /* J is not used */
2835 case 'L': remoteoffset
= -11; break;
2836 case 'M': remoteoffset
= -12; break;
2837 case 'N': remoteoffset
= 1; break;
2838 case 'O': remoteoffset
= 2; break;
2839 case 'P': remoteoffset
= 3; break;
2840 case 'Q': remoteoffset
= 4; break;
2841 case 'R': remoteoffset
= 5; break;
2842 case 'S': remoteoffset
= 6; break;
2843 case 'T': remoteoffset
= 7; break;
2844 case 'U': remoteoffset
= 8; break;
2845 case 'V': remoteoffset
= 9; break;
2846 case 'W': remoteoffset
= 10; break;
2847 case 'X': remoteoffset
= 11; break;
2848 case 'Y': remoteoffset
= 12; break;
2849 default: remoteoffset
= 0; break;
2851 remoteoffset
= remoteoffset
* 3600;
2855 return remoteoffset
;
2858 time_t tzoffset_sec(time_t *now
)
2862 struct tm buf1
, buf2
;
2864 if (now
&& *now
< 0)
2867 gmt
= *gmtime_r(now
, &buf1
);
2868 lt
= localtime_r(now
, &buf2
);
2870 off
= (lt
->tm_hour
- gmt
.tm_hour
) * 60 + lt
->tm_min
- gmt
.tm_min
;
2872 if (lt
->tm_year
< gmt
.tm_year
)
2874 else if (lt
->tm_year
> gmt
.tm_year
)
2876 else if (lt
->tm_yday
< gmt
.tm_yday
)
2878 else if (lt
->tm_yday
> gmt
.tm_yday
)
2881 if (off
>= 24 * 60) /* should be impossible */
2882 off
= 23 * 60 + 59; /* if not, insert silly value */
2883 if (off
<= -24 * 60)
2884 off
= -(23 * 60 + 59);
2889 /* calculate timezone offset */
2890 gchar
*tzoffset(time_t *now
)
2892 static gchar offset_string
[6];
2896 struct tm buf1
, buf2
;
2898 if (now
&& *now
< 0)
2901 gmt
= *gmtime_r(now
, &buf1
);
2902 lt
= localtime_r(now
, &buf2
);
2904 off
= (lt
->tm_hour
- gmt
.tm_hour
) * 60 + lt
->tm_min
- gmt
.tm_min
;
2906 if (lt
->tm_year
< gmt
.tm_year
)
2908 else if (lt
->tm_year
> gmt
.tm_year
)
2910 else if (lt
->tm_yday
< gmt
.tm_yday
)
2912 else if (lt
->tm_yday
> gmt
.tm_yday
)
2920 if (off
>= 24 * 60) /* should be impossible */
2921 off
= 23 * 60 + 59; /* if not, insert silly value */
2923 sprintf(offset_string
, "%c%02d%02d", sign
, off
/ 60, off
% 60);
2925 return offset_string
;
2928 static void _get_rfc822_date(gchar
*buf
, gint len
, gboolean hidetz
)
2932 gchar day
[4], mon
[4];
2933 gint dd
, hh
, mm
, ss
, yyyy
;
2935 gchar buf2
[RFC822_DATE_BUFFSIZE
];
2939 lt
= gmtime_r(&t
, &buf1
);
2941 lt
= localtime_r(&t
, &buf1
);
2943 if (sscanf(asctime_r(lt
, buf2
), "%3s %3s %d %d:%d:%d %d\n",
2944 day
, mon
, &dd
, &hh
, &mm
, &ss
, &yyyy
) != 7)
2945 g_warning("failed reading date/time");
2947 g_snprintf(buf
, len
, "%s, %d %s %d %02d:%02d:%02d %s",
2948 day
, dd
, mon
, yyyy
, hh
, mm
, ss
, (hidetz
? "-0000": tzoffset(&t
)));
2951 void get_rfc822_date(gchar
*buf
, gint len
)
2953 _get_rfc822_date(buf
, len
, FALSE
);
2956 void get_rfc822_date_hide_tz(gchar
*buf
, gint len
)
2958 _get_rfc822_date(buf
, len
, TRUE
);
2961 void debug_set_mode(gboolean mode
)
2966 gboolean
debug_get_mode(void)
2972 void debug_print_real(const char *file
, int line
, const gchar
*format
, ...)
2975 gchar buf
[BUFFSIZE
];
2978 if (!debug_mode
) return;
2980 prefix_len
= g_snprintf(buf
, sizeof(buf
), "%s:%d:", debug_srcname(file
), line
);
2982 va_start(args
, format
);
2983 g_vsnprintf(buf
+ prefix_len
, sizeof(buf
) - prefix_len
, format
, args
);
2989 void debug_print_real(const gchar
*format
, ...)
2992 gchar buf
[BUFFSIZE
];
2994 if (!debug_mode
) return;
2996 va_start(args
, format
);
2997 g_vsnprintf(buf
, sizeof(buf
), format
, args
);
3005 const char * debug_srcname(const char *file
)
3007 const char *s
= strrchr (file
, '/');
3012 void * subject_table_lookup(GHashTable
*subject_table
, gchar
* subject
)
3014 if (subject
== NULL
)
3017 subject
+= subject_get_prefix_length(subject
);
3019 return g_hash_table_lookup(subject_table
, subject
);
3022 void subject_table_insert(GHashTable
*subject_table
, gchar
* subject
,
3025 if (subject
== NULL
|| *subject
== 0)
3027 subject
+= subject_get_prefix_length(subject
);
3028 g_hash_table_insert(subject_table
, subject
, data
);
3031 void subject_table_remove(GHashTable
*subject_table
, gchar
* subject
)
3033 if (subject
== NULL
)
3036 subject
+= subject_get_prefix_length(subject
);
3037 g_hash_table_remove(subject_table
, subject
);
3040 static regex_t u_regex
;
3041 static gboolean u_init_
;
3043 void utils_free_regex(void)
3052 *\brief Check if a string is prefixed with known (combinations)
3053 * of prefixes. The function assumes that each prefix
3054 * is terminated by zero or exactly _one_ space.
3056 *\param str String to check for a prefixes
3058 *\return int Number of chars in the prefix that should be skipped
3059 * for a "clean" subject line. If no prefix was found, 0
3062 int subject_get_prefix_length(const gchar
*subject
)
3064 /*!< Array with allowable reply prefixes regexps. */
3065 static const gchar
* const prefixes
[] = {
3066 "Re\\:", /* "Re:" */
3067 "Re\\[[1-9][0-9]*\\]\\:", /* "Re[XXX]:" (non-conforming news mail clients) */
3068 "Antw\\:", /* "Antw:" (Dutch / German Outlook) */
3069 "Aw\\:", /* "Aw:" (German) */
3070 "Antwort\\:", /* "Antwort:" (German Lotus Notes) */
3071 "Res\\:", /* "Res:" (Spanish/Brazilian Outlook) */
3072 "Fw\\:", /* "Fw:" Forward */
3073 "Fwd\\:", /* "Fwd:" Forward */
3074 "Enc\\:", /* "Enc:" Forward (Brazilian Outlook) */
3075 "Odp\\:", /* "Odp:" Re (Polish Outlook) */
3076 "Rif\\:", /* "Rif:" (Italian Outlook) */
3077 "Sv\\:", /* "Sv" (Norwegian) */
3078 "Vs\\:", /* "Vs" (Norwegian) */
3079 "Ad\\:", /* "Ad" (Norwegian) */
3080 "\347\255\224\345\244\215\\:", /* "Re" (Chinese, UTF-8) */
3081 "R\303\251f\\. \\:", /* "R�f. :" (French Lotus Notes) */
3082 "Re \\:", /* "Re :" (French Yahoo Mail) */
3085 const int PREFIXES
= sizeof prefixes
/ sizeof prefixes
[0];
3089 if (!subject
) return 0;
3090 if (!*subject
) return 0;
3093 GString
*s
= g_string_new("");
3095 for (n
= 0; n
< PREFIXES
; n
++)
3096 /* Terminate each prefix regexpression by a
3097 * "\ ?" (zero or ONE space), and OR them */
3098 g_string_append_printf(s
, "(%s\\ ?)%s",
3103 g_string_prepend(s
, "(");
3104 g_string_append(s
, ")+"); /* match at least once */
3105 g_string_prepend(s
, "^\\ *"); /* from beginning of line */
3108 /* We now have something like "^\ *((PREFIX1\ ?)|(PREFIX2\ ?))+"
3109 * TODO: Should this be "^\ *(((PREFIX1)|(PREFIX2))\ ?)+" ??? */
3110 if (regcomp(&u_regex
, s
->str
, REG_EXTENDED
| REG_ICASE
)) {
3111 debug_print("Error compiling regexp %s\n", s
->str
);
3112 g_string_free(s
, TRUE
);
3116 g_string_free(s
, TRUE
);
3120 if (!regexec(&u_regex
, subject
, 1, &pos
, 0) && pos
.rm_so
!= -1)
3126 static guint
g_stricase_hash(gconstpointer gptr
)
3128 guint hash_result
= 0;
3131 for (str
= gptr
; str
&& *str
; str
++) {
3132 hash_result
+= toupper(*str
);
3138 static gint
g_stricase_equal(gconstpointer gptr1
, gconstpointer gptr2
)
3140 const char *str1
= gptr1
;
3141 const char *str2
= gptr2
;
3143 return !strcasecmp(str1
, str2
);
3146 gint
g_int_compare(gconstpointer a
, gconstpointer b
)
3148 return GPOINTER_TO_INT(a
) - GPOINTER_TO_INT(b
);
3152 quote_cmd_argument()
3154 return a quoted string safely usable in argument of a command.
3156 code is extracted and adapted from etPan! project -- DINH V. Ho�.
3159 gint
quote_cmd_argument(gchar
* result
, guint size
,
3169 for(p
= path
; * p
!= '\0' ; p
++) {
3171 if (isalnum((guchar
)*p
) || (* p
== '/')) {
3172 if (remaining
> 0) {
3178 result
[size
- 1] = '\0';
3183 if (remaining
>= 2) {
3191 result
[size
- 1] = '\0';
3196 if (remaining
> 0) {
3200 result
[size
- 1] = '\0';
3214 static void g_node_map_recursive(GNode
*node
, gpointer data
)
3216 GNodeMapData
*mapdata
= (GNodeMapData
*) data
;
3218 GNodeMapData newmapdata
;
3221 newdata
= mapdata
->func(node
->data
, mapdata
->data
);
3222 if (newdata
!= NULL
) {
3223 newnode
= g_node_new(newdata
);
3224 g_node_append(mapdata
->parent
, newnode
);
3226 newmapdata
.parent
= newnode
;
3227 newmapdata
.func
= mapdata
->func
;
3228 newmapdata
.data
= mapdata
->data
;
3230 g_node_children_foreach(node
, G_TRAVERSE_ALL
, g_node_map_recursive
, &newmapdata
);
3234 GNode
*g_node_map(GNode
*node
, GNodeMapFunc func
, gpointer data
)
3237 GNodeMapData mapdata
;
3239 cm_return_val_if_fail(node
!= NULL
, NULL
);
3240 cm_return_val_if_fail(func
!= NULL
, NULL
);
3242 root
= g_node_new(func(node
->data
, data
));
3244 mapdata
.parent
= root
;
3245 mapdata
.func
= func
;
3246 mapdata
.data
= data
;
3248 g_node_children_foreach(node
, G_TRAVERSE_ALL
, g_node_map_recursive
, &mapdata
);
3253 #define HEX_TO_INT(val, hex) \
3257 if ('0' <= c && c <= '9') { \
3259 } else if ('a' <= c && c <= 'f') { \
3260 val = c - 'a' + 10; \
3261 } else if ('A' <= c && c <= 'F') { \
3262 val = c - 'A' + 10; \
3268 gboolean
get_hex_value(guchar
*out
, gchar c1
, gchar c2
)
3275 if (hi
== -1 || lo
== -1)
3278 *out
= (hi
<< 4) + lo
;
3282 #define INT_TO_HEX(hex, val) \
3285 hex = '0' + (val); \
3287 hex = 'A' + (val) - 10; \
3290 void get_hex_str(gchar
*out
, guchar ch
)
3294 INT_TO_HEX(hex
, ch
>> 4);
3296 INT_TO_HEX(hex
, ch
& 0x0f);
3302 #define G_PRINT_REF 1 == 1 ? (void) 0 : (void)
3304 #define G_PRINT_REF g_print
3308 *\brief Register ref counted pointer. It is based on GBoxed, so should
3309 * work with anything that uses the GType system. The semantics
3310 * are similar to a C++ auto pointer, with the exception that
3311 * C doesn't have automatic closure (calling destructors) when
3312 * exiting a block scope.
3313 * Use the \ref G_TYPE_AUTO_POINTER macro instead of calling this
3314 * function directly.
3316 *\return GType A GType type.
3318 GType
g_auto_pointer_register(void)
3320 static GType auto_pointer_type
;
3321 if (!auto_pointer_type
)
3323 g_boxed_type_register_static
3324 ("G_TYPE_AUTO_POINTER",
3325 (GBoxedCopyFunc
) g_auto_pointer_copy
,
3326 (GBoxedFreeFunc
) g_auto_pointer_free
);
3327 return auto_pointer_type
;
3331 *\brief Structure with g_new() allocated pointer guarded by the
3334 typedef struct AutoPointerRef
{
3335 void (*free
) (gpointer
);
3341 *\brief The auto pointer opaque structure that references the
3342 * pointer guard block.
3344 typedef struct AutoPointer
{
3345 AutoPointerRef
*ref
;
3346 gpointer ptr
; /*!< access to protected pointer */
3350 *\brief Creates an auto pointer for a g_new()ed pointer. Example:
3354 * ... tell gtk_list_store it should use a G_TYPE_AUTO_POINTER
3355 * ... when assigning, copying and freeing storage elements
3357 * gtk_list_store_new(N_S_COLUMNS,
3358 * G_TYPE_AUTO_POINTER,
3362 * Template *precious_data = g_new0(Template, 1);
3363 * g_pointer protect = g_auto_pointer_new(precious_data);
3365 * gtk_list_store_set(container, &iter,
3369 * ... the gtk_list_store has copied the pointer and
3370 * ... incremented its reference count, we should free
3371 * ... the auto pointer (in C++ a destructor would do
3372 * ... this for us when leaving block scope)
3374 * g_auto_pointer_free(protect);
3376 * ... gtk_list_store_set() now manages the data. When
3377 * ... *explicitly* requesting a pointer from the list
3378 * ... store, don't forget you get a copy that should be
3379 * ... freed with g_auto_pointer_free() eventually.
3383 *\param pointer Pointer to be guarded.
3385 *\return GAuto * Pointer that should be used in containers with
3388 GAuto
*g_auto_pointer_new(gpointer p
)
3390 AutoPointerRef
*ref
;
3396 ref
= g_new0(AutoPointerRef
, 1);
3397 ptr
= g_new0(AutoPointer
, 1);
3407 G_PRINT_REF ("XXXX ALLOC(%lx)\n", p
);
3413 *\brief Allocate an autopointer using the passed \a free function to
3414 * free the guarded pointer
3416 GAuto
*g_auto_pointer_new_with_free(gpointer p
, GFreeFunc free_
)
3423 aptr
= g_auto_pointer_new(p
);
3424 aptr
->ref
->free
= free_
;
3428 gpointer
g_auto_pointer_get_ptr(GAuto
*auto_ptr
)
3430 if (auto_ptr
== NULL
)
3432 return ((AutoPointer
*) auto_ptr
)->ptr
;
3436 *\brief Copies an auto pointer by. It's mostly not necessary
3437 * to call this function directly, unless you copy/assign
3438 * the guarded pointer.
3440 *\param auto_ptr Auto pointer returned by previous call to
3441 * g_auto_pointer_new_XXX()
3443 *\return gpointer An auto pointer
3445 GAuto
*g_auto_pointer_copy(GAuto
*auto_ptr
)
3448 AutoPointerRef
*ref
;
3451 if (auto_ptr
== NULL
)
3456 newp
= g_new0(AutoPointer
, 1);
3459 newp
->ptr
= ref
->pointer
;
3463 G_PRINT_REF ("XXXX COPY(%lx) -- REF (%d)\n", ref
->pointer
, ref
->cnt
);
3469 *\brief Free an auto pointer
3471 void g_auto_pointer_free(GAuto
*auto_ptr
)
3474 AutoPointerRef
*ref
;
3476 if (auto_ptr
== NULL
)
3482 if (--(ref
->cnt
) == 0) {
3484 G_PRINT_REF ("XXXX FREE(%lx) -- REF (%d)\n", ref
->pointer
, ref
->cnt
);
3486 ref
->free(ref
->pointer
);
3491 G_PRINT_REF ("XXXX DEREF(%lx) -- REF (%d)\n", ref
->pointer
, ref
->cnt
);
3496 /* get_uri_part() - retrieves a URI starting from scanpos.
3497 Returns TRUE if successful */
3498 gboolean
get_uri_part(const gchar
*start
, const gchar
*scanpos
,
3499 const gchar
**bp
, const gchar
**ep
, gboolean hdr
)
3502 gint parenthese_cnt
= 0;
3504 cm_return_val_if_fail(start
!= NULL
, FALSE
);
3505 cm_return_val_if_fail(scanpos
!= NULL
, FALSE
);
3506 cm_return_val_if_fail(bp
!= NULL
, FALSE
);
3507 cm_return_val_if_fail(ep
!= NULL
, FALSE
);
3511 /* find end point of URI */
3512 for (ep_
= scanpos
; *ep_
!= '\0'; ep_
= g_utf8_next_char(ep_
)) {
3513 gunichar u
= g_utf8_get_char_validated(ep_
, -1);
3514 if (!g_unichar_isgraph(u
) ||
3515 u
== (gunichar
)-1 ||
3516 strchr("[]{}<>\"", *ep_
)) {
3518 } else if (strchr("(", *ep_
)) {
3520 } else if (strchr(")", *ep_
)) {
3521 if (parenthese_cnt
> 0)
3528 /* no punctuation at end of string */
3530 /* FIXME: this stripping of trailing punctuations may bite with other URIs.
3531 * should pass some URI type to this function and decide on that whether
3532 * to perform punctuation stripping */
3534 #define IS_REAL_PUNCT(ch) (g_ascii_ispunct(ch) && !strchr("$/?=-_~)", ch))
3536 for (; ep_
- 1 > scanpos
+ 1 &&
3537 IS_REAL_PUNCT(*(ep_
- 1));
3541 #undef IS_REAL_PUNCT
3548 gchar
*make_uri_string(const gchar
*bp
, const gchar
*ep
)
3550 while (bp
&& *bp
&& g_ascii_isspace(*bp
))
3552 return g_strndup(bp
, ep
- bp
);
3555 /* valid mail address characters */
3556 #define IS_RFC822_CHAR(ch) \
3560 !g_ascii_isspace(ch) && \
3561 !strchr("(),;<>\"", (ch)))
3563 /* alphabet and number within 7bit ASCII */
3564 #define IS_ASCII_ALNUM(ch) (IS_ASCII(ch) && g_ascii_isalnum(ch))
3565 #define IS_QUOTE(ch) ((ch) == '\'' || (ch) == '"')
3567 static GHashTable
*create_domain_tab(void)
3570 GHashTable
*htab
= g_hash_table_new(g_stricase_hash
, g_stricase_equal
);
3572 cm_return_val_if_fail(htab
, NULL
);
3573 for (n
= 0; n
< sizeof toplvl_domains
/ sizeof toplvl_domains
[0]; n
++)
3574 g_hash_table_insert(htab
, (gpointer
) toplvl_domains
[n
], (gpointer
) toplvl_domains
[n
]);
3578 static gboolean
is_toplvl_domain(GHashTable
*tab
, const gchar
*first
, const gchar
*last
)
3580 gchar buf
[BUFFSIZE
+ 1];
3581 const gchar
*m
= buf
+ BUFFSIZE
+ 1;
3584 if (last
- first
> BUFFSIZE
|| first
> last
)
3587 for (p
= buf
; p
< m
&& first
< last
; *p
++ = *first
++)
3591 return g_hash_table_lookup(tab
, buf
) != NULL
;
3594 /* get_email_part() - retrieves an email address. Returns TRUE if successful */
3595 gboolean
get_email_part(const gchar
*start
, const gchar
*scanpos
,
3596 const gchar
**bp
, const gchar
**ep
, gboolean hdr
)
3598 /* more complex than the uri part because we need to scan back and forward starting from
3599 * the scan position. */
3600 gboolean result
= FALSE
;
3601 const gchar
*bp_
= NULL
;
3602 const gchar
*ep_
= NULL
;
3603 static GHashTable
*dom_tab
;
3604 const gchar
*last_dot
= NULL
;
3605 const gchar
*prelast_dot
= NULL
;
3606 const gchar
*last_tld_char
= NULL
;
3608 /* the informative part of the email address (describing the name
3609 * of the email address owner) may contain quoted parts. the
3610 * closure stack stores the last encountered quotes. */
3611 gchar closure_stack
[128];
3612 gchar
*ptr
= closure_stack
;
3614 cm_return_val_if_fail(start
!= NULL
, FALSE
);
3615 cm_return_val_if_fail(scanpos
!= NULL
, FALSE
);
3616 cm_return_val_if_fail(bp
!= NULL
, FALSE
);
3617 cm_return_val_if_fail(ep
!= NULL
, FALSE
);
3620 const gchar
*start_quote
= NULL
;
3621 const gchar
*end_quote
= NULL
;
3623 /* go to the real start */
3624 if (start
[0] == ',')
3626 if (start
[0] == ';')
3628 while (start
[0] == '\n' || start
[0] == '\r')
3630 while (start
[0] == ' ' || start
[0] == '\t')
3635 /* check if there are quotes (to skip , in them) */
3636 if (*start
== '"') {
3637 start_quote
= start
;
3639 end_quote
= strstr(start
, "\"");
3645 /* skip anything between quotes */
3646 if (start_quote
&& end_quote
) {
3651 /* find end (either , or ; or end of line) */
3652 if (strstr(start
, ",") && strstr(start
, ";"))
3653 *ep
= strstr(start
,",") < strstr(start
, ";")
3654 ? strstr(start
, ",") : strstr(start
, ";");
3655 else if (strstr(start
, ","))
3656 *ep
= strstr(start
, ",");
3657 else if (strstr(start
, ";"))
3658 *ep
= strstr(start
, ";");
3660 *ep
= start
+strlen(start
);
3662 /* go back to real start */
3663 if (start_quote
&& end_quote
) {
3664 start
= start_quote
;
3667 /* check there's still an @ in that, or search
3668 * further if possible */
3669 if (strstr(start
, "@") && strstr(start
, "@") < *ep
)
3671 else if (*ep
< start
+strlen(start
)) {
3674 } else if (start_quote
&& strstr(start
, "\"") && strstr(start
, "\"") < *ep
) {
3682 dom_tab
= create_domain_tab();
3683 cm_return_val_if_fail(dom_tab
, FALSE
);
3685 /* scan start of address */
3686 for (bp_
= scanpos
- 1;
3687 bp_
>= start
&& IS_RFC822_CHAR(*(const guchar
*)bp_
); bp_
--)
3690 /* TODO: should start with an alnum? */
3692 for (; bp_
< scanpos
&& !IS_ASCII_ALNUM(*(const guchar
*)bp_
); bp_
++)
3695 if (bp_
!= scanpos
) {
3696 /* scan end of address */
3697 for (ep_
= scanpos
+ 1;
3698 *ep_
&& IS_RFC822_CHAR(*(const guchar
*)ep_
); ep_
++)
3700 prelast_dot
= last_dot
;
3702 if (*(last_dot
+ 1) == '.') {
3703 if (prelast_dot
== NULL
)
3705 last_dot
= prelast_dot
;
3710 /* TODO: really should terminate with an alnum? */
3711 for (; ep_
> scanpos
&& !IS_ASCII_ALNUM(*(const guchar
*)ep_
);
3716 if (last_dot
== NULL
)
3718 if (last_dot
>= ep_
)
3719 last_dot
= prelast_dot
;
3720 if (last_dot
== NULL
|| (scanpos
+ 1 >= last_dot
))
3724 for (last_tld_char
= last_dot
; last_tld_char
< ep_
; last_tld_char
++)
3725 if (*last_tld_char
== '?')
3728 if (is_toplvl_domain(dom_tab
, last_dot
, last_tld_char
))
3735 if (!result
) return FALSE
;
3737 if (*ep_
&& bp_
!= start
&& *(bp_
- 1) == '"' && *(ep_
) == '"'
3738 && *(ep_
+ 1) == ' ' && *(ep_
+ 2) == '<'
3739 && IS_RFC822_CHAR(*(ep_
+ 3))) {
3740 /* this informative part with an @ in it is
3741 * followed by the email address */
3744 /* go to matching '>' (or next non-rfc822 char, like \n) */
3745 for (; *ep_
!= '>' && *ep_
!= '\0' && IS_RFC822_CHAR(*ep_
); ep_
++)
3748 /* include the bracket */
3749 if (*ep_
== '>') ep_
++;
3751 /* include the leading quote */
3759 /* skip if it's between quotes "'alfons@proteus.demon.nl'" <alfons@proteus.demon.nl> */
3760 if (bp_
- 1 > start
&& IS_QUOTE(*(bp_
- 1)) && IS_QUOTE(*ep_
))
3763 /* see if this is <bracketed>; in this case we also scan for the informative part. */
3764 if (bp_
- 1 <= start
|| *(bp_
- 1) != '<' || *ep_
!= '>')
3767 #define FULL_STACK() ((size_t) (ptr - closure_stack) >= sizeof closure_stack)
3768 #define IN_STACK() (ptr > closure_stack)
3769 /* has underrun check */
3770 #define POP_STACK() if(IN_STACK()) --ptr
3771 /* has overrun check */
3772 #define PUSH_STACK(c) if(!FULL_STACK()) *ptr++ = (c); else return TRUE
3773 /* has underrun check */
3774 #define PEEK_STACK() (IN_STACK() ? *(ptr - 1) : 0)
3778 /* scan for the informative part. */
3779 for (bp_
-= 2; bp_
>= start
; bp_
--) {
3780 /* if closure on the stack keep scanning */
3781 if (PEEK_STACK() == *bp_
) {
3785 if (!IN_STACK() && (*bp_
== '\'' || *bp_
== '"')) {
3790 /* if nothing in the closure stack, do the special conditions
3791 * the following if..else expression simply checks whether
3792 * a token is acceptable. if not acceptable, the clause
3793 * should terminate the loop with a 'break' */
3794 if (!PEEK_STACK()) {
3796 && (((bp_
- 1) >= start
) && isalnum(*(bp_
- 1)))
3797 && (((bp_
+ 1) < ep_
) && isalnum(*(bp_
+ 1)))) {
3798 /* hyphens are allowed, but only in
3800 } else if (strchr(" \"'", *bp_
)) {
3801 /* but anything not being a punctiation
3804 break; /* anything else is rejected */
3811 /* scan forward (should start with an alnum) */
3812 for (; *bp_
!= '<' && isspace(*bp_
) && *bp_
!= '"'; bp_
++)
3828 #undef IS_ASCII_ALNUM
3829 #undef IS_RFC822_CHAR
3831 gchar
*make_email_string(const gchar
*bp
, const gchar
*ep
)
3833 /* returns a mailto: URI; mailto: is also used to detect the
3834 * uri type later on in the button_pressed signal handler */
3839 tmp
= g_strndup(bp
, ep
- bp
);
3841 /* If there is a colon in the username part of the address,
3842 * we're dealing with an URI for some other protocol - do
3843 * not prefix with mailto: in such case. */
3844 colon
= strchr(tmp
, ':');
3845 at
= strchr(tmp
, '@');
3846 if (colon
!= NULL
&& at
!= NULL
&& colon
< at
) {
3849 result
= g_strconcat("mailto:", tmp
, NULL
);
3856 gchar
*make_http_string(const gchar
*bp
, const gchar
*ep
)
3858 /* returns an http: URI; */
3862 while (bp
&& *bp
&& g_ascii_isspace(*bp
))
3864 tmp
= g_strndup(bp
, ep
- bp
);
3865 result
= g_strconcat("http://", tmp
, NULL
);
3871 static gchar
*mailcap_get_command_in_file(const gchar
*path
, const gchar
*type
, const gchar
*file_to_open
)
3873 FILE *fp
= claws_fopen(path
, "rb");
3874 gchar buf
[BUFFSIZE
];
3875 gchar
*result
= NULL
;
3878 while (claws_fgets(buf
, sizeof (buf
), fp
) != NULL
) {
3879 gchar
**parts
= g_strsplit(buf
, ";", 3);
3880 gchar
*trimmed
= parts
[0];
3881 while (trimmed
[0] == ' ' || trimmed
[0] == '\t')
3883 while (trimmed
[strlen(trimmed
)-1] == ' ' || trimmed
[strlen(trimmed
)-1] == '\t')
3884 trimmed
[strlen(trimmed
)-1] = '\0';
3886 if (!strcmp(trimmed
, type
)) {
3887 gboolean needsterminal
= FALSE
;
3888 if (parts
[2] && strstr(parts
[2], "needsterminal")) {
3889 needsterminal
= TRUE
;
3891 if (parts
[2] && strstr(parts
[2], "test=")) {
3892 gchar
*orig_testcmd
= g_strdup(strstr(parts
[2], "test=")+5);
3893 gchar
*testcmd
= orig_testcmd
;
3894 if (strstr(testcmd
,";"))
3895 *(strstr(testcmd
,";")) = '\0';
3896 while (testcmd
[0] == ' ' || testcmd
[0] == '\t')
3898 while (testcmd
[strlen(testcmd
)-1] == '\n')
3899 testcmd
[strlen(testcmd
)-1] = '\0';
3900 while (testcmd
[strlen(testcmd
)-1] == '\r')
3901 testcmd
[strlen(testcmd
)-1] = '\0';
3902 while (testcmd
[strlen(testcmd
)-1] == ' ' || testcmd
[strlen(testcmd
)-1] == '\t')
3903 testcmd
[strlen(testcmd
)-1] = '\0';
3905 if (strstr(testcmd
, "%s")) {
3906 gchar
*tmp
= g_strdup_printf(testcmd
, file_to_open
);
3907 gint res
= system(tmp
);
3909 g_free(orig_testcmd
);
3916 gint res
= system(testcmd
);
3917 g_free(orig_testcmd
);
3927 while (trimmed
[0] == ' ' || trimmed
[0] == '\t')
3929 while (trimmed
[strlen(trimmed
)-1] == '\n')
3930 trimmed
[strlen(trimmed
)-1] = '\0';
3931 while (trimmed
[strlen(trimmed
)-1] == '\r')
3932 trimmed
[strlen(trimmed
)-1] = '\0';
3933 while (trimmed
[strlen(trimmed
)-1] == ' ' || trimmed
[strlen(trimmed
)-1] == '\t')
3934 trimmed
[strlen(trimmed
)-1] = '\0';
3935 result
= g_strdup(trimmed
);
3938 if (needsterminal
) {
3939 gchar
*tmp
= g_strdup_printf("xterm -e %s", result
);
3950 gchar
*mailcap_get_command_for_type(const gchar
*type
, const gchar
*file_to_open
)
3952 gchar
*result
= NULL
;
3956 path
= g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S
, ".mailcap", NULL
);
3957 result
= mailcap_get_command_in_file(path
, type
, file_to_open
);
3961 result
= mailcap_get_command_in_file("/etc/mailcap", type
, file_to_open
);
3965 void mailcap_update_default(const gchar
*type
, const gchar
*command
)
3967 gchar
*path
= NULL
, *outpath
= NULL
;
3968 path
= g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S
, ".mailcap", NULL
);
3969 outpath
= g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S
, ".mailcap.new", NULL
);
3970 FILE *fp
= claws_fopen(path
, "rb");
3972 gchar buf
[BUFFSIZE
];
3973 gboolean err
= FALSE
;
3976 fp
= claws_fopen(path
, "a");
3978 g_warning("failed to create file %s", path
);
3983 fp
= g_freopen(path
, "rb", fp
);
3985 g_warning("failed to reopen file %s", path
);
3992 outfp
= claws_fopen(outpath
, "wb");
3994 g_warning("failed to create file %s", outpath
);
4000 while (fp
&& claws_fgets(buf
, sizeof (buf
), fp
) != NULL
) {
4001 gchar
**parts
= g_strsplit(buf
, ";", 3);
4002 gchar
*trimmed
= parts
[0];
4003 while (trimmed
[0] == ' ')
4005 while (trimmed
[strlen(trimmed
)-1] == ' ')
4006 trimmed
[strlen(trimmed
)-1] = '\0';
4008 if (!strcmp(trimmed
, type
)) {
4013 if(claws_fputs(buf
, outfp
) == EOF
) {
4021 if (fprintf(outfp
, "%s; %s\n", type
, command
) < 0)
4027 if (claws_safe_fclose(outfp
) == EOF
)
4031 g_rename(outpath
, path
);
4037 /* crude test to see if a file is an email. */
4038 gboolean
file_is_email (const gchar
*filename
)
4043 if (filename
== NULL
)
4045 if ((fp
= claws_fopen(filename
, "rb")) == NULL
)
4048 && claws_fgets(buffer
, sizeof (buffer
), fp
) != NULL
) {
4049 if (!strncmp(buffer
, "From:", strlen("From:")))
4051 else if (!strncmp(buffer
, "Date:", strlen("Date:")))
4053 else if (!strncmp(buffer
, "Message-ID:", strlen("Message-ID:")))
4055 else if (!strncmp(buffer
, "Subject:", strlen("Subject:")))
4057 else if (!strcmp(buffer
, "\r\n")) {
4058 debug_print("End of headers\n");
4063 return (score
>= 3);
4066 gboolean
sc_g_list_bigger(GList
*list
, gint max
)
4070 while (cur
&& i
<= max
+1) {
4077 gboolean
sc_g_slist_bigger(GSList
*list
, gint max
)
4081 while (cur
&& i
<= max
+1) {
4088 const gchar
*daynames
[] = {NULL
, NULL
, NULL
, NULL
, NULL
, NULL
, NULL
};
4089 const gchar
*monthnames
[] = {NULL
, NULL
, NULL
, NULL
, NULL
, NULL
,
4090 NULL
, NULL
, NULL
, NULL
, NULL
, NULL
};
4091 const gchar
*s_daynames
[] = {NULL
, NULL
, NULL
, NULL
, NULL
, NULL
, NULL
};
4092 const gchar
*s_monthnames
[] = {NULL
, NULL
, NULL
, NULL
, NULL
, NULL
,
4093 NULL
, NULL
, NULL
, NULL
, NULL
, NULL
};
4095 gint daynames_len
[] = {0,0,0,0,0,0,0};
4096 gint monthnames_len
[] = {0,0,0,0,0,0,
4098 gint s_daynames_len
[] = {0,0,0,0,0,0,0};
4099 gint s_monthnames_len
[] = {0,0,0,0,0,0,
4101 const gchar
*s_am_up
= NULL
;
4102 const gchar
*s_pm_up
= NULL
;
4103 const gchar
*s_am_low
= NULL
;
4104 const gchar
*s_pm_low
= NULL
;
4106 gint s_am_up_len
= 0;
4107 gint s_pm_up_len
= 0;
4108 gint s_am_low_len
= 0;
4109 gint s_pm_low_len
= 0;
4111 static gboolean time_names_init_done
= FALSE
;
4113 static void init_time_names(void)
4117 daynames
[0] = C_("Complete day name for use by strftime", "Sunday");
4118 daynames
[1] = C_("Complete day name for use by strftime", "Monday");
4119 daynames
[2] = C_("Complete day name for use by strftime", "Tuesday");
4120 daynames
[3] = C_("Complete day name for use by strftime", "Wednesday");
4121 daynames
[4] = C_("Complete day name for use by strftime", "Thursday");
4122 daynames
[5] = C_("Complete day name for use by strftime", "Friday");
4123 daynames
[6] = C_("Complete day name for use by strftime", "Saturday");
4125 monthnames
[0] = C_("Complete month name for use by strftime", "January");
4126 monthnames
[1] = C_("Complete month name for use by strftime", "February");
4127 monthnames
[2] = C_("Complete month name for use by strftime", "March");
4128 monthnames
[3] = C_("Complete month name for use by strftime", "April");
4129 monthnames
[4] = C_("Complete month name for use by strftime", "May");
4130 monthnames
[5] = C_("Complete month name for use by strftime", "June");
4131 monthnames
[6] = C_("Complete month name for use by strftime", "July");
4132 monthnames
[7] = C_("Complete month name for use by strftime", "August");
4133 monthnames
[8] = C_("Complete month name for use by strftime", "September");
4134 monthnames
[9] = C_("Complete month name for use by strftime", "October");
4135 monthnames
[10] = C_("Complete month name for use by strftime", "November");
4136 monthnames
[11] = C_("Complete month name for use by strftime", "December");
4138 s_daynames
[0] = C_("Abbr. day name for use by strftime", "Sun");
4139 s_daynames
[1] = C_("Abbr. day name for use by strftime", "Mon");
4140 s_daynames
[2] = C_("Abbr. day name for use by strftime", "Tue");
4141 s_daynames
[3] = C_("Abbr. day name for use by strftime", "Wed");
4142 s_daynames
[4] = C_("Abbr. day name for use by strftime", "Thu");
4143 s_daynames
[5] = C_("Abbr. day name for use by strftime", "Fri");
4144 s_daynames
[6] = C_("Abbr. day name for use by strftime", "Sat");
4146 s_monthnames
[0] = C_("Abbr. month name for use by strftime", "Jan");
4147 s_monthnames
[1] = C_("Abbr. month name for use by strftime", "Feb");
4148 s_monthnames
[2] = C_("Abbr. month name for use by strftime", "Mar");
4149 s_monthnames
[3] = C_("Abbr. month name for use by strftime", "Apr");
4150 s_monthnames
[4] = C_("Abbr. month name for use by strftime", "May");
4151 s_monthnames
[5] = C_("Abbr. month name for use by strftime", "Jun");
4152 s_monthnames
[6] = C_("Abbr. month name for use by strftime", "Jul");
4153 s_monthnames
[7] = C_("Abbr. month name for use by strftime", "Aug");
4154 s_monthnames
[8] = C_("Abbr. month name for use by strftime", "Sep");
4155 s_monthnames
[9] = C_("Abbr. month name for use by strftime", "Oct");
4156 s_monthnames
[10] = C_("Abbr. month name for use by strftime", "Nov");
4157 s_monthnames
[11] = C_("Abbr. month name for use by strftime", "Dec");
4159 for (i
= 0; i
< 7; i
++) {
4160 daynames_len
[i
] = strlen(daynames
[i
]);
4161 s_daynames_len
[i
] = strlen(s_daynames
[i
]);
4163 for (i
= 0; i
< 12; i
++) {
4164 monthnames_len
[i
] = strlen(monthnames
[i
]);
4165 s_monthnames_len
[i
] = strlen(s_monthnames
[i
]);
4168 s_am_up
= C_("For use by strftime (morning)", "AM");
4169 s_pm_up
= C_("For use by strftime (afternoon)", "PM");
4170 s_am_low
= C_("For use by strftime (morning, lowercase)", "am");
4171 s_pm_low
= C_("For use by strftime (afternoon, lowercase)", "pm");
4173 s_am_up_len
= strlen(s_am_up
);
4174 s_pm_up_len
= strlen(s_pm_up
);
4175 s_am_low_len
= strlen(s_am_low
);
4176 s_pm_low_len
= strlen(s_pm_low
);
4178 time_names_init_done
= TRUE
;
4181 #define CHECK_SIZE() { \
4182 total_done += len; \
4183 if (total_done >= buflen) { \
4184 buf[buflen-1] = '\0'; \
4189 size_t fast_strftime(gchar
*buf
, gint buflen
, const gchar
*format
, struct tm
*lt
)
4191 gchar
*curpos
= buf
;
4192 gint total_done
= 0;
4193 gchar subbuf
[64], subfmt
[64];
4194 static time_t last_tzset
= (time_t)0;
4196 if (!time_names_init_done
)
4199 if (format
== NULL
|| lt
== NULL
)
4202 if (last_tzset
!= time(NULL
)) {
4204 last_tzset
= time(NULL
);
4207 if (*format
== '%') {
4208 gint len
= 0, tmp
= 0;
4212 len
= 1; CHECK_SIZE();
4216 len
= s_daynames_len
[lt
->tm_wday
]; CHECK_SIZE();
4217 strncpy2(curpos
, s_daynames
[lt
->tm_wday
], buflen
- total_done
);
4220 len
= daynames_len
[lt
->tm_wday
]; CHECK_SIZE();
4221 strncpy2(curpos
, daynames
[lt
->tm_wday
], buflen
- total_done
);
4225 len
= s_monthnames_len
[lt
->tm_mon
]; CHECK_SIZE();
4226 strncpy2(curpos
, s_monthnames
[lt
->tm_mon
], buflen
- total_done
);
4229 len
= monthnames_len
[lt
->tm_mon
]; CHECK_SIZE();
4230 strncpy2(curpos
, monthnames
[lt
->tm_mon
], buflen
- total_done
);
4233 strftime(subbuf
, 64, "%c", lt
);
4234 len
= strlen(subbuf
); CHECK_SIZE();
4235 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4238 total_done
+= 2; CHECK_SIZE();
4239 tmp
= (lt
->tm_year
+ 1900)/100;
4240 *curpos
++ = '0'+(tmp
/ 10);
4241 *curpos
++ = '0'+(tmp
% 10);
4244 total_done
+= 2; CHECK_SIZE();
4245 *curpos
++ = '0'+(lt
->tm_mday
/ 10);
4246 *curpos
++ = '0'+(lt
->tm_mday
% 10);
4249 total_done
+= 8; CHECK_SIZE();
4250 *curpos
++ = '0'+((lt
->tm_mon
+1) / 10);
4251 *curpos
++ = '0'+((lt
->tm_mon
+1) % 10);
4253 *curpos
++ = '0'+(lt
->tm_mday
/ 10);
4254 *curpos
++ = '0'+(lt
->tm_mday
% 10);
4256 tmp
= lt
->tm_year
%100;
4257 *curpos
++ = '0'+(tmp
/ 10);
4258 *curpos
++ = '0'+(tmp
% 10);
4261 len
= 2; CHECK_SIZE();
4262 snprintf(curpos
, buflen
- total_done
, "%2d", lt
->tm_mday
);
4265 len
= 10; CHECK_SIZE();
4266 snprintf(curpos
, buflen
- total_done
, "%4d-%02d-%02d",
4267 lt
->tm_year
+ 1900, lt
->tm_mon
+1, lt
->tm_mday
);
4270 total_done
+= 2; CHECK_SIZE();
4271 *curpos
++ = '0'+(lt
->tm_hour
/ 10);
4272 *curpos
++ = '0'+(lt
->tm_hour
% 10);
4275 total_done
+= 2; CHECK_SIZE();
4281 *curpos
++ = '0'+(tmp
/ 10);
4282 *curpos
++ = '0'+(tmp
% 10);
4285 len
= 3; CHECK_SIZE();
4286 snprintf(curpos
, buflen
- total_done
, "%03d", lt
->tm_yday
+1);
4289 len
= 2; CHECK_SIZE();
4290 snprintf(curpos
, buflen
- total_done
, "%2d", lt
->tm_hour
);
4293 len
= 2; CHECK_SIZE();
4299 snprintf(curpos
, buflen
- total_done
, "%2d", tmp
);
4302 total_done
+= 2; CHECK_SIZE();
4303 tmp
= lt
->tm_mon
+ 1;
4304 *curpos
++ = '0'+(tmp
/ 10);
4305 *curpos
++ = '0'+(tmp
% 10);
4308 total_done
+= 2; CHECK_SIZE();
4309 *curpos
++ = '0'+(lt
->tm_min
/ 10);
4310 *curpos
++ = '0'+(lt
->tm_min
% 10);
4313 len
= 1; CHECK_SIZE();
4317 if (lt
->tm_hour
>= 12) {
4318 len
= s_pm_up_len
; CHECK_SIZE();
4319 snprintf(curpos
, buflen
-total_done
, "%s", s_pm_up
);
4321 len
= s_am_up_len
; CHECK_SIZE();
4322 snprintf(curpos
, buflen
-total_done
, "%s", s_am_up
);
4326 if (lt
->tm_hour
>= 12) {
4327 len
= s_pm_low_len
; CHECK_SIZE();
4328 snprintf(curpos
, buflen
-total_done
, "%s", s_pm_low
);
4330 len
= s_am_low_len
; CHECK_SIZE();
4331 snprintf(curpos
, buflen
-total_done
, "%s", s_am_low
);
4336 strftime(subbuf
, 64, "%I:%M:%S %p", lt
);
4338 strftime(subbuf
, 64, "%r", lt
);
4340 len
= strlen(subbuf
); CHECK_SIZE();
4341 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4344 total_done
+= 5; CHECK_SIZE();
4345 *curpos
++ = '0'+(lt
->tm_hour
/ 10);
4346 *curpos
++ = '0'+(lt
->tm_hour
% 10);
4348 *curpos
++ = '0'+(lt
->tm_min
/ 10);
4349 *curpos
++ = '0'+(lt
->tm_min
% 10);
4352 snprintf(subbuf
, 64, "%" CM_TIME_FORMAT
, mktime(lt
));
4353 len
= strlen(subbuf
); CHECK_SIZE();
4354 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4357 total_done
+= 2; CHECK_SIZE();
4358 *curpos
++ = '0'+(lt
->tm_sec
/ 10);
4359 *curpos
++ = '0'+(lt
->tm_sec
% 10);
4362 len
= 1; CHECK_SIZE();
4366 total_done
+= 8; CHECK_SIZE();
4367 *curpos
++ = '0'+(lt
->tm_hour
/ 10);
4368 *curpos
++ = '0'+(lt
->tm_hour
% 10);
4370 *curpos
++ = '0'+(lt
->tm_min
/ 10);
4371 *curpos
++ = '0'+(lt
->tm_min
% 10);
4373 *curpos
++ = '0'+(lt
->tm_sec
/ 10);
4374 *curpos
++ = '0'+(lt
->tm_sec
% 10);
4377 len
= 1; CHECK_SIZE();
4378 snprintf(curpos
, buflen
- total_done
, "%d", lt
->tm_wday
== 0 ? 7: lt
->tm_wday
);
4381 len
= 1; CHECK_SIZE();
4382 snprintf(curpos
, buflen
- total_done
, "%d", lt
->tm_wday
);
4385 strftime(subbuf
, 64, "%x", lt
);
4386 len
= strlen(subbuf
); CHECK_SIZE();
4387 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4390 strftime(subbuf
, 64, "%X", lt
);
4391 len
= strlen(subbuf
); CHECK_SIZE();
4392 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4395 total_done
+= 2; CHECK_SIZE();
4396 tmp
= lt
->tm_year
%100;
4397 *curpos
++ = '0'+(tmp
/ 10);
4398 *curpos
++ = '0'+(tmp
% 10);
4401 len
= 4; CHECK_SIZE();
4402 snprintf(curpos
, buflen
- total_done
, "%4d", lt
->tm_year
+ 1900);
4412 /* let these complicated ones be done with the libc */
4413 snprintf(subfmt
, 64, "%%%c", *format
);
4414 strftime(subbuf
, 64, subfmt
, lt
);
4415 len
= strlen(subbuf
); CHECK_SIZE();
4416 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4420 /* let these complicated modifiers be done with the libc */
4421 snprintf(subfmt
, 64, "%%%c%c", *format
, *(format
+1));
4422 strftime(subbuf
, 64, subfmt
, lt
);
4423 len
= strlen(subbuf
); CHECK_SIZE();
4424 strncpy2(curpos
, subbuf
, buflen
- total_done
);
4428 g_warning("format error (%c)", *format
);
4435 int len
= 1; CHECK_SIZE();
4436 *curpos
++ = *format
++;
4444 #define WEXITSTATUS(x) (x)
4447 static gchar
*canonical_list_to_file(GSList
*list
)
4449 GString
*result
= g_string_new(NULL
);
4450 GSList
*pathlist
= g_slist_reverse(g_slist_copy(list
));
4454 result
= g_string_append(result
, G_DIR_SEPARATOR_S
);
4456 if (pathlist
->data
) {
4457 const gchar
*root
= (gchar
*)pathlist
->data
;
4458 if (root
[0] != '\0' && g_ascii_isalpha(root
[0]) &&
4460 /* drive - don't prepend dir separator */
4462 result
= g_string_append(result
, G_DIR_SEPARATOR_S
);
4467 for (cur
= pathlist
; cur
; cur
= cur
->next
) {
4468 result
= g_string_append(result
, (gchar
*)cur
->data
);
4470 result
= g_string_append(result
, G_DIR_SEPARATOR_S
);
4472 g_slist_free(pathlist
);
4474 return g_string_free(result
, FALSE
);
4477 static GSList
*cm_split_path(const gchar
*filename
, int depth
)
4480 GSList
*canonical_parts
= NULL
;
4484 gboolean follow_symlinks
= TRUE
;
4491 errno
= EINVAL
; /* can't happen, no symlink handling */
4496 if (!g_path_is_absolute(filename
)) {
4501 path_parts
= g_strsplit(filename
, G_DIR_SEPARATOR_S
, -1);
4503 for (i
= 0; path_parts
[i
] != NULL
; i
++) {
4504 if (!strcmp(path_parts
[i
], ""))
4506 if (!strcmp(path_parts
[i
], "."))
4508 else if (!strcmp(path_parts
[i
], "..")) {
4511 g_strfreev(path_parts
);
4514 else /* Remove the last inserted element */
4516 g_slist_delete_link(canonical_parts
,
4521 canonical_parts
= g_slist_prepend(canonical_parts
,
4522 g_strdup(path_parts
[i
]));
4524 tmp_path
= canonical_list_to_file(canonical_parts
);
4526 if(g_stat(tmp_path
, &st
) < 0) {
4527 if (errno
== ENOENT
) {
4530 follow_symlinks
= FALSE
;
4535 slist_free_strings_full(canonical_parts
);
4536 g_strfreev(path_parts
);
4542 if (follow_symlinks
&& g_file_test(tmp_path
, G_FILE_TEST_IS_SYMLINK
)) {
4543 GError
*error
= NULL
;
4544 gchar
*target
= g_file_read_link(tmp_path
, &error
);
4546 if (!g_path_is_absolute(target
)) {
4547 /* remove the last inserted element */
4549 g_slist_delete_link(canonical_parts
,
4551 /* add the target */
4552 canonical_parts
= g_slist_prepend(canonical_parts
,
4556 /* and get the new target */
4557 target
= canonical_list_to_file(canonical_parts
);
4560 /* restart from absolute target */
4561 slist_free_strings_full(canonical_parts
);
4562 canonical_parts
= NULL
;
4564 canonical_parts
= cm_split_path(target
, depth
+ 1);
4566 g_error_free(error
);
4567 if (canonical_parts
== NULL
) {
4569 g_strfreev(path_parts
);
4578 g_strfreev(path_parts
);
4579 return canonical_parts
;
4583 * Canonicalize a filename, resolving symlinks along the way.
4584 * Returns a negative errno in case of error.
4586 int cm_canonicalize_filename(const gchar
*filename
, gchar
**canonical_name
) {
4587 GSList
*canonical_parts
;
4588 gboolean is_absolute
;
4590 if (filename
== NULL
)
4592 if (canonical_name
== NULL
)
4594 *canonical_name
= NULL
;
4596 is_absolute
= g_path_is_absolute(filename
);
4598 /* Always work on absolute filenames. */
4599 gchar
*cur
= g_get_current_dir();
4600 gchar
*absolute_filename
= g_strconcat(cur
, G_DIR_SEPARATOR_S
,
4603 canonical_parts
= cm_split_path(absolute_filename
, 0);
4604 g_free(absolute_filename
);
4607 canonical_parts
= cm_split_path(filename
, 0);
4609 if (canonical_parts
== NULL
)
4612 *canonical_name
= canonical_list_to_file(canonical_parts
);
4613 slist_free_strings_full(canonical_parts
);
4617 /* Returns a decoded base64 string, guaranteed to be null-terminated. */
4618 guchar
*g_base64_decode_zero(const gchar
*text
, gsize
*out_len
)
4620 gchar
*tmp
= g_base64_decode(text
, out_len
);
4621 gchar
*out
= g_strndup(tmp
, *out_len
);
4625 if (strlen(out
) != *out_len
) {
4626 g_warning("strlen(out) %"G_GSIZE_FORMAT
" != *out_len %"G_GSIZE_FORMAT
, strlen(out
), *out_len
);
4632 /* Attempts to read count bytes from a PRNG into memory area starting at buf.
4633 * It is up to the caller to make sure there is at least count bytes
4634 * available at buf. */
4636 get_random_bytes(void *buf
, size_t count
)
4638 /* Open our prng source. */
4639 #if defined G_OS_WIN32
4642 if (!CryptAcquireContext(&rnd
, NULL
, NULL
, PROV_RSA_FULL
, 0) &&
4643 !CryptAcquireContext(&rnd
, NULL
, NULL
, PROV_RSA_FULL
, CRYPT_NEWKEYSET
)) {
4644 debug_print("Could not acquire a CSP handle.\n");
4651 rnd
= open("/dev/urandom", O_RDONLY
);
4653 FILE_OP_ERROR("/dev/urandom", "open");
4654 debug_print("Could not open /dev/urandom.\n");
4659 /* Read data from the source into buf. */
4660 #if defined G_OS_WIN32
4661 if (!CryptGenRandom(rnd
, count
, buf
)) {
4662 debug_print("Could not read %"G_GSIZE_FORMAT
" random bytes.\n", count
);
4663 CryptReleaseContext(rnd
, 0);
4667 ret
= read(rnd
, buf
, count
);
4669 FILE_OP_ERROR("/dev/urandom", "read");
4670 debug_print("Could not read enough data from /dev/urandom, read only %ld of %lu bytes.\n", ret
, count
);
4676 /* Close the prng source. */
4677 #if defined G_OS_WIN32
4678 CryptReleaseContext(rnd
, 0);
4686 /* returns FALSE if parsing failed, otherwise returns TRUE and sets *server, *port
4687 and eventually *fp from filename (if not NULL, they must be free'd by caller after
4689 filenames we expect: 'host.name.port.cert' or 'host.name.port.f:i:n:g:e:r:p:r:i:n:t.cert' */
4690 gboolean
get_serverportfp_from_filename(const gchar
*str
, gchar
**server
, gchar
**port
, gchar
**fp
)
4692 const gchar
*pos
, *dotport_pos
= NULL
, *dotcert_pos
= NULL
, *dotfp_pos
= NULL
;
4694 g_return_val_if_fail(str
!= NULL
, FALSE
);
4696 pos
= str
+ strlen(str
) - 1;
4697 while ((pos
> str
) && !dotport_pos
) {
4700 /* match the .cert suffix */
4701 if (strcmp(pos
, ".cert") == 0) {
4706 /* match an eventual fingerprint */
4707 /* or the port number */
4708 if (strncmp(pos
+ 3, ":", 1) == 0) {
4714 /* match the port number */
4721 if (!dotport_pos
|| !dotcert_pos
) {
4722 g_warning("could not parse filename %s", str
);
4727 *server
= g_strndup(str
, dotport_pos
- str
);
4730 *port
= g_strndup(dotport_pos
+ 1, dotfp_pos
- dotport_pos
- 1);
4732 *fp
= g_strndup(dotfp_pos
+ 1, dotcert_pos
- dotfp_pos
- 1);
4735 *port
= g_strndup(dotport_pos
+ 1, dotcert_pos
- dotport_pos
- 1);
4740 debug_print("filename='%s' => server='%s' port='%s' fp='%s'\n",
4742 (server
? *server
: "(n/a)"),
4743 (port
? *port
: "(n/a)"),
4744 (fp
? *fp
: "(n/a)"));
4746 if (!(server
&& *server
) || !(port
&& *port
))
4753 gchar
*win32_debug_log_path(void)
4755 return g_strconcat(g_get_tmp_dir(), G_DIR_SEPARATOR_S
,
4756 "claws-win32.log", NULL
);