| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516 |
- /*=========================================================================
- Program: KWSys - Kitware System Library
- Module: $RCSfile$
- Copyright (c) Kitware, Inc., Insight Consortium. All rights reserved.
- See Copyright.txt or http://www.kitware.com/Copyright.htm for details.
- This software is distributed WITHOUT ANY WARRANTY; without even
- the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
- PURPOSE. See the above copyright notices for more information.
- =========================================================================*/
- #include "kwsysPrivate.h"
- #include KWSYS_HEADER(Glob.hxx)
- #include KWSYS_HEADER(Configure.hxx)
- #include KWSYS_HEADER(RegularExpression.hxx)
- #include KWSYS_HEADER(SystemTools.hxx)
- #include KWSYS_HEADER(Directory.hxx)
- #include KWSYS_HEADER(stl/string)
- #include KWSYS_HEADER(stl/vector)
- // Work-around CMake dependency scanning limitation. This must
- // duplicate the above list of headers.
- #if 0
- # include "Glob.hxx.in"
- # include "Directory.hxx.in"
- # include "Configure.hxx.in"
- # include "RegularExpression.hxx.in"
- # include "SystemTools.hxx.in"
- # include "kwsys_stl.hxx.in"
- # include "kwsys_stl_string.hxx.in"
- #endif
- #include <ctype.h>
- #include <stdio.h>
- #include <string.h>
- namespace KWSYS_NAMESPACE
- {
- #if defined(_WIN32) || defined(__APPLE__) || defined(__CYGWIN__)
- // On Windows and apple, no difference between lower and upper case
- # define KWSYS_GLOB_CASE_INDEPENDENT
- #endif
- #if defined(_WIN32) || defined(__CYGWIN__)
- // Handle network paths
- # define KWSYS_GLOB_SUPPORT_NETWORK_PATHS
- #endif
- //----------------------------------------------------------------------------
- class GlobInternals
- {
- public:
- kwsys_stl::vector<kwsys_stl::string> Files;
- kwsys_stl::vector<kwsys::RegularExpression> Expressions;
- };
- //----------------------------------------------------------------------------
- Glob::Glob()
- {
- this->Internals = new GlobInternals;
- this->Recurse = false;
- this->Relative = "";
- this->RecurseThroughSymlinks = true;
- // RecurseThroughSymlinks is true by default for backwards compatibility,
- // not because it's a good idea...
- this->FollowedSymlinkCount = 0;
- }
- //----------------------------------------------------------------------------
- Glob::~Glob()
- {
- delete this->Internals;
- }
- //----------------------------------------------------------------------------
- kwsys_stl::vector<kwsys_stl::string>& Glob::GetFiles()
- {
- return this->Internals->Files;
- }
- //----------------------------------------------------------------------------
- kwsys_stl::string Glob::PatternToRegex(const kwsys_stl::string& pattern,
- bool require_whole_string)
- {
- // Incrementally build the regular expression from the pattern.
- kwsys_stl::string regex = require_whole_string? "^" : "";
- kwsys_stl::string::const_iterator pattern_first = pattern.begin();
- kwsys_stl::string::const_iterator pattern_last = pattern.end();
- for(kwsys_stl::string::const_iterator i = pattern_first;
- i != pattern_last; ++i)
- {
- int c = *i;
- if(c == '*')
- {
- // A '*' (not between brackets) matches any string.
- // We modify this to not match slashes since the orignal glob
- // pattern documentation was meant for matching file name
- // components separated by slashes.
- regex += "[^/]*";
- }
- else if(c == '?')
- {
- // A '?' (not between brackets) matches any single character.
- // We modify this to not match slashes since the orignal glob
- // pattern documentation was meant for matching file name
- // components separated by slashes.
- regex += "[^/]";
- }
- else if(c == '[')
- {
- // Parse out the bracket expression. It begins just after the
- // opening character.
- kwsys_stl::string::const_iterator bracket_first = i+1;
- kwsys_stl::string::const_iterator bracket_last = bracket_first;
- // The first character may be complementation '!' or '^'.
- if(bracket_last != pattern_last &&
- (*bracket_last == '!' || *bracket_last == '^'))
- {
- ++bracket_last;
- }
- // If the next character is a ']' it is included in the brackets
- // because the bracket string may not be empty.
- if(bracket_last != pattern_last && *bracket_last == ']')
- {
- ++bracket_last;
- }
- // Search for the closing ']'.
- while(bracket_last != pattern_last && *bracket_last != ']')
- {
- ++bracket_last;
- }
- // Check whether we have a complete bracket string.
- if(bracket_last == pattern_last)
- {
- // The bracket string did not end, so it was opened simply by
- // a '[' that is supposed to be matched literally.
- regex += "\\[";
- }
- else
- {
- // Convert the bracket string to its regex equivalent.
- kwsys_stl::string::const_iterator k = bracket_first;
- // Open the regex block.
- regex += "[";
- // A regex range complement uses '^' instead of '!'.
- if(k != bracket_last && *k == '!')
- {
- regex += "^";
- ++k;
- }
- // Convert the remaining characters.
- for(; k != bracket_last; ++k)
- {
- // Backslashes must be escaped.
- if(*k == '\\')
- {
- regex += "\\";
- }
- // Store this character.
- regex += *k;
- }
- // Close the regex block.
- regex += "]";
- // Jump to the end of the bracket string.
- i = bracket_last;
- }
- }
- else
- {
- // A single character matches itself.
- int ch = c;
- if(!(('a' <= ch && ch <= 'z') ||
- ('A' <= ch && ch <= 'Z') ||
- ('0' <= ch && ch <= '9')))
- {
- // Escape the non-alphanumeric character.
- regex += "\\";
- }
- #if defined(KWSYS_GLOB_CASE_INDEPENDENT)
- else
- {
- // On case-insensitive systems file names are converted to lower
- // case before matching.
- ch = tolower(ch);
- }
- #endif
- // Store the character.
- regex.append(1, static_cast<char>(ch));
- }
- }
- if(require_whole_string)
- {
- regex += "$";
- }
- return regex;
- }
- //----------------------------------------------------------------------------
- void Glob::RecurseDirectory(kwsys_stl::string::size_type start,
- const kwsys_stl::string& dir, bool dir_only)
- {
- kwsys::Directory d;
- if ( !d.Load(dir.c_str()) )
- {
- return;
- }
- unsigned long cc;
- kwsys_stl::string fullname;
- kwsys_stl::string realname;
- kwsys_stl::string fname;
- for ( cc = 0; cc < d.GetNumberOfFiles(); cc ++ )
- {
- fname = d.GetFile(cc);
- if ( strcmp(fname.c_str(), ".") == 0 ||
- strcmp(fname.c_str(), "..") == 0 )
- {
- continue;
- }
- if ( start == 0 )
- {
- realname = dir + fname;
- }
- else
- {
- realname = dir + "/" + fname;
- }
- #if defined( KWSYS_GLOB_CASE_INDEPENDENT )
- // On Windows and apple, no difference between lower and upper case
- fname = kwsys::SystemTools::LowerCase(fname);
- #endif
- if ( start == 0 )
- {
- fullname = dir + fname;
- }
- else
- {
- fullname = dir + "/" + fname;
- }
- if ( !dir_only || !kwsys::SystemTools::FileIsDirectory(realname.c_str()) )
- {
- if ( (this->Internals->Expressions.size() > 0) &&
- this->Internals->Expressions[
- this->Internals->Expressions.size()-1].find(fname.c_str()) )
- {
- this->AddFile(this->Internals->Files, realname.c_str());
- }
- }
- if ( kwsys::SystemTools::FileIsDirectory(realname.c_str()) )
- {
- bool isSymLink = kwsys::SystemTools::FileIsSymlink(realname.c_str());
- if (!isSymLink || this->RecurseThroughSymlinks)
- {
- if (isSymLink)
- {
- ++this->FollowedSymlinkCount;
- }
- this->RecurseDirectory(start+1, realname, dir_only);
- }
- }
- }
- }
- //----------------------------------------------------------------------------
- void Glob::ProcessDirectory(kwsys_stl::string::size_type start,
- const kwsys_stl::string& dir, bool dir_only)
- {
- //kwsys_ios::cout << "ProcessDirectory: " << dir << kwsys_ios::endl;
- bool last = ( start == this->Internals->Expressions.size()-1 );
- if ( last && this->Recurse )
- {
- this->RecurseDirectory(start, dir, dir_only);
- return;
- }
- if ( start >= this->Internals->Expressions.size() )
- {
- return;
- }
- kwsys::Directory d;
- if ( !d.Load(dir.c_str()) )
- {
- return;
- }
- unsigned long cc;
- kwsys_stl::string fullname;
- kwsys_stl::string realname;
- kwsys_stl::string fname;
- for ( cc = 0; cc < d.GetNumberOfFiles(); cc ++ )
- {
- fname = d.GetFile(cc);
- if ( strcmp(fname.c_str(), ".") == 0 ||
- strcmp(fname.c_str(), "..") == 0 )
- {
- continue;
- }
- if ( start == 0 )
- {
- realname = dir + fname;
- }
- else
- {
- realname = dir + "/" + fname;
- }
- #if defined(KWSYS_GLOB_CASE_INDEPENDENT)
- // On case-insensitive file systems convert to lower case for matching.
- fname = kwsys::SystemTools::LowerCase(fname);
- #endif
- if ( start == 0 )
- {
- fullname = dir + fname;
- }
- else
- {
- fullname = dir + "/" + fname;
- }
- //kwsys_ios::cout << "Look at file: " << fname << kwsys_ios::endl;
- //kwsys_ios::cout << "Match: "
- // << this->Internals->TextExpressions[start].c_str() << kwsys_ios::endl;
- //kwsys_ios::cout << "Full name: " << fullname << kwsys_ios::endl;
- if ( (!dir_only || !last) &&
- !kwsys::SystemTools::FileIsDirectory(realname.c_str()) )
- {
- continue;
- }
- if ( this->Internals->Expressions[start].find(fname.c_str()) )
- {
- if ( last )
- {
- this->AddFile(this->Internals->Files, realname.c_str());
- }
- else
- {
- this->ProcessDirectory(start+1, realname + "/", dir_only);
- }
- }
- }
- }
- //----------------------------------------------------------------------------
- bool Glob::FindFiles(const kwsys_stl::string& inexpr)
- {
- kwsys_stl::string cexpr;
- kwsys_stl::string::size_type cc;
- kwsys_stl::string expr = inexpr;
- this->Internals->Expressions.clear();
- this->Internals->Files.clear();
- if ( !kwsys::SystemTools::FileIsFullPath(expr.c_str()) )
- {
- expr = kwsys::SystemTools::GetCurrentWorkingDirectory();
- expr += "/" + inexpr;
- }
- kwsys_stl::string fexpr = expr;
- int skip = 0;
- int last_slash = 0;
- for ( cc = 0; cc < expr.size(); cc ++ )
- {
- if ( cc > 0 && expr[cc] == '/' && expr[cc-1] != '\\' )
- {
- last_slash = static_cast<int>(cc);
- }
- if ( cc > 0 &&
- (expr[cc] == '[' || expr[cc] == '?' || expr[cc] == '*') &&
- expr[cc-1] != '\\' )
- {
- break;
- }
- }
- if ( last_slash > 0 )
- {
- //kwsys_ios::cout << "I can skip: " << fexpr.substr(0, last_slash)
- //<< kwsys_ios::endl;
- skip = last_slash;
- }
- if ( skip == 0 )
- {
- #if defined( KWSYS_GLOB_SUPPORT_NETWORK_PATHS )
- // Handle network paths
- if ( expr[0] == '/' && expr[1] == '/' )
- {
- int cnt = 0;
- for ( cc = 2; cc < expr.size(); cc ++ )
- {
- if ( expr[cc] == '/' )
- {
- cnt ++;
- if ( cnt == 2 )
- {
- break;
- }
- }
- }
- skip = int(cc + 1);
- }
- else
- #endif
- // Handle drive letters on Windows
- if ( expr[1] == ':' && expr[0] != '/' )
- {
- skip = 2;
- }
- }
- if ( skip > 0 )
- {
- expr = expr.substr(skip);
- }
- cexpr = "";
- for ( cc = 0; cc < expr.size(); cc ++ )
- {
- int ch = expr[cc];
- if ( ch == '/' )
- {
- if ( cexpr.size() > 0 )
- {
- this->AddExpression(cexpr.c_str());
- }
- cexpr = "";
- }
- else
- {
- cexpr.append(1, static_cast<char>(ch));
- }
- }
- if ( cexpr.size() > 0 )
- {
- this->AddExpression(cexpr.c_str());
- }
- // Handle network paths
- if ( skip > 0 )
- {
- this->ProcessDirectory(0, fexpr.substr(0, skip) + "/",
- true);
- }
- else
- {
- this->ProcessDirectory(0, "/", true);
- }
- return true;
- }
- //----------------------------------------------------------------------------
- void Glob::AddExpression(const char* expr)
- {
- this->Internals->Expressions.push_back(
- kwsys::RegularExpression(
- this->PatternToRegex(expr).c_str()));
- }
- //----------------------------------------------------------------------------
- void Glob::SetRelative(const char* dir)
- {
- if ( !dir )
- {
- this->Relative = "";
- return;
- }
- this->Relative = dir;
- }
- //----------------------------------------------------------------------------
- const char* Glob::GetRelative()
- {
- if ( this->Relative.empty() )
- {
- return 0;
- }
- return this->Relative.c_str();
- }
- //----------------------------------------------------------------------------
- void Glob::AddFile(kwsys_stl::vector<kwsys_stl::string>& files, const char* file)
- {
- if ( !this->Relative.empty() )
- {
- files.push_back(kwsys::SystemTools::RelativePath(this->Relative.c_str(), file));
- }
- else
- {
- files.push_back(file);
- }
- }
- } // namespace KWSYS_NAMESPACE
|