mirror of https://github.com/AxioDL/metaforce.git
266 lines
7.8 KiB
C++
266 lines
7.8 KiB
C++
#include "HECL/HECL.hpp"
|
|
#include <regex>
|
|
|
|
namespace HECL
|
|
{
|
|
static const SystemRegex regGLOB(_S("\\*"), SystemRegex::ECMAScript|SystemRegex::optimize);
|
|
static const SystemRegex regPATHCOMP(_S("[/\\\\]*([^/\\\\]+)"), SystemRegex::ECMAScript|SystemRegex::optimize);
|
|
static const SystemRegex regDRIVELETTER(_S("^([^/]*)/"), SystemRegex::ECMAScript|SystemRegex::optimize);
|
|
|
|
static SystemString canonRelPath(const SystemString& path)
|
|
{
|
|
/* Absolute paths not allowed */
|
|
if (path[0] == _S('/') || path[0] == _S('\\'))
|
|
{
|
|
LogModule.report(LogVisor::Error, "Absolute path provided; expected relative: %s", path.c_str());
|
|
return _S(".");
|
|
}
|
|
|
|
/* Tokenize Path */
|
|
std::vector<SystemString> comps;
|
|
HECL::SystemRegexMatch matches;
|
|
SystemString in = path;
|
|
SanitizePath(in);
|
|
for (; std::regex_search(in, matches, regPATHCOMP) ; in = matches.suffix())
|
|
{
|
|
const SystemString& match = matches[1];
|
|
if (!match.compare(_S(".")))
|
|
continue;
|
|
else if (!match.compare(_S("..")))
|
|
{
|
|
if (comps.empty())
|
|
{
|
|
/* Unable to resolve outside project */
|
|
LogModule.report(LogVisor::Error, _S("Unable to resolve outside project root in %s"), path.c_str());
|
|
return _S(".");
|
|
}
|
|
comps.pop_back();
|
|
continue;
|
|
}
|
|
comps.push_back(match);
|
|
}
|
|
|
|
/* Emit relative path */
|
|
if (comps.size())
|
|
{
|
|
auto it = comps.begin();
|
|
SystemString retval = *it;
|
|
for (++it ; it != comps.end() ; ++it)
|
|
{
|
|
retval += _S('/');
|
|
retval += *it;
|
|
}
|
|
return retval;
|
|
}
|
|
return _S(".");
|
|
}
|
|
|
|
void ProjectPath::assign(const ProjectPath& parentPath, const SystemString& path)
|
|
{
|
|
SystemString in = path;
|
|
m_projRoot = parentPath.m_projRoot;
|
|
m_relPath = canonRelPath(parentPath.m_relPath + _S('/') + in);
|
|
m_absPath = parentPath.m_projRoot + _S('/') + m_relPath;
|
|
SanitizePath(m_projRoot);
|
|
SanitizePath(m_relPath);
|
|
SanitizePath(m_absPath);
|
|
m_hash = Hash(m_relPath);
|
|
|
|
#if HECL_UCS2
|
|
m_utf8AbsPath = WideToUTF8(m_absPath);
|
|
m_utf8RelPath = WideToUTF8(m_relPath);
|
|
#endif
|
|
}
|
|
|
|
#if HECL_UCS2
|
|
void ProjectPath::assign(const ProjectPath& parentPath, const std::string& path)
|
|
{
|
|
std::string in = path;
|
|
m_projRoot = parentPath.m_projRoot;
|
|
std::wstring wpath = UTF8ToWide(in);
|
|
m_relPath = canonRelPath(parentPath.m_relPath + _S('/') + wpath);
|
|
m_absPath = parentPath.m_projRoot + _S('/') + m_relPath;
|
|
SanitizePath(m_projRoot);
|
|
SanitizePath(m_relPath);
|
|
SanitizePath(m_absPath);
|
|
m_hash = Hash(m_relPath);
|
|
m_utf8AbsPath = WideToUTF8(m_absPath);
|
|
m_utf8RelPath = WideToUTF8(m_relPath);
|
|
}
|
|
#endif
|
|
|
|
ProjectPath::PathType ProjectPath::getPathType() const
|
|
{
|
|
if (std::regex_search(m_absPath, regGLOB))
|
|
return PT_GLOB;
|
|
Sstat theStat;
|
|
if (HECL::Stat(m_absPath.c_str(), &theStat))
|
|
return PT_NONE;
|
|
if (S_ISDIR(theStat.st_mode))
|
|
return PT_DIRECTORY;
|
|
if (S_ISREG(theStat.st_mode))
|
|
return PT_FILE;
|
|
return PT_NONE;
|
|
}
|
|
|
|
Time ProjectPath::getModtime() const
|
|
{
|
|
Sstat theStat;
|
|
time_t latestTime = 0;
|
|
if (std::regex_search(m_absPath, regGLOB))
|
|
{
|
|
std::vector<SystemString> globReults;
|
|
getGlobResults(globReults);
|
|
for (SystemString& path : globReults)
|
|
{
|
|
if (!HECL::Stat(path.c_str(), &theStat))
|
|
{
|
|
if (S_ISREG(theStat.st_mode) && theStat.st_mtime > latestTime)
|
|
latestTime = theStat.st_mtime;
|
|
}
|
|
}
|
|
}
|
|
if (!HECL::Stat(m_absPath.c_str(), &theStat))
|
|
{
|
|
if (S_ISREG(theStat.st_mode))
|
|
{
|
|
return Time(theStat.st_mtime);
|
|
}
|
|
else if (S_ISDIR(theStat.st_mode))
|
|
{
|
|
#if _WIN32
|
|
#else
|
|
DIR* dir = opendir(m_absPath.c_str());
|
|
dirent* de;
|
|
while ((de = readdir(dir)))
|
|
{
|
|
if (de->d_name[0] == '.')
|
|
continue;
|
|
if (!HECL::Stat(de->d_name, &theStat))
|
|
{
|
|
if (S_ISREG(theStat.st_mode) && theStat.st_mtime > latestTime)
|
|
latestTime = theStat.st_mtime;
|
|
}
|
|
}
|
|
closedir(dir);
|
|
#endif
|
|
return Time(latestTime);
|
|
}
|
|
}
|
|
LogModule.report(LogVisor::Error, _S("invalid path type for computing modtime"));
|
|
return Time();
|
|
}
|
|
|
|
static void _recursiveGlob(std::vector<SystemString>& outPaths,
|
|
size_t level,
|
|
const SystemRegexMatch& pathCompMatches,
|
|
const SystemString& itStr,
|
|
bool needSlash)
|
|
{
|
|
if (level >= pathCompMatches.size())
|
|
return;
|
|
|
|
SystemString comp = pathCompMatches.str(level);
|
|
if (!std::regex_search(comp, regGLOB))
|
|
{
|
|
SystemString nextItStr = itStr;
|
|
if (needSlash)
|
|
nextItStr += _S('/');
|
|
nextItStr += comp;
|
|
_recursiveGlob(outPaths, level+1, pathCompMatches, nextItStr, true);
|
|
return;
|
|
}
|
|
|
|
/* Compile component into regex */
|
|
SystemRegex regComp(comp, SystemRegex::ECMAScript);
|
|
|
|
#if _WIN32
|
|
#else
|
|
DIR* dir = opendir(itStr.c_str());
|
|
if (!dir)
|
|
{
|
|
LogModule.report(LogVisor::Error, "unable to open directory for traversal at '%s'", itStr.c_str());
|
|
return;
|
|
}
|
|
|
|
struct dirent* de;
|
|
while ((de = readdir(dir)))
|
|
{
|
|
if (std::regex_search(de->d_name, regComp))
|
|
{
|
|
SystemString nextItStr = itStr;
|
|
if (needSlash)
|
|
nextItStr += '/';
|
|
nextItStr += de->d_name;
|
|
|
|
struct stat theStat;
|
|
if (stat(nextItStr.c_str(), &theStat))
|
|
continue;
|
|
|
|
if (S_ISDIR(theStat.st_mode))
|
|
_recursiveGlob(outPaths, level+1, pathCompMatches, nextItStr, true);
|
|
else if (S_ISREG(theStat.st_mode))
|
|
outPaths.push_back(nextItStr);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void ProjectPath::getGlobResults(std::vector<SystemString>& outPaths) const
|
|
{
|
|
#if _WIN32
|
|
SystemString itStr;
|
|
SystemRegexMatch letterMatch;
|
|
if (m_absPath.compare(0, 2, _S("//")))
|
|
itStr = _S("\\\\");
|
|
else if (std::regex_search(m_absPath, letterMatch, regDRIVELETTER))
|
|
if (letterMatch[1].str().size())
|
|
itStr = letterMatch[1];
|
|
#else
|
|
SystemString itStr = _S("/");
|
|
#endif
|
|
|
|
SystemRegexMatch pathCompMatches;
|
|
if (std::regex_search(m_absPath, pathCompMatches, regPATHCOMP))
|
|
_recursiveGlob(outPaths, 1, pathCompMatches, itStr, false);
|
|
}
|
|
|
|
std::unique_ptr<ProjectRootPath> SearchForProject(const SystemString& path)
|
|
{
|
|
ProjectRootPath testRoot(path);
|
|
SystemString::const_iterator begin = testRoot.getAbsolutePath().begin();
|
|
SystemString::const_iterator end = testRoot.getAbsolutePath().end();
|
|
while (begin != end)
|
|
{
|
|
SystemString testPath(begin, end);
|
|
SystemString testIndexPath = testPath + _S("/.hecl/beacon");
|
|
Sstat theStat;
|
|
if (!HECL::Stat(testIndexPath.c_str(), &theStat))
|
|
{
|
|
if (S_ISREG(theStat.st_mode))
|
|
{
|
|
FILE* fp = HECL::Fopen(testIndexPath.c_str(), _S("rb"));
|
|
if (!fp)
|
|
continue;
|
|
char magic[4];
|
|
size_t readSize = fread(magic, 1, 4, fp);
|
|
fclose(fp);
|
|
if (readSize != 4)
|
|
continue;
|
|
static const HECL::FourCC hecl("HECL");
|
|
if (HECL::FourCC(magic) != hecl)
|
|
continue;
|
|
return std::unique_ptr<ProjectRootPath>(new ProjectRootPath(testPath));
|
|
}
|
|
}
|
|
|
|
while (begin != end && *(end-1) != _S('/') && *(end-1) != _S('\\'))
|
|
--end;
|
|
if (begin != end)
|
|
--end;
|
|
}
|
|
return std::unique_ptr<ProjectRootPath>();
|
|
}
|
|
|
|
}
|