add nocanon option

This commit is contained in:
dockes 2009-11-10 17:37:56 +00:00
parent 66d9896d81
commit d91d471b7d
2 changed files with 39 additions and 20 deletions

View File

@ -39,7 +39,7 @@ using namespace std;
#endif /* NO_NAMESPACES */ #endif /* NO_NAMESPACES */
class FsTreeWalker::Internal { class FsTreeWalker::Internal {
Options options; int options;
stringstream reason; stringstream reason;
list<string> skippedNames; list<string> skippedNames;
list<string> skippedPaths; list<string> skippedPaths;
@ -53,7 +53,7 @@ class FsTreeWalker::Internal {
friend class FsTreeWalker; friend class FsTreeWalker;
}; };
FsTreeWalker::FsTreeWalker(Options opts) FsTreeWalker::FsTreeWalker(int opts)
{ {
data = new Internal; data = new Internal;
if (data) { if (data) {
@ -112,7 +112,7 @@ bool FsTreeWalker::inSkippedNames(const string& name)
bool FsTreeWalker::addSkippedPath(const string& ipath) bool FsTreeWalker::addSkippedPath(const string& ipath)
{ {
string path = path_canon(ipath); string path = (data->options & FtwNoCanon) ? ipath : path_canon(ipath);
if (find(data->skippedPaths.begin(), if (find(data->skippedPaths.begin(),
data->skippedPaths.end(), path) == data->skippedPaths.end()) data->skippedPaths.end(), path) == data->skippedPaths.end())
data->skippedPaths.push_back(path); data->skippedPaths.push_back(path);
@ -123,7 +123,8 @@ bool FsTreeWalker::setSkippedPaths(const list<string> &paths)
data->skippedPaths = paths; data->skippedPaths = paths;
for (list<string>::iterator it = data->skippedPaths.begin(); for (list<string>::iterator it = data->skippedPaths.begin();
it != data->skippedPaths.end(); it++) it != data->skippedPaths.end(); it++)
*it = path_canon(*it); if (!(data->options & FtwNoCanon))
*it = path_canon(*it);
data->skippedPaths.sort(); data->skippedPaths.sort();
data->skippedPaths.unique(); data->skippedPaths.unique();
return true; return true;
@ -142,7 +143,7 @@ bool FsTreeWalker::inSkippedPaths(const string& path)
FsTreeWalker::Status FsTreeWalker::walk(const string& _top, FsTreeWalker::Status FsTreeWalker::walk(const string& _top,
FsTreeWalkerCB& cb) FsTreeWalkerCB& cb)
{ {
string top = path_canon(_top); string top = (data->options & FtwNoCanon) ? _top : path_canon(_top);
struct stat st; struct stat st;
int statret; int statret;
@ -225,9 +226,12 @@ FsTreeWalker::Status FsTreeWalker::iwalk(const string &top,
} }
if (status & (FtwStop|FtwError)) if (status & (FtwStop|FtwError))
goto out; goto out;
if ((status = cb.processone(top, &st, FtwDirReturn))
& (FtwStop|FtwError)) // No DirReturn call when not recursing
goto out; if (!(data->options & FtwNoRecurse))
if ((status = cb.processone(top, &st, FtwDirReturn))
& (FtwStop|FtwError))
goto out;
} else if (S_ISREG(st.st_mode)) { } else if (S_ISREG(st.st_mode)) {
if ((status = cb.processone(fn, &st, FtwRegular)) & if ((status = cb.processone(fn, &st, FtwRegular)) &
(FtwStop|FtwError)) { (FtwStop|FtwError)) {
@ -256,14 +260,24 @@ FsTreeWalker::Status FsTreeWalker::iwalk(const string &top,
using namespace std; using namespace std;
static int op_flags;
#define OPT_MOINS 0x1
#define OPT_p 0x2
#define OPT_P 0x4
#define OPT_r 0x8
#define OPT_c 0x10
class myCB : public FsTreeWalkerCB { class myCB : public FsTreeWalkerCB {
public: public:
FsTreeWalker::Status processone(const string &path, FsTreeWalker::Status processone(const string &path,
const struct stat *st, const struct stat *st,
FsTreeWalker::CbFlag flg) FsTreeWalker::CbFlag flg)
{ {
if (flg == FsTreeWalker::FtwDirEnter) { if (flg == FsTreeWalker::FtwDirEnter) {
cout << "[Entering " << path << "]" << endl; if (op_flags & OPT_r)
cout << path << endl;
else
cout << "[Entering " << path << "]" << endl;
} else if (flg == FsTreeWalker::FtwDirReturn) { } else if (flg == FsTreeWalker::FtwDirReturn) {
cout << "[Returning to " << path << "]" << endl; cout << "[Returning to " << path << "]" << endl;
} else if (flg == FsTreeWalker::FtwRegular) { } else if (flg == FsTreeWalker::FtwRegular) {
@ -276,7 +290,9 @@ class myCB : public FsTreeWalkerCB {
static const char *thisprog; static const char *thisprog;
static char usage [] = static char usage [] =
"trfstreewalk [-p pattern] [-P ignpath] topdir\n\n" "trfstreewalk [-p pattern] [-P ignpath] [-r] [-c] topdir\n"
" -r : norecurse\n"
" -c : no path canonification\n"
; ;
static void static void
Usage(void) Usage(void)
@ -285,11 +301,6 @@ Usage(void)
exit(1); exit(1);
} }
static int op_flags;
#define OPT_MOINS 0x1
#define OPT_p 0x2
#define OPT_P 0x4
int main(int argc, const char **argv) int main(int argc, const char **argv)
{ {
list<string> patterns; list<string> patterns;
@ -304,6 +315,8 @@ int main(int argc, const char **argv)
Usage(); Usage();
while (**argv) while (**argv)
switch (*(*argv)++) { switch (*(*argv)++) {
case 'r': op_flags |= OPT_r; break;
case 'c': op_flags |= OPT_c; break;
case 'p': op_flags |= OPT_p; if (argc < 2) Usage(); case 'p': op_flags |= OPT_p; if (argc < 2) Usage();
patterns.push_back(*(++argv)); patterns.push_back(*(++argv));
argc--; argc--;
@ -321,7 +334,12 @@ int main(int argc, const char **argv)
Usage(); Usage();
string topdir = *argv++;argc--; string topdir = *argv++;argc--;
FsTreeWalker walker; int opt = 0;
if (op_flags & OPT_r)
opt |= FsTreeWalker::FtwNoRecurse;
if (op_flags & OPT_c)
opt |= FsTreeWalker::FtwNoCanon;
FsTreeWalker walker(opt);
walker.setSkippedNames(patterns); walker.setSkippedNames(patterns);
walker.setSkippedPaths(paths); walker.setSkippedPaths(paths);
myCB cb; myCB cb;

View File

@ -45,9 +45,10 @@ class FsTreeWalker {
enum CbFlag {FtwRegular, FtwDirEnter, FtwDirReturn}; enum CbFlag {FtwRegular, FtwDirEnter, FtwDirReturn};
enum Status {FtwOk=0, FtwError=1, FtwStop=2, enum Status {FtwOk=0, FtwError=1, FtwStop=2,
FtwStatAll = FtwError|FtwStop}; FtwStatAll = FtwError|FtwStop};
enum Options {FtwOptNone = 0, FtwNoRecurse = 1, FtwFollow = 2}; enum Options {FtwOptNone = 0, FtwNoRecurse = 1, FtwFollow = 2,
FtwNoCanon = 4};
FsTreeWalker(Options opts = FtwOptNone); FsTreeWalker(int opts = FtwOptNone);
~FsTreeWalker(); ~FsTreeWalker();
void setOpts(Options opts); void setOpts(Options opts);