added Poco::File::linkTo()

This commit is contained in:
Günter Obiltschnig 2017-12-14 10:35:07 +01:00
parent 9472b163b4
commit d29972ef24
12 changed files with 129 additions and 26 deletions

View File

@ -63,6 +63,13 @@ class Foundation_API File: private FileImpl
public: public:
typedef FileSizeImpl FileSize; typedef FileSizeImpl FileSize;
enum LinkType
/// Type of link for linkTo().
{
LINK_HARD = 0, /// hard link
LINK_SYMBOLIC = 1 /// symbolic link
};
File(); File();
/// Creates the file. /// Creates the file.
@ -189,6 +196,14 @@ public:
void renameTo(const std::string& path); void renameTo(const std::string& path);
/// Renames the file to the new name. /// Renames the file to the new name.
void linkTo(const std::string& path, LinkType type = LINK_SYMBOLIC) const;
/// Creates a link (symbolic or hard, depending on type argument)
/// at the given path to the file or directory.
///
/// May not be supported on all platforms.
/// Furthermore, some operating systems do not allow creating
/// hard links to directories.
void remove(bool recursive = false); void remove(bool recursive = false);
/// Deletes the file. If recursive is true and the /// Deletes the file. If recursive is true and the
/// file is a directory, recursively deletes all /// file is a directory, recursively deletes all

View File

@ -53,6 +53,7 @@ protected:
void setExecutableImpl(bool flag = true); void setExecutableImpl(bool flag = true);
void copyToImpl(const std::string& path) const; void copyToImpl(const std::string& path) const;
void renameToImpl(const std::string& path); void renameToImpl(const std::string& path);
void linkToImpl(const std::string& path, int type) const;
void removeImpl(); void removeImpl();
bool createFileImpl(); bool createFileImpl();
bool createDirectoryImpl(); bool createDirectoryImpl();

View File

@ -53,6 +53,7 @@ protected:
void setExecutableImpl(bool flag = true); void setExecutableImpl(bool flag = true);
void copyToImpl(const std::string& path) const; void copyToImpl(const std::string& path) const;
void renameToImpl(const std::string& path); void renameToImpl(const std::string& path);
void linkToImpl(const std::string& path, int type) const;
void removeImpl(); void removeImpl();
bool createFileImpl(); bool createFileImpl();
bool createDirectoryImpl(); bool createDirectoryImpl();

View File

@ -54,6 +54,7 @@ protected:
void setExecutableImpl(bool flag = true); void setExecutableImpl(bool flag = true);
void copyToImpl(const std::string& path) const; void copyToImpl(const std::string& path) const;
void renameToImpl(const std::string& path); void renameToImpl(const std::string& path);
void linkToImpl(const std::string& path, int type) const;
void removeImpl(); void removeImpl();
bool createFileImpl(); bool createFileImpl();
bool createDirectoryImpl(); bool createDirectoryImpl();

View File

@ -54,6 +54,7 @@ protected:
void setExecutableImpl(bool flag = true); void setExecutableImpl(bool flag = true);
void copyToImpl(const std::string& path) const; void copyToImpl(const std::string& path) const;
void renameToImpl(const std::string& path); void renameToImpl(const std::string& path);
void linkToImpl(const std::string& path, int type) const;
void removeImpl(); void removeImpl();
bool createFileImpl(); bool createFileImpl();
bool createDirectoryImpl(); bool createDirectoryImpl();

View File

@ -54,6 +54,7 @@ protected:
void setExecutableImpl(bool flag = true); void setExecutableImpl(bool flag = true);
void copyToImpl(const std::string& path) const; void copyToImpl(const std::string& path) const;
void renameToImpl(const std::string& path); void renameToImpl(const std::string& path);
void linkToImpl(const std::string& path, int type) const;
void removeImpl(); void removeImpl();
bool createFileImpl(); bool createFileImpl();
bool createDirectoryImpl(); bool createDirectoryImpl();

View File

@ -256,6 +256,12 @@ void File::renameTo(const std::string& path)
} }
void File::linkTo(const std::string& path, LinkType type) const
{
linkToImpl(path, type);
}
void File::remove(bool recursive) void File::remove(bool recursive)
{ {
if (recursive && !isLink() && isDirectory()) if (recursive && !isLink() && isDirectory())

View File

@ -368,6 +368,23 @@ void FileImpl::renameToImpl(const std::string& path)
} }
void FileImpl::linkToImpl(const std::string& path, int type) const
{
poco_assert (!_path.empty());
if (type == 0)
{
if (link(_path.c_str(), path.c_str()) != 0)
handleLastErrorImpl(_path);
}
else
{
if (symlink(_path.c_str(), path.c_str()) != 0)
handleLastErrorImpl(_path);
}
}
void FileImpl::removeImpl() void FileImpl::removeImpl()
{ {
poco_assert (!_path.empty()); poco_assert (!_path.empty());

View File

@ -285,6 +285,12 @@ void FileImpl::renameToImpl(const std::string& path)
} }
void FileImpl::linkToImpl(const std::string& path, int type) const
{
throw Poco::NotImplementedException("File::linkTo() is not available on this platform");
}
void FileImpl::removeImpl() void FileImpl::removeImpl()
{ {
poco_assert (!_path.empty()); poco_assert (!_path.empty());

View File

@ -163,10 +163,16 @@ bool FileImpl::isDirectoryImpl() const
bool FileImpl::isLinkImpl() const bool FileImpl::isLinkImpl() const
{ {
return false; poco_assert (!_path.empty());
DWORD attr = GetFileAttributes(_upath.c_str());
if (attr == INVALID_FILE_ATTRIBUTES)
handleLastErrorImpl(_path);
return (attr & FILE_ATTRIBUTE_DIRECTORY) == 0 && (attr & FILE_ATTRIBUTE_REPARSE_POINT) != 0;
} }
bool FileImpl::isDeviceImpl() const bool FileImpl::isDeviceImpl() const
{ {
return return
@ -300,6 +306,23 @@ void FileImpl::renameToImpl(const std::string& path)
} }
void FileImpl::linkToImpl(const std::string& path, int type) const
{
poco_assert (!_path.empty());
if (type == 0)
{
if (CreateHardLinkA(path.c_str(), _path.c_str(), NULL) == 0)
handleLastErrorImpl(_path);
}
else
{
if (CreateSymbolicLinkA(path.c_str(), _path.c_str(), (isDirectoryImpl() ? SYMBOLIC_LINK_FLAG_DIRECTORY : 0) | SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE) == 0)
handleLastErrorImpl(_path);
}
}
void FileImpl::removeImpl() void FileImpl::removeImpl()
{ {
poco_assert (!_path.empty()); poco_assert (!_path.empty());

View File

@ -167,7 +167,12 @@ bool FileImpl::isDirectoryImpl() const
bool FileImpl::isLinkImpl() const bool FileImpl::isLinkImpl() const
{ {
return false; poco_assert (!_path.empty());
DWORD attr = GetFileAttributesW(_upath.c_str());
if (attr == INVALID_FILE_ATTRIBUTES)
handleLastErrorImpl(_path);
return (attr & FILE_ATTRIBUTE_DIRECTORY) == 0 && (attr & FILE_ATTRIBUTE_REPARSE_POINT) != 0;
} }
@ -308,6 +313,26 @@ void FileImpl::renameToImpl(const std::string& path)
} }
void FileImpl::linkToImpl(const std::string& path, int type) const
{
poco_assert (!_path.empty());
std::wstring upath;
convertPath(path, upath);
if (type == 0)
{
if (CreateHardLinkW(upath.c_str(), _upath.c_str(), NULL) == 0)
handleLastErrorImpl(_path);
}
else
{
if (CreateSymbolicLinkW(upath.c_str(), _upath.c_str(), (isDirectoryImpl() ? SYMBOLIC_LINK_FLAG_DIRECTORY : 0) | SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE) == 0)
handleLastErrorImpl(_path);
}
}
void FileImpl::removeImpl() void FileImpl::removeImpl()
{ {
poco_assert (!_path.empty()); poco_assert (!_path.empty());

View File

@ -299,6 +299,12 @@ void FileImpl::renameToImpl(const std::string& path)
} }
void FileImpl::linkToImpl(const std::string& path, int type) const
{
throw Poco::NotImplementedException("File::linkTo() is not available on this platform");
}
void FileImpl::removeImpl() void FileImpl::removeImpl()
{ {
poco_assert (!_path.empty()); poco_assert (!_path.empty());