--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/daemon/stat.cc Sun Jan 20 00:22:09 2008 +0100
@@ -0,0 +1,165 @@
+// Copyright (C) 1999,2000 Bruce Guenter <bruceg@em.ca>
+//
+// This program is free software; you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation; either version 2 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program; if not, write to the Free Software
+// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+
+#include <config.h>
+#include <stdlib.h>
+#include "daemon.h"
+#include "misc/pwentry.h"
+#include "misc/lookup.h"
+#include "misc/utoa.h"
+#include "misc/stat_fns.h"
+#include "ac/dirent.h"
+
+struct count_size
+{
+ unsigned long count;
+ unsigned long size;
+ count_size() : count(0), size(0) { }
+};
+
+struct stats
+{
+ count_size unseen_new;
+ count_size unseen;
+ count_size seen;
+
+ mystring rep() const;
+};
+
+mystring stats::rep() const
+{
+ return utoa(unseen_new.count) + mystring::NUL +
+ utoa(unseen_new.size) + mystring::NUL +
+ utoa(unseen.count) + mystring::NUL +
+ utoa(unseen.size) + mystring::NUL +
+ utoa(seen.count) + mystring::NUL +
+ utoa(seen.size) + mystring::NUL;
+}
+
+class statdir
+{
+ mystring path;
+ DIR* dir;
+ struct dirent* curr;
+ struct stat statbuf;
+public:
+ statdir(const mystring& dirname);
+ ~statdir() { close(); }
+ void close();
+ operator void*() const { return (void*)dir; }
+ bool operator!() const { return !dir; }
+ const struct stat* operator->() const { return &statbuf; }
+ const char* currname() const { return curr->d_name; }
+ void operator++() { advance(); }
+ void advance();
+};
+
+statdir::statdir(const mystring& dirname)
+ : path(dirname),
+ dir(opendir(dirname.c_str()))
+{
+ advance();
+}
+
+void statdir::close()
+{
+ if(dir)
+ closedir(dir);
+ dir = NULL;
+}
+
+void statdir::advance()
+{
+ if(dir) {
+ while((curr = readdir(dir)) != 0) {
+ if(curr->d_name[0] == '.')
+ continue;
+ break;
+ }
+ }
+ if(!curr)
+ close();
+ else {
+ mystring fullpath = path + "/" + curr->d_name;
+ if(stat(fullpath.c_str(), &statbuf) == -1)
+ close();
+ }
+}
+
+bool stat_new_dir(const mystring& basename, stats& stats)
+{
+ statdir dir(basename + "/new");
+ if(!dir)
+ return false;
+ while(dir) {
+ if(S_ISREG(dir->st_mode)) {
+ ++stats.unseen_new.count;
+ stats.unseen_new.size += dir->st_blocks * 512;
+ }
+ ++dir;
+ }
+ return true;
+}
+
+bool stat_cur_dir(const mystring& basename, stats& stats)
+{
+ statdir dir(basename + "/cur");
+ if(!dir)
+ return false;
+ while(dir) {
+ if(S_ISREG(dir->st_mode)) {
+ count_size* stat = &stats.unseen;
+ const char* colon = strchr(dir.currname(), ':');
+ if(colon) {
+ ++colon;
+ if(*colon++ == '2' && *colon++ == ',' && !strchr(colon, 'S'))
+ stat = &stats.seen;
+ }
+ ++stat->count;
+ stat->size += dir->st_blocks * 512;
+ }
+ ++dir;
+ }
+ return true;
+}
+
+bool stat_dir(const mystring& basename, stats& stats)
+{
+ return stat_new_dir(basename, stats) &&
+ stat_cur_dir(basename, stats);
+}
+
+CMD(stat)
+ // Usage: stat baseuser-virtuser pass
+{
+ mystring user = args[0];
+ mystring pass = args[1];
+
+ pwentry* pw;
+ vpwentry* vpw;
+
+ OK_RESPONSE(lookup_and_validate(user, pw, vpw, pass, true, false));
+
+ if(!vpw->has_mailbox)
+ RETURN(err, "User is alias");
+
+ mystring dirname = pw->home + "/" + vpw->directory;
+ stats stats;
+ if(!stat_dir(dirname, stats))
+ RETURN(err, "Failed to stat maildir");
+
+ RETURN(ok, mystring::NUL + stats.rep());
+}