]> git.notmuchmail.org Git - notmuch/blob - util/gmime-extra.c
emacs: Add new option notmuch-search-hide-excluded
[notmuch] / util / gmime-extra.c
1 #include "gmime-extra.h"
2 #include <string.h>
3
4 static
5 GMimeStream *
6 _gzfile_maybe_filter (GMimeStream *file_stream)
7 {
8     char buf[4];
9     int bytes_read;
10
11     if ((bytes_read = g_mime_stream_read (file_stream, buf, sizeof (buf))) < 0)
12         return NULL;
13
14     if (g_mime_stream_reset (file_stream))
15         return NULL;
16
17     /* check for gzipped input */
18     if (bytes_read >= 2 && buf[0] == 0x1f && (unsigned char) buf[1] == 0x8b) {
19         GMimeStream *gzstream;
20         GMimeFilter *gzfilter;
21
22         gzfilter = g_mime_filter_gzip_new (GMIME_FILTER_GZIP_MODE_UNZIP, 0);
23         if (! gzfilter)
24             return NULL;
25
26         gzstream = g_mime_stream_filter_new (file_stream);
27         if (! gzstream)
28             return NULL;
29
30         /* ignore filter id */
31         (void) g_mime_stream_filter_add ((GMimeStreamFilter *) gzstream, gzfilter);
32         g_object_unref (gzfilter);
33         g_object_unref (file_stream);
34         return gzstream;
35     } else {
36         return file_stream;
37     }
38 }
39
40 GMimeStream *
41 g_mime_stream_gzfile_new (int fd)
42 {
43     GMimeStream *file_stream;
44
45     file_stream = g_mime_stream_fs_new (fd);
46     if (! file_stream)
47         return NULL;
48
49     return _gzfile_maybe_filter (file_stream);
50 }
51
52 GMimeStream *
53 g_mime_stream_gzfile_open (const char *filename)
54 {
55     GMimeStream *file_stream;
56
57     file_stream = g_mime_stream_fs_open (filename, 0, 0, NULL);
58     if (! file_stream)
59         return NULL;
60
61     return _gzfile_maybe_filter (file_stream);
62 }
63
64 GMimeStream *
65 g_mime_stream_stdout_new ()
66 {
67     GMimeStream *stream_stdout = NULL;
68     GMimeStream *stream_buffered = NULL;
69
70     stream_stdout = g_mime_stream_pipe_new (STDOUT_FILENO);
71     if (! stream_stdout)
72         return NULL;
73
74     g_mime_stream_pipe_set_owner (GMIME_STREAM_PIPE (stream_stdout), FALSE);
75
76     stream_buffered = g_mime_stream_buffer_new (stream_stdout, GMIME_STREAM_BUFFER_BLOCK_WRITE);
77
78     g_object_unref (stream_stdout);
79
80     return stream_buffered;
81 }
82
83 /**
84  * copy a glib string into a talloc context, and free it.
85  */
86 static char *
87 g_string_talloc_strdup (void *ctx, char *g_string)
88 {
89     char *new_str = talloc_strdup (ctx, g_string);
90
91     g_free (g_string);
92     return new_str;
93 }
94
95 const char *
96 g_mime_certificate_get_valid_userid (GMimeCertificate *cert)
97 {
98     /* output user id only if validity is FULL or ULTIMATE. */
99     const char *uid = g_mime_certificate_get_user_id (cert);
100
101     if (uid == NULL)
102         return uid;
103     GMimeValidity validity = g_mime_certificate_get_id_validity (cert);
104
105     if (validity == GMIME_VALIDITY_FULL || validity == GMIME_VALIDITY_ULTIMATE)
106         return uid;
107     return NULL;
108 }
109
110 const char *
111 g_mime_certificate_get_fpr16 (GMimeCertificate *cert)
112 {
113     const char *fpr = g_mime_certificate_get_fingerprint (cert);
114
115     if (! fpr || strlen (fpr) < 16)
116         return fpr;
117
118     return fpr + (strlen (fpr) - 16);
119 }
120
121 char *
122 g_mime_message_get_address_string (GMimeMessage *message, GMimeAddressType type)
123 {
124     InternetAddressList *list = g_mime_message_get_addresses (message, type);
125
126     return internet_address_list_to_string (list, NULL, 0);
127 }
128
129 char *
130 g_mime_message_get_date_string (void *ctx, GMimeMessage *message)
131 {
132     GDateTime *parsed_date = g_mime_message_get_date (message);
133
134     if (parsed_date) {
135         char *date = g_mime_utils_header_format_date (parsed_date);
136         return g_string_talloc_strdup (ctx, date);
137     } else {
138         return talloc_strdup (ctx, "Thu, 01 Jan 1970 00:00:00 +0000");
139     }
140 }
141
142 InternetAddressList *
143 g_mime_message_get_reply_to_list (GMimeMessage *message)
144 {
145     return g_mime_message_get_reply_to (message);
146 }
147
148 const char *
149 g_mime_message_get_from_string (GMimeMessage *message)
150 {
151     return g_mime_object_get_header (GMIME_OBJECT (message), "From");
152 }
153
154 char *
155 g_mime_message_get_reply_to_string (void *ctx, GMimeMessage *message)
156 {
157     InternetAddressList *list = g_mime_message_get_reply_to (message);
158
159     return g_string_talloc_strdup (ctx, internet_address_list_to_string (list, NULL, 0));
160 }
161
162 void
163 g_mime_parser_set_scan_from (GMimeParser *parser, gboolean flag)
164 {
165     g_mime_parser_set_format (parser, flag ? GMIME_FORMAT_MBOX : GMIME_FORMAT_MESSAGE);
166 }
167
168 /* In GMime 3.0, status GOOD and VALID both imply something about the
169  * validity of the UIDs attached to the signing key. This forces us to
170  * use following somewhat relaxed definition of a "good" signature to
171  * preserve current notmuch semantics.
172  */
173
174 gboolean
175 g_mime_signature_status_good (GMimeSignatureStatus status)
176 {
177     return ((status & (GMIME_SIGNATURE_STATUS_RED | GMIME_SIGNATURE_STATUS_ERROR_MASK)) == 0);
178 }
179
180 gboolean
181 g_mime_signature_status_bad (GMimeSignatureStatus status)
182 {
183     return (status & GMIME_SIGNATURE_STATUS_RED);
184 }
185
186 gboolean
187 g_mime_signature_status_error (GMimeSignatureStatus status)
188 {
189     return (status & GMIME_SIGNATURE_STATUS_ERROR_MASK);
190 }
191
192 gint64
193 g_mime_utils_header_decode_date_unix (const char *date)
194 {
195     GDateTime *parsed_date = g_mime_utils_header_decode_date (date);
196     time_t ret;
197
198     if (parsed_date) {
199         ret = g_date_time_to_unix (parsed_date);
200         g_date_time_unref (parsed_date);
201     } else {
202         ret = 0;
203     }
204
205     return ret;
206 }