1
/* Copyright (c) 2013 Dovecot authors, see the included COPYING file */
6
#include "message-parser.h"
7
#include "mail-storage.h"
8
#include "mail-namespace.h"
10
#include "imap-msgpart.h"
11
#include "imap-msgpart-url.h"
13
struct imap_msgpart_url {
18
uoff_t partial_offset, partial_size;
20
struct imap_msgpart *part;
22
struct mail_user *user;
23
struct mailbox *selected_box;
25
struct mailbox_transaction_context *trans;
28
struct imap_msgpart_open_result result;
30
unsigned int decode_cte_to_binary:1;
33
int imap_msgpart_url_create(struct mail_user *user, const struct imap_url *url,
34
struct imap_msgpart_url **mpurl_r,
37
const char *section = url->section == NULL ? "" : url->section;
38
struct imap_msgpart_url *mpurl;
39
struct imap_msgpart *msgpart;
41
i_assert(url->mailbox != NULL && url->uid != 0 &&
42
url->search_program == NULL);
44
if (imap_msgpart_parse(section, &msgpart) < 0) {
45
*error_r = "Invalid section";
49
mpurl = i_new(struct imap_msgpart_url, 1);
50
mpurl->part = msgpart;
52
mpurl->mailbox = i_strdup(url->mailbox);
53
mpurl->uidvalidity = url->uidvalidity;
54
mpurl->uid = url->uid;
55
if (url->section != NULL)
56
mpurl->section = i_strdup(url->section);
57
mpurl->partial_offset = url->partial_offset;
58
mpurl->partial_size = url->partial_size;
60
imap_msgpart_set_partial(msgpart, url->partial_offset,
61
url->partial_size == 0 ?
62
(uoff_t)-1 : url->partial_size);
68
int imap_msgpart_url_parse(struct mail_user *user, struct mailbox *selected_box,
69
const char *urlstr, struct imap_msgpart_url **mpurl_r,
72
struct mailbox_status box_status;
73
struct imap_url base_url, *url;
77
memset(&base_url, 0, sizeof(base_url));
78
if (selected_box != NULL) {
79
mailbox_get_open_status(selected_box, STATUS_UIDVALIDITY,
81
base_url.mailbox = mailbox_get_vname(selected_box);
82
base_url.uidvalidity = box_status.uidvalidity;
86
if (imap_url_parse(urlstr, &base_url,
87
IMAP_URL_PARSE_REQUIRE_RELATIVE, &url, &error) < 0) {
88
*error_r = t_strconcat("Invalid IMAP URL: ", error, NULL);
91
if (url->mailbox == NULL) {
92
*error_r = "Mailbox-relative IMAP URL, but no mailbox selected";
95
if (url->uid == 0 || url->search_program != NULL) {
96
*error_r = "Invalid messagepart IMAP URL";
99
if (imap_msgpart_url_create(user, url, mpurl_r, error_r) < 0)
101
(*mpurl_r)->selected_box = selected_box;
105
struct mailbox *imap_msgpart_url_get_mailbox(struct imap_msgpart_url *mpurl)
110
int imap_msgpart_url_open_mailbox(struct imap_msgpart_url *mpurl,
111
struct mailbox **box_r, enum mail_error *error_code_r,
112
const char **error_r)
114
struct mailbox_status box_status;
115
enum mailbox_flags flags = MAILBOX_FLAG_READONLY;
116
struct mail_namespace *ns;
119
if (mpurl->box != NULL) {
121
*error_code_r = MAIL_ERROR_NONE;
125
/* find mailbox namespace */
126
ns = mail_namespace_find(mpurl->user->namespaces, mpurl->mailbox);
129
if (mpurl->selected_box != NULL &&
130
mailbox_equals(mpurl->selected_box, ns, mpurl->mailbox))
131
box = mpurl->selected_box;
133
box = mailbox_alloc(ns->list, mpurl->mailbox, flags);
134
if (mailbox_open(box) < 0) {
135
*error_r = mail_storage_get_last_error(mailbox_get_storage(box),
137
if (box != mpurl->selected_box)
139
return *error_code_r == MAIL_ERROR_TEMP ? -1 : 0;
142
/* verify UIDVALIDITY */
143
mailbox_get_open_status(box, STATUS_UIDVALIDITY, &box_status);
144
if (mpurl->uidvalidity > 0 &&
145
box_status.uidvalidity != mpurl->uidvalidity) {
146
*error_r = "Invalid UIDVALIDITY";
147
*error_code_r = MAIL_ERROR_EXPUNGED;
148
if (box != mpurl->selected_box)
157
int imap_msgpart_url_open_mail(struct imap_msgpart_url *mpurl,
158
struct mail **mail_r, const char **error_r)
160
struct mailbox_transaction_context *t;
162
enum mail_error error_code;
166
if (mpurl->mail != NULL) {
167
*mail_r = mpurl->mail;
171
/* open mailbox if it is not yet open */
172
if ((ret = imap_msgpart_url_open_mailbox(mpurl, &box, &error_code,
176
/* start transaction */
177
t = mailbox_transaction_begin(box, 0);
178
mail = mail_alloc(t, MAIL_FETCH_MESSAGE_PARTS |
179
MAIL_FETCH_IMAP_BODYSTRUCTURE, NULL);
181
/* find the message */
182
if (!mail_set_uid(mail, mpurl->uid)) {
183
*error_r = "Message not found";
185
mailbox_transaction_rollback(&t);
195
struct imap_msgpart *
196
imap_msgpart_url_get_part(struct imap_msgpart_url *mpurl)
201
void imap_msgpart_url_set_decode_to_binary(struct imap_msgpart_url *mpurl)
203
imap_msgpart_set_decode_to_binary(mpurl->part);
206
int imap_msgpart_url_read_part(struct imap_msgpart_url *mpurl,
207
struct imap_msgpart_open_result *result_r,
208
const char **error_r)
213
if (mpurl->result.input != NULL) {
214
i_stream_seek(mpurl->result.input, 0);
215
*result_r = mpurl->result;
219
/* open mail if it is not yet open */
220
ret = imap_msgpart_url_open_mail(mpurl, &mail, error_r);
224
/* open the referenced part as a stream */
225
ret = imap_msgpart_open(mail, mpurl->part, result_r);
227
*error_r = mailbox_get_last_error(mpurl->box, NULL);
231
mpurl->result = *result_r;
235
int imap_msgpart_url_verify(struct imap_msgpart_url *mpurl,
236
const char **error_r)
241
if (mpurl->result.input != NULL)
244
/* open mail if it is not yet open */
245
ret = imap_msgpart_url_open_mail(mpurl, &mail, error_r);
249
int imap_msgpart_url_get_bodypartstructure(struct imap_msgpart_url *mpurl,
250
const char **bpstruct_r,
251
const char **error_r)
256
/* open mail if it is not yet open */
257
ret = imap_msgpart_url_open_mail(mpurl, &mail, error_r);
261
ret = imap_msgpart_bodypartstructure(mail, mpurl->part, bpstruct_r);
263
*error_r = mailbox_get_last_error(mpurl->box, NULL);
265
*error_r = "Message part not found";
269
void imap_msgpart_url_free(struct imap_msgpart_url **_mpurl)
271
struct imap_msgpart_url *mpurl = *_mpurl;
275
if (mpurl->result.input != NULL)
276
i_stream_unref(&mpurl->result.input);
277
if (mpurl->part != NULL)
278
imap_msgpart_free(&mpurl->part);
279
if (mpurl->mail != NULL)
280
mail_free(&mpurl->mail);
281
if (mpurl->trans != NULL)
282
mailbox_transaction_rollback(&mpurl->trans);
283
if (mpurl->box != NULL && mpurl->box != mpurl->selected_box)
284
mailbox_free(&mpurl->box);
285
if (mpurl->section != NULL)
286
i_free(mpurl->section);
287
i_free(mpurl->mailbox);