dircount/dircount_v2.cc

89 lines
2.1 KiB
C++
Executable File

#include <dirent.h>
#include <string>
#include <sys/stat.h>
#include <unordered_map>
#include <unistd.h>
#include <iostream>
using namespace std;
int dircnt = 0, filecnt = 0, lnkcnt = 0;
unsigned long space_used;
struct stat buf;
unordered_map<ino_t, bool> ht;
bool listFileAndType(const string &dir) {
DIR *dirp = opendir(dir.c_str());
if (dirp) {
struct dirent *dp = 0;
while ((dp = readdir(dirp)) != 0 ) {
string file( dp->d_name );
if (file == "." || file == "..") continue;
if (dp->d_type == DT_DIR) {
string dirPath = dir + "/" + file;
cout << "[d] " << dirPath << endl;
dircnt++;
// recurse into directory
listFileAndType( dirPath );
}
else {
if (ht[dp->d_ino]) continue;
string statpath;
// char *linkname;
statpath = dir + "/" + file;
stat(statpath.c_str(), &buf);
switch(dp->d_type) {
case DT_REG:
filecnt++;
space_used += buf.st_blocks;
cout << "[f] ";
break;
case DT_LNK:
lnkcnt++;
// char* linkname;
// char linkname[buf.st_size + 1];
// linkname = malloc(buf.st_size + 1);
// ssize_t r;
// r = readlink(statpath.c_str(), linkname, buf.st_size + 1);
// linkname[buf.st_size] = '\0';
cout << "[l] ";
// cout << statpath << " points to " << linkname;
break;
default:
cout << "[no type] ";
break;
}
ht[dp->d_ino] = true;
cout << statpath << endl;
}
}
closedir( dirp );
return true;
}
else {
return false;
}
}
int main( int argc, char **argv ) {
const string dir = (argc > 1 ? argv[1] : ".");
if (!listFileAndType(dir)) {
cout << "Error: Cannot open directory '" << dir << "'" << endl;
}
cout << endl << "totals" << endl;
cout << "file count: " << filecnt << "\tdir cnt: " << dircnt << "\tlink cnt: " << lnkcnt << endl;
cout << "space used: " << space_used << " blocks" << endl;
cout << "\t" << space_used*512 << " bytes" << endl;
return 0;
}