blob: 4d28a6bf8ec5ee10092985a807ee878621e66a81 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
/**
* @file maildir-updateflags.cc
* @brief Implementation of the Maildir class.
* @author Andreas Aardal Hanssen
* @date 2002-2005
*/
#include "iodevice.h"
#include "iofactory.h"
#include "maildir.h"
#include <errno.h>
#include <dirent.h>
#include <unistd.h>
using std::string;
void Binc::Maildir::updateFlags()
{
// don't update a read-only mailbox.
if (readOnly) return;
// open the cur/ directory
string curpath = path + "/cur/";
DIR *pdir = opendir(curpath.c_str());
if (pdir == nullptr) {
bincError << "failed to open " << curpath << ": " << strerror(errno) << std::endl;
return;
}
// read all entries in the directory
std::vector<string> entries;
struct dirent *pdirent;
while ((pdirent = readdir(pdir)) != nullptr) {
string filename = pdirent->d_name;
if (filename[0] == '.') continue;
entries.push_back(filename);
}
closedir(pdir);
bool customFlagsChanged = false;
// loop through all messages in cur/, and update the flags that have changed.
for (const auto &filename : entries) {
// separate the unique name from the flags. accept messages that do not
// contain the flags section.
string uniquename;
string::size_type pos;
if ((pos = filename.find(":2,")) != string::npos)
uniquename = filename.substr(0, pos);
else
uniquename = filename;
// only update flags for messages that are known.
MaildirMessage *message = get(uniquename);
if (!message) continue;
// check if custom flags have changed; if so, we need to regenerate the
// cache file.
if (message->internalFlags & MaildirMessage::CustomFlagsChanged) customFlagsChanged = true;
// generate the flag string.
string flags;
int mflags = message->getStdFlags();
if (mflags & Message::F_DRAFT) flags += "D";
if (mflags & Message::F_FLAGGED) flags += "F";
if (mflags & Message::F_PASSED) flags += "P";
if (mflags & Message::F_ANSWERED) flags += "R";
if (mflags & Message::F_SEEN) flags += "S";
if (mflags & Message::F_DELETED) flags += "T";
// prepare the old and new message name. if equal, skip to the next
// message.
string srcname = curpath + filename;
string destname = curpath + uniquename + ":2," + flags;
if (srcname == destname) continue;
// rename the file
if (rename(srcname.c_str(), destname.c_str()) != 0) {
if (errno == ENOENT) {
closedir(pdir);
if ((pdir = opendir(curpath.c_str())) == nullptr) {
bincError << "failed to open " << curpath << ": " << strerror(errno) << std::endl;
return;
}
// restart scan. this catches the race condition where concurrent
// clients may have set a flag or removed the message.
continue;
}
bincError << "failed to rename " << srcname << " to " << destname << ": "
<< strerror(errno) << std::endl;
} else {
index.insert(uniquename, 0, uniquename + ":2," + flags);
}
}
// finally, regenerate the cache file if the custom flags have changed.
if (customFlagsChanged) {
Lock lock(path);
writeCache();
}
}
|