Reworked copy operation. Added create_directory with two paths.
The updated copy implementation follows C++20 [fs.op.copy] definition and implements additional copy options. The new create_directory overloads accepting two paths are functionally similar to copy_directory, but are compliant with C++20 [fs.op.create.directory]. The copy_directory operation has been deprecated in favor of create_directory.
This commit is contained in:
parent
4e6317e4b0
commit
80709a9411
@ -293,6 +293,16 @@ BOOST_FILESYSTEM_NO_DEPRECATED</code> is defined.</p>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
<i>Function removed</i></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
<code>operations.hpp</code></td>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
<code>copy_directory(const path& from, const path& to)</code></td>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
</td>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
<i>Function removed, use <code>create_directory(const path& to, const path& from)</code> instead (note the reversed order of arguments)</i></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style="font-size: 10pt" valign="top">
|
||||
<code>class directory_entry</code></td>
|
||||
|
@ -774,9 +774,17 @@ nothing else."</p>
|
||||
enum class <a name="copy_options">copy_options</a>
|
||||
{
|
||||
none = 0u,
|
||||
// <a href="#copy_file">copy_file</a> options
|
||||
skip_existing,
|
||||
overwrite_existing,
|
||||
update_existing
|
||||
update_existing,
|
||||
// <a href="#copy">copy</a> options
|
||||
recursive,
|
||||
copy_symlinks,
|
||||
skip_symlinks,
|
||||
directories_only,
|
||||
create_symlinks,
|
||||
create_hard_links
|
||||
};
|
||||
|
||||
// Deprecated, use <a href="#copy_options">copy_options</a> instead
|
||||
@ -815,9 +823,14 @@ nothing else."</p>
|
||||
void <a href="#copy">copy</a>(const path& from, const path& to);
|
||||
void <a href="#copy">copy</a>(const path& from, const path& to,
|
||||
system::error_code& ec);
|
||||
void <a href="#copy">copy</a>(const path& from, const path& to,
|
||||
<a href="#copy_options">copy_options</a> options);
|
||||
void <a href="#copy">copy</a>(const path& from, const path& to,
|
||||
<a href="#copy_options">copy_options</a> options, system::error_code& ec);
|
||||
|
||||
void <a href="#create_directory">copy_directory</a>(const path& from, const path& to);
|
||||
void <a href="#create_directory">copy_directory</a>(const path& from, const path& to,
|
||||
// Deprecated, use <a href="#create_directory">create_directory</a> instead
|
||||
void <a href="#copy_directory">copy_directory</a>(const path& from, const path& to);
|
||||
void <a href="#copy_directory">copy_directory</a>(const path& from, const path& to,
|
||||
system::error_code& ec);
|
||||
|
||||
bool <a href="#copy_file">copy_file</a>(const path& from, const path& to);
|
||||
@ -844,6 +857,8 @@ nothing else."</p>
|
||||
|
||||
bool <a href="#create_directory">create_directory</a>(const path& p);
|
||||
bool <a href="#create_directory">create_directory</a>(const path& p, system::error_code& ec);
|
||||
bool <a href="#create_directory">create_directory</a>(const path& p, const path& existing);
|
||||
bool <a href="#create_directory">create_directory</a>(const path& p, const path& existing, system::error_code& ec);
|
||||
|
||||
void <a href="#create_directory_symlink">create_directory_symlink</a>(const path& to,
|
||||
const path& new_symlink);
|
||||
@ -3080,58 +3095,87 @@ without a <code>base</code> argument, <code>base</code> is <code>current_path()<
|
||||
does this path live in /home/goodguy or /home/badguy?) —end note]</p>
|
||||
|
||||
</blockquote>
|
||||
<pre>void <a name="copy">copy</a>(const path& from, const path& to);
|
||||
<pre>void copy(const path& from, const path& to);
|
||||
void copy(const path& from, const path& to, system::error_code& ec);</pre>
|
||||
<blockquote>
|
||||
<p><i>Effects:</i> As if</p>
|
||||
<p><i>Effects: </i><code>copy(from, to, copy_options::none</code><i>[</i><code>, ec</code><i>]</i><code>)</code>.</p>
|
||||
<p><i>Throws:</i> As specified in <a href="#Error-reporting">Error reporting</a>.</p>
|
||||
</blockquote>
|
||||
<pre>void <a name="copy">copy</a>(const path& from, const path& to, <a href="#copy_options">copy_options</a> options);
|
||||
void copy(const path& from, const path& to, <a href="#copy_options">copy_options</a> options, system::error_code& ec);</pre>
|
||||
<blockquote>
|
||||
<p><i>Precondition:</i> <code>options</code> must contain at most one option from each of the following groups:
|
||||
<ul>
|
||||
<li><code>copy_options::skip_existing</code>, <code>copy_options::overwrite_existing</code> or <code>copy_options::update_existing</code>;</li>
|
||||
<li><code>copy_options::recursive</code>;</li>
|
||||
<li><code>copy_options::copy_symlinks</code> or <code>copy_options::skip_symlinks</code>;</li>
|
||||
<li><code>copy_options::directories_only</code>, <code>copy_options::create_symlinks</code> or <code>copy_options::create_hard_links</code>.</li>
|
||||
</ul></p>
|
||||
<p><i>Effects:</i> Let <code>f</code> and <code>t</code> be <code>file_status</code> objects obtained the following way:
|
||||
<ul>
|
||||
<li>If <code>(options & (copy_options::create_symlinks | copy_options::skip_symlinks)) != copy_options::none</code> then <code>f = <a href="#symlink_status">symlink_status</a>(from)</code> and <code>t = <a href="#symlink_status">symlink_status</a>(to)</code>;</li>
|
||||
<li>Otherwise, if <code>(options & copy_options::copy_symlinks) != copy_options::none</code> then <code>f = <a href="#symlink_status">symlink_status</a>(from)</code> and <code>t = <a href="#status">status</a>(to)</code>;</li>
|
||||
<li>Otherwise, <code>f = <a href="#status">status</a>(from)</code> and <code>t = <a href="#status">status</a>(to)</code>.</li>
|
||||
</ul>
|
||||
Then, report an error if:
|
||||
<ul>
|
||||
<li><code>!exists(f)</code>, or</li>
|
||||
<li><code>equivalent(from, to)</code>, or</li>
|
||||
<li><code>is_other(f) || is_other(t)</code>, or</li>
|
||||
<li><code>is_directory(f) && is_regular_file(t)</code>.</li>
|
||||
</ul>
|
||||
Otherwise, if <code>is_symlink(f)</code>, then:
|
||||
<ul>
|
||||
<li>If <code>(options & copy_options::skip_symlinks) != copy_options::none</code> then return;</li>
|
||||
<li>Otherwise if <code>!exists(t) && (options & copy_options::copy_symlinks) != copy_options::none</code> then <code><a href="#copy_symlink">copy_symlink</a>(from, to)</code>;</li>
|
||||
<li>Otherwise report error.</li>
|
||||
</ul>
|
||||
Otherwise, if <code>is_regular_file(f)</code>, then:
|
||||
<ul>
|
||||
<li>If <code>(options & copy_options::directories_only) != copy_options::none</code> then return;</li>
|
||||
<li>Otherwise if <code>(options & copy_options::create_symlinks) != copy_options::none</code> then <code>create_symlink(from, to)</code>;</li>
|
||||
<li>otherwise if <code>(options & copy_options::create_hard_links) != copy_options::none</code> then <code>create_hard_link(from, to)</code>;</li>
|
||||
<li>Otherwise if <code>is_directory(t)</code> then <code><a href="#copy_file">copy_file</a>(from, to / from.filename(), options)</code>;</li>
|
||||
<li>Otherwise <code><a href="#copy_file">copy_file</a>(from, to, options)</code>.</li>
|
||||
</ul>
|
||||
Otherwise, if <code>is_directory(f)</code>, then:
|
||||
<ul>
|
||||
<li>If <code>(options & copy_options::create_symlinks) != copy_options::none</code> then report error with error code equal to <code>make_error_code(system::errc::is_a_directory)</code>;</li>
|
||||
<li>Otherwise if
|
||||
<ul>
|
||||
<li><code>(options & copy_options::recursive) != copy_options::none</code>, or</li>
|
||||
<li><code>options == copy_options::none</code> and this call to <code>copy</code> is not a recursive call from <code>copy</code></li>
|
||||
</ul>
|
||||
then:
|
||||
<ul>
|
||||
<li>If <code>!exists(t)</code>, then <code><a href="#create_directory">create_directory</a>(to, from)</code>.</li>
|
||||
<li>Then, iterate over files in <code>from</code> and for each <code>directory_entry x</code> obtained during iteration invoke <code>copy(x.path(), to / x.path().filename(), options)</code>.</li>
|
||||
</ul>
|
||||
</li>
|
||||
<li>Otherwise, return.</li>
|
||||
</ul>
|
||||
Otherwise, for all unsupported file types of <code>f</code> report error.</p>
|
||||
|
||||
<blockquote>
|
||||
<pre>file_status s(symlink_status(from<i>[</i><code>, ec</code><i>]</i>));
|
||||
if(is_symlink(s))
|
||||
copy_symlink(from, to<i>[</i><code>, ec</code><i>]</i>);
|
||||
else if(is_directory(s))
|
||||
copy_directory(from, to<i>[</i><code>, ec</code><i>]</i>);
|
||||
else if(is_regular_file(s))
|
||||
copy_file(from, to, copy_options::none<i>[</i><code>, ec</code><i>]</i>);
|
||||
else
|
||||
<i> Report error as specified in <a href="#Error-reporting">Error reporting</a>.</i></pre>
|
||||
</blockquote>
|
||||
<p><i>Throws:</i> As specified in <a href="#Error-reporting">Error reporting</a>.</p>
|
||||
|
||||
</blockquote>
|
||||
<pre>void <a name="copy_directory">copy_directory</a>(const path& from, const path& to);
|
||||
void copy_directory(const path& from, const path& to, system::error_code& ec);</pre>
|
||||
<blockquote>
|
||||
<table border="1" cellpadding="5" cellspacing="0" style="border-collapse: collapse"
|
||||
bordercolor="#111111" width="90%" bgcolor="#E0E0E0">
|
||||
<tr>
|
||||
<td width="100%">
|
||||
<p><i>This function is poorly named; it should probably be an overload of
|
||||
<code>create_directory</code> with an additional argument.</i></td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
<p><i>Effects: </i>Creates directory <code>to</code>, with
|
||||
attributes copied from directory <code>from</code>. The set of attributes
|
||||
copied is operating system dependent.</p>
|
||||
|
||||
<blockquote>
|
||||
<table border="1" cellpadding="5" cellspacing="0" style="border-collapse: collapse"
|
||||
bordercolor="#111111" width="90%" bgcolor="#E8FFE8">
|
||||
<tr>
|
||||
<td width="100%">
|
||||
<p>[<i>Note:</i> For ISO 9945/POSIX based operating systems the
|
||||
attributes are those copied by native API <code>stat(from.c_str(), &from_stat)</code>
|
||||
followed by <code>mkdir(to.c_str(),from_stat.st_mode)</code>. For
|
||||
Windows based operating systems the attributes are those copied by native
|
||||
API <code>CreateDirectoryExW(from.c_str(), to.c_str(), 0)</code>. <i>
|
||||
—end note</i>]</td>
|
||||
</tr>
|
||||
</table>
|
||||
</blockquote>
|
||||
<p>[<i>Note:</i> For ISO 9945/POSIX based operating systems the
|
||||
attributes are those copied by native API <code>stat(from.c_str(), &from_stat)</code>
|
||||
followed by <code>mkdir(to.c_str(),from_stat.st_mode)</code>. For
|
||||
Windows based operating systems the attributes are those copied by native
|
||||
API <code>CreateDirectoryExW(from.c_str(), to.c_str(), 0)</code>. <i>
|
||||
—end note</i>]</td>
|
||||
|
||||
<p><i>Throws:</i> As specified in <a href="#Error-reporting">Error reporting</a>.</p>
|
||||
|
||||
<p>[<i>Note:</i> This operation is deprecated, use <a href="#create_directory">create_directory</a> instead. —end note]</p>
|
||||
</blockquote>
|
||||
|
||||
<pre>bool copy_file(const path& from, const path& to);
|
||||
@ -3143,9 +3187,9 @@ bool copy_file(const path& from, const path& to, system::error_code&
|
||||
|
||||
</blockquote>
|
||||
<pre>bool <a name="copy_file">copy_file</a>(const path& from, const path& to, <a href="#copy_options">copy_options</a> options);
|
||||
bool <a name="copy_file2">copy_file</a>(const path& from, const path& to, <a href="#copy_options">copy_options</a> options, system::error_code& ec);
|
||||
bool <a name="copy_file3">copy_file</a>(const path& from, const path& to, <a href="#copy_option">copy_option</a> options);
|
||||
bool <a name="copy_file4">copy_file</a>(const path& from, const path& to, <a href="#copy_option">copy_option</a> options, system::error_code& ec);</pre>
|
||||
bool copy_file(const path& from, const path& to, <a href="#copy_options">copy_options</a> options, system::error_code& ec);
|
||||
bool copy_file(const path& from, const path& to, <a href="#copy_option">copy_option</a> options);
|
||||
bool copy_file(const path& from, const path& to, <a href="#copy_option">copy_option</a> options, system::error_code& ec);</pre>
|
||||
<blockquote>
|
||||
<p><i>Precondition:</i> At most one of <code>copy_options::skip_existing</code>, <code>copy_options::overwrite_existing</code> or <code>copy_options::update_existing</code> must be specified in <code>options</code>.</p>
|
||||
<p><i>Effects:</i> Report an error if:
|
||||
@ -3153,12 +3197,12 @@ bool <a name="copy_file4">copy_file</a>(const path& from, const path& to
|
||||
<li><code>!is_regular_file(from)</code>, or</li>
|
||||
<li><code>exists(to) && !is_regular_file(to)</code>, or</li>
|
||||
<li><code>exists(to) && equivalent(from, to)</code>, or</li>
|
||||
<li><code>exists(to) && (options & (copy_options::skip_existing | copy_options::overwrite_existing)) == copy_options::none</code>.</li>
|
||||
<li><code>exists(to) && (options & (copy_options::skip_existing | copy_options::overwrite_existing)) == copy_options::none</code>.</li>
|
||||
</ul>
|
||||
Otherwise, return successfully with no effect if:
|
||||
<ul>
|
||||
<li><code>exists(to) && (options & copy_options::skip_existing) != copy_options::none</code>, or</li>
|
||||
<li><code>exists(to) && (options & copy_options::update_existing) != copy_options::none</code> and last write time of <code>from</code> is more recent than that of <code>to</code>.</li>
|
||||
<li><code>exists(to) && (options & copy_options::skip_existing) != copy_options::none</code>, or</li>
|
||||
<li><code>exists(to) && (options & copy_options::update_existing) != copy_options::none</code> and last write time of <code>from</code> is more recent than that of <code>to</code>.</li>
|
||||
</ul>
|
||||
Otherwise, the contents and attributes of the file <code>from</code> resolves to are copied to the file <code>to</code> resolves to.</p>
|
||||
<p><i>Returns:</i> <code>true</code> if the file was copied without error, otherwise <code>false</code>.</p>
|
||||
@ -3188,13 +3232,17 @@ bool <a name="create_directories2">create_directories</a>(const path& p, sys
|
||||
of <code>p</code> that do not exist.</p>
|
||||
</blockquote>
|
||||
<pre>bool <a name="create_directory">create_directory</a>(const path& p);
|
||||
bool <a name="create_directory2">create_directory</a>(const path& p, system::error_code& ec);</pre>
|
||||
bool create_directory(const path& p, system::error_code& ec);
|
||||
bool create_directory(const path& p, const path& existing);
|
||||
bool create_directory(const path& p, const path& existing, system::error_code& ec);</pre>
|
||||
<blockquote>
|
||||
<p><i>Effects:</i> Establishes the postcondition by attempting to create the
|
||||
directory <code>p</code> resolves to, as if by ISO/IEC 9945 <code><a href="http://www.opengroup.org/onlinepubs/000095399/functions/mkdir.html">
|
||||
mkdir()</a></code> with a second argument of S_IRWXU|S_IRWXG|S_IRWXO. Creation
|
||||
failure because <code>p</code> resolves to an existing directory shall not be
|
||||
treated as an error. </p>
|
||||
mkdir()</a></code>. For overloads without <code>existing</code> argument, the new directory is created with <code>S_IRWXU|S_IRWXG|S_IRWXO</code> mode.
|
||||
Overloads with <code>existing</code> argument obtain mode from <code>existing</code>, which must be a path to an existing directory. On Windows,
|
||||
<code>CreateDirectoryW(p.c_str(), NULL)</code> is used when <code>existing</code> is not specified and <code>CreateDirectoryExW(existing.c_str(), p.c_str(), NULL)</code> otherwise.
|
||||
Creation failure because <code>p</code> resolves to an existing directory shall not be
|
||||
treated as an error.</p>
|
||||
<p><i>Postcondition:</i> <code>is_directory(p)</code></p>
|
||||
<p><i>Returns:</i> <code>true</code> if a new directory was created, otherwise <code>false</code>.</p>
|
||||
<p><i>Throws:</i> As specified in <a href="#Error-reporting">Error reporting</a>.</p>
|
||||
|
@ -34,7 +34,7 @@
|
||||
<a href="v3.html">V3 Intro</a>
|
||||
<a href="v3_design.html">V3 Design</a>
|
||||
<a href="deprecated.html">Deprecated</a>
|
||||
<a href="issue_reporting.html">Bug Reports </a>
|
||||
<a href="issue_reporting.html">Bug Reports</a>
|
||||
</td>
|
||||
</table>
|
||||
|
||||
@ -50,6 +50,9 @@
|
||||
<li><b>New:</b> Added <code>copy_options::skip_existing</code> option, which allows <code>copy_file</code> operation to succeed without overwriting the target file, if it exists.</li>
|
||||
<li><b>New:</b> Added <code>copy_options::update_existing</code> option, which allows <code>copy_file</code> operation to conditionally overwrite the target file, if it exists, if its last write time is older than that of the replacement file.</li>
|
||||
<li><b>New:</b> <code>copy_file</code> now returns <code>bool</code>, which indicates whether the file was copied.</li>
|
||||
<li><b>New, breaking change:</b> <code>copy</code> operation has been extended and reworked to implement behavior specified in C++20 [fs.op.copy]. This includes support for <code>copy_options::recursive</code>, <code>copy_options::copy_symlinks</code>, <code>copy_options::skip_symlinks</code>, <code>copy_options::directories_only</code>, <code>copy_options::create_symlinks</code> and <code>copy_options::create_hard_links</code> options. The operation performs additional checks based on the specified options. Applying <code>copy</code> to a directory with default <code>copy_options</code> will now also copy files residing in that directory (but not nested directories or files in those directories).</li>
|
||||
<li><b>New:</b> Added <code>create_directory</code> overload taking two paths. The second path is a path to an existing directory, which is used as a source of permission attributes to use in the directory to create.</li>
|
||||
<li><b>Deprecated:</b> <code>copy_directory</code> operation has been deprecated in favor of the new <code>create_directory</code> overload. Note that the two operations have reversed order of the path arguments.</li>
|
||||
<li><code>equivalent</code> on POSIX systems now returns the actual error code from the OS if one of the paths does not resolve to a file. Previously the function would return an error code of 1. (<a href="https://github.com/boostorg/filesystem/issues/141">#141</a>)</li>
|
||||
<li><code>equivalent</code> no longer considers file size and last modification time in order to test whether the two paths refer to the same file. These checks could result in a false negative if the file was modified during the <code>equivalent</code> call.</li>
|
||||
<li><code>space</code> now initializes the <code>space_info</code> structure members to -1 values on error, as required by C++20 ([fs.op.space]/1).</li>
|
||||
|
@ -55,10 +55,21 @@ struct space_info
|
||||
|
||||
BOOST_SCOPED_ENUM_UT_DECLARE_BEGIN(copy_options, unsigned int)
|
||||
{
|
||||
none = 0u, // Default, error if the target file exists
|
||||
none = 0u, // Default. For copy_file: error if the target file exists. For copy: do not recurse, follow symlinks, copy file contents.
|
||||
|
||||
// copy_file options:
|
||||
skip_existing = 1u, // Don't overwrite the existing target file, don't report an error
|
||||
overwrite_existing = 1u << 1, // Overwrite existing file
|
||||
update_existing = 1u << 2 // Overwrite existing file if its last write time is older than the replacement file
|
||||
update_existing = 1u << 2, // Overwrite existing file if its last write time is older than the replacement file
|
||||
|
||||
// copy options:
|
||||
recursive = 1u << 8, // Recurse into sub-directories
|
||||
copy_symlinks = 1u << 9, // Copy symlinks as symlinks instead of copying the referenced file
|
||||
skip_symlinks = 1u << 10, // Don't copy symlinks
|
||||
directories_only = 1u << 11, // Only copy directory structure, do not copy non-directory files
|
||||
create_symlinks = 1u << 12, // Create symlinks instead of copying files
|
||||
create_hard_links = 1u << 13, // Create hard links instead of copying files
|
||||
_detail_recursing = 1u << 14 // Internal use only, do not use
|
||||
}
|
||||
BOOST_SCOPED_ENUM_DECLARE_END(copy_options)
|
||||
|
||||
@ -91,9 +102,11 @@ path initial_path(system::error_code* ec=0);
|
||||
BOOST_FILESYSTEM_DECL
|
||||
path canonical(const path& p, const path& base, system::error_code* ec=0);
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void copy(const path& from, const path& to, system::error_code* ec=0);
|
||||
void copy(const path& from, const path& to, unsigned int options, system::error_code* ec=0);
|
||||
#if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void copy_directory(const path& from, const path& to, system::error_code* ec=0);
|
||||
#endif
|
||||
BOOST_FILESYSTEM_DECL
|
||||
bool copy_file(const path& from, const path& to, // See ticket #2925
|
||||
unsigned int options, system::error_code* ec=0); // see copy_options for options
|
||||
@ -102,7 +115,7 @@ void copy_symlink(const path& existing_symlink, const path& new_symlink, system:
|
||||
BOOST_FILESYSTEM_DECL
|
||||
bool create_directories(const path& p, system::error_code* ec=0);
|
||||
BOOST_FILESYSTEM_DECL
|
||||
bool create_directory(const path& p, system::error_code* ec=0);
|
||||
bool create_directory(const path& p, const path* existing, system::error_code* ec=0);
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void create_directory_symlink(const path& to, const path& from,
|
||||
system::error_code* ec=0);
|
||||
@ -247,17 +260,34 @@ path complete(const path& p, const path& base)
|
||||
#endif
|
||||
|
||||
inline
|
||||
void copy(const path& from, const path& to) {detail::copy(from, to);}
|
||||
|
||||
void copy(const path& from, const path& to)
|
||||
{
|
||||
detail::copy(from, to, static_cast< unsigned int >(copy_options::none));
|
||||
}
|
||||
inline
|
||||
void copy(const path& from, const path& to, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{detail::copy(from, to, &ec);}
|
||||
{
|
||||
detail::copy(from, to, static_cast< unsigned int >(copy_options::none), &ec);
|
||||
}
|
||||
inline
|
||||
void copy(const path& from, const path& to, BOOST_SCOPED_ENUM_NATIVE(copy_options) options)
|
||||
{
|
||||
detail::copy(from, to, static_cast< unsigned int >(options));
|
||||
}
|
||||
inline
|
||||
void copy(const path& from, const path& to, BOOST_SCOPED_ENUM_NATIVE(copy_options) options, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{
|
||||
detail::copy(from, to, static_cast< unsigned int >(options), &ec);
|
||||
}
|
||||
|
||||
#if !defined(BOOST_FILESYSTEM_NO_DEPRECATED)
|
||||
inline
|
||||
void copy_directory(const path& from, const path& to)
|
||||
{detail::copy_directory(from, to);}
|
||||
inline
|
||||
void copy_directory(const path& from, const path& to, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{detail::copy_directory(from, to, &ec);}
|
||||
#endif
|
||||
inline
|
||||
bool copy_file(const path& from, const path& to)
|
||||
{
|
||||
@ -309,11 +339,17 @@ inline
|
||||
bool create_directories(const path& p, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{return detail::create_directories(p, &ec);}
|
||||
inline
|
||||
bool create_directory(const path& p) {return detail::create_directory(p);}
|
||||
bool create_directory(const path& p) {return detail::create_directory(p, 0);}
|
||||
|
||||
inline
|
||||
bool create_directory(const path& p, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{return detail::create_directory(p, &ec);}
|
||||
{return detail::create_directory(p, 0, &ec);}
|
||||
inline
|
||||
bool create_directory(const path& p, const path& existing)
|
||||
{return detail::create_directory(p, &existing);}
|
||||
inline
|
||||
bool create_directory(const path& p, const path& existing, system::error_code& ec) BOOST_NOEXCEPT
|
||||
{return detail::create_directory(p, &existing, &ec);}
|
||||
inline
|
||||
void create_directory_symlink(const path& to, const path& from)
|
||||
{detail::create_directory_symlink(to, from);}
|
||||
|
@ -34,8 +34,9 @@ typedef int err_t;
|
||||
// POSIX uses a 0 return to indicate success
|
||||
#define BOOST_ERRNO errno
|
||||
|
||||
#define BOOST_ERROR_NOT_SUPPORTED ENOSYS
|
||||
#define BOOST_ERROR_FILE_NOT_FOUND ENOENT
|
||||
#define BOOST_ERROR_ALREADY_EXISTS EEXIST
|
||||
#define BOOST_ERROR_NOT_SUPPORTED ENOSYS
|
||||
|
||||
#else
|
||||
|
||||
@ -44,6 +45,7 @@ typedef boost::winapi::DWORD_ err_t;
|
||||
// Windows uses a non-0 return to indicate success
|
||||
#define BOOST_ERRNO boost::winapi::GetLastError()
|
||||
|
||||
#define BOOST_ERROR_FILE_NOT_FOUND boost::winapi::ERROR_FILE_NOT_FOUND_
|
||||
#define BOOST_ERROR_ALREADY_EXISTS boost::winapi::ERROR_ALREADY_EXISTS_
|
||||
#define BOOST_ERROR_NOT_SUPPORTED boost::winapi::ERROR_NOT_SUPPORTED_
|
||||
|
||||
|
@ -256,7 +256,6 @@ union reparse_data_buffer
|
||||
# if defined(BOOST_POSIX_API)
|
||||
|
||||
# define BOOST_SET_CURRENT_DIRECTORY(P)(::chdir(P)== 0)
|
||||
# define BOOST_CREATE_DIRECTORY(P)(::mkdir(P, S_IRWXU|S_IRWXG|S_IRWXO)== 0)
|
||||
# define BOOST_CREATE_HARD_LINK(F,T)(::link(T, F)== 0)
|
||||
# define BOOST_CREATE_SYMBOLIC_LINK(F,T,Flag)(::symlink(T, F)== 0)
|
||||
# define BOOST_REMOVE_DIRECTORY(P)(::rmdir(P)== 0)
|
||||
@ -269,7 +268,6 @@ union reparse_data_buffer
|
||||
# else // BOOST_WINDOWS_API
|
||||
|
||||
# define BOOST_SET_CURRENT_DIRECTORY(P)(::SetCurrentDirectoryW(P)!= 0)
|
||||
# define BOOST_CREATE_DIRECTORY(P)(::CreateDirectoryW(P, 0)!= 0)
|
||||
# define BOOST_CREATE_HARD_LINK(F,T)(create_hard_link_api(F, T, 0)!= 0)
|
||||
# define BOOST_CREATE_SYMBOLIC_LINK(F,T,Flag)(create_symbolic_link_api(F, T, Flag)!= 0)
|
||||
# define BOOST_REMOVE_DIRECTORY(P)(::RemoveDirectoryW(P)!= 0)
|
||||
@ -304,8 +302,9 @@ fs::file_type query_file_type(const path& p, error_code* ec);
|
||||
|
||||
bool is_empty_directory(const path& p, error_code* ec)
|
||||
{
|
||||
return (ec != 0 ? fs::directory_iterator(p, *ec) : fs::directory_iterator(p))
|
||||
== fs::directory_iterator();
|
||||
fs::directory_iterator itr;
|
||||
detail::directory_iterator_construct(itr, p, static_cast< unsigned int >(directory_options::none), ec);
|
||||
return itr == fs::directory_iterator();
|
||||
}
|
||||
|
||||
bool not_found_error(int errval) BOOST_NOEXCEPT; // forward declaration
|
||||
@ -356,19 +355,14 @@ bool remove_file_or_directory(const path& p, fs::file_type type, error_code* ec)
|
||||
boost::uintmax_t remove_all_aux(const path& p, fs::file_type type,
|
||||
error_code* ec)
|
||||
{
|
||||
boost::uintmax_t count = 0;
|
||||
boost::uintmax_t count = 0u;
|
||||
|
||||
if (type == fs::directory_file) // but not a directory symlink
|
||||
{
|
||||
fs::directory_iterator itr;
|
||||
if (ec != 0)
|
||||
{
|
||||
itr = fs::directory_iterator(p, *ec);
|
||||
if (*ec)
|
||||
return count;
|
||||
}
|
||||
else
|
||||
itr = fs::directory_iterator(p);
|
||||
fs::detail::directory_iterator_construct(itr, p, static_cast< unsigned int >(directory_options::none), ec);
|
||||
if (ec && *ec)
|
||||
return count;
|
||||
|
||||
const fs::directory_iterator end_dit;
|
||||
while(itr != end_dit)
|
||||
@ -970,40 +964,145 @@ path canonical(const path& p, const path& base, system::error_code* ec)
|
||||
}
|
||||
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void copy(const path& from, const path& to, system::error_code* ec)
|
||||
void copy(const path& from, const path& to, unsigned int options, system::error_code* ec)
|
||||
{
|
||||
file_status s(detail::symlink_status(from, ec));
|
||||
if (ec != 0 && *ec) return;
|
||||
BOOST_ASSERT((((options & static_cast< unsigned int >(copy_options::overwrite_existing)) != 0u) +
|
||||
((options & static_cast< unsigned int >(copy_options::skip_existing)) != 0u) +
|
||||
((options & static_cast< unsigned int >(copy_options::update_existing)) != 0u)) <= 1u);
|
||||
|
||||
if (is_symlink(s))
|
||||
BOOST_ASSERT((((options & static_cast< unsigned int >(copy_options::copy_symlinks)) != 0u) +
|
||||
((options & static_cast< unsigned int >(copy_options::skip_symlinks)) != 0u)) <= 1u);
|
||||
|
||||
BOOST_ASSERT((((options & static_cast< unsigned int >(copy_options::directories_only)) != 0u) +
|
||||
((options & static_cast< unsigned int >(copy_options::create_symlinks)) != 0u) +
|
||||
((options & static_cast< unsigned int >(copy_options::create_hard_links)) != 0u)) <= 1u);
|
||||
|
||||
file_status from_stat;
|
||||
if ((options & (static_cast< unsigned int >(copy_options::copy_symlinks) |
|
||||
static_cast< unsigned int >(copy_options::skip_symlinks) |
|
||||
static_cast< unsigned int >(copy_options::create_symlinks))) != 0u)
|
||||
{
|
||||
detail::copy_symlink(from, to, ec);
|
||||
}
|
||||
else if (is_directory(s))
|
||||
{
|
||||
detail::copy_directory(from, to, ec);
|
||||
}
|
||||
else if (is_regular_file(s))
|
||||
{
|
||||
detail::copy_file(from, to, static_cast< unsigned int >(copy_options::none), ec);
|
||||
from_stat = detail::symlink_status(from, ec);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (ec == 0)
|
||||
BOOST_FILESYSTEM_THROW(filesystem_error("boost::filesystem::copy",
|
||||
from, to, error_code(BOOST_ERROR_NOT_SUPPORTED, system_category())));
|
||||
ec->assign(BOOST_ERROR_NOT_SUPPORTED, system_category());
|
||||
from_stat = detail::status(from, ec);
|
||||
}
|
||||
}
|
||||
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void copy_directory(const path& from, const path& to, system::error_code* ec)
|
||||
{
|
||||
# ifdef BOOST_POSIX_API
|
||||
struct stat from_stat;
|
||||
# endif
|
||||
error(!BOOST_COPY_DIRECTORY(from.c_str(), to.c_str()) ? BOOST_ERRNO : 0,
|
||||
from, to, ec, "boost::filesystem::copy_directory");
|
||||
if (ec && *ec)
|
||||
return;
|
||||
|
||||
if (!exists(from_stat))
|
||||
{
|
||||
emit_error(BOOST_ERROR_FILE_NOT_FOUND, from, to, ec, "boost::filesystem::copy");
|
||||
return;
|
||||
}
|
||||
|
||||
if (is_symlink(from_stat))
|
||||
{
|
||||
if ((options & static_cast< unsigned int >(copy_options::skip_symlinks)) != 0u)
|
||||
return;
|
||||
|
||||
if ((options & static_cast< unsigned int >(copy_options::copy_symlinks)) == 0u)
|
||||
goto fail;
|
||||
|
||||
detail::copy_symlink(from, to, ec);
|
||||
}
|
||||
else if (is_regular_file(from_stat))
|
||||
{
|
||||
if ((options & static_cast< unsigned int >(copy_options::directories_only)) != 0u)
|
||||
return;
|
||||
|
||||
if ((options & static_cast< unsigned int >(copy_options::create_symlinks)) != 0u)
|
||||
{
|
||||
detail::create_symlink(from, to, ec);
|
||||
return;
|
||||
}
|
||||
|
||||
if ((options & static_cast< unsigned int >(copy_options::create_hard_links)) != 0u)
|
||||
{
|
||||
detail::create_hard_link(from, to, ec);
|
||||
return;
|
||||
}
|
||||
|
||||
file_status to_stat;
|
||||
if ((options & (static_cast< unsigned int >(copy_options::skip_symlinks) |
|
||||
static_cast< unsigned int >(copy_options::create_symlinks))) != 0u)
|
||||
{
|
||||
to_stat = detail::symlink_status(to, ec);
|
||||
}
|
||||
else
|
||||
{
|
||||
to_stat = detail::status(to, ec);
|
||||
}
|
||||
|
||||
if (ec && *ec)
|
||||
return;
|
||||
|
||||
if (is_directory(to_stat))
|
||||
detail::copy_file(from, to / from.filename(), options, ec);
|
||||
else
|
||||
detail::copy_file(from, to, options, ec);
|
||||
}
|
||||
else if (is_directory(from_stat))
|
||||
{
|
||||
if ((options & static_cast< unsigned int >(copy_options::create_symlinks)) != 0u)
|
||||
{
|
||||
error_code err_code = make_error_code(system::errc::is_a_directory);
|
||||
if (!ec)
|
||||
BOOST_FILESYSTEM_THROW(filesystem_error("boost::filesystem::copy", from, to, err_code));
|
||||
*ec = err_code;
|
||||
return;
|
||||
}
|
||||
|
||||
file_status to_stat;
|
||||
if ((options & (static_cast< unsigned int >(copy_options::skip_symlinks) |
|
||||
static_cast< unsigned int >(copy_options::create_symlinks))) != 0u)
|
||||
{
|
||||
to_stat = detail::symlink_status(to, ec);
|
||||
}
|
||||
else
|
||||
{
|
||||
to_stat = detail::status(to, ec);
|
||||
}
|
||||
|
||||
if (ec && *ec)
|
||||
return;
|
||||
|
||||
if (!exists(to_stat))
|
||||
{
|
||||
detail::create_directory(to, &from, ec);
|
||||
if (ec && *ec)
|
||||
return;
|
||||
}
|
||||
|
||||
if ((options & static_cast< unsigned int >(copy_options::recursive)) != 0u || options == 0u)
|
||||
{
|
||||
fs::directory_iterator itr;
|
||||
detail::directory_iterator_construct(itr, from, static_cast< unsigned int >(directory_options::none), ec);
|
||||
if (ec && *ec)
|
||||
return;
|
||||
|
||||
const fs::directory_iterator end_dit;
|
||||
while (itr != end_dit)
|
||||
{
|
||||
path const& p = itr->path();
|
||||
// Set _detail_recursing flag so that we don't recurse more than for one level deeper into the directory if options are copy_options::none
|
||||
detail::copy(p, to / p.filename(), options | static_cast< unsigned int >(copy_options::_detail_recursing), ec);
|
||||
if (ec && *ec)
|
||||
return;
|
||||
|
||||
detail::directory_iterator_increment(itr, ec);
|
||||
if (ec && *ec)
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
fail:
|
||||
emit_error(BOOST_ERROR_NOT_SUPPORTED, from, to, ec, "boost::filesystem::copy");
|
||||
}
|
||||
}
|
||||
|
||||
BOOST_FILESYSTEM_DECL
|
||||
@ -1033,7 +1132,7 @@ bool copy_file(const path& from, const path& to, unsigned int options, error_cod
|
||||
continue;
|
||||
|
||||
fail:
|
||||
error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
emit_error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -1190,7 +1289,7 @@ bool copy_file(const path& from, const path& to, unsigned int options, error_cod
|
||||
{
|
||||
fail_last_error:
|
||||
DWORD err = ::GetLastError();
|
||||
error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
emit_error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -1222,7 +1321,7 @@ bool copy_file(const path& from, const path& to, unsigned int options, error_cod
|
||||
DWORD err = ::GetLastError();
|
||||
if ((err == ERROR_FILE_EXISTS || err == ERROR_ALREADY_EXISTS) && (options & static_cast< unsigned int >(copy_options::skip_existing)) != 0u)
|
||||
return false;
|
||||
error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
emit_error(err, from, to, ec, "boost::filesystem::copy_file");
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -1291,40 +1390,75 @@ bool create_directories(const path& p, system::error_code* ec)
|
||||
}
|
||||
|
||||
// create the directory
|
||||
return create_directory(p, ec);
|
||||
return create_directory(p, NULL, ec);
|
||||
}
|
||||
|
||||
BOOST_FILESYSTEM_DECL
|
||||
bool create_directory(const path& p, error_code* ec)
|
||||
bool create_directory(const path& p, const path* existing, error_code* ec)
|
||||
{
|
||||
if (BOOST_CREATE_DIRECTORY(p.c_str()))
|
||||
if (ec)
|
||||
ec->clear();
|
||||
|
||||
#if defined(BOOST_POSIX_API)
|
||||
|
||||
mode_t mode = S_IRWXU | S_IRWXG | S_IRWXO;
|
||||
if (existing)
|
||||
{
|
||||
if (ec != 0)
|
||||
ec->clear();
|
||||
return true;
|
||||
struct ::stat existing_stat = {};
|
||||
if (::stat(existing->c_str(), &existing_stat) < 0)
|
||||
{
|
||||
emit_error(errno, p, *existing, ec, "boost::filesystem::create_directory");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!S_ISDIR(existing_stat.st_mode))
|
||||
{
|
||||
emit_error(ENOTDIR, p, *existing, ec, "boost::filesystem::create_directory");
|
||||
return false;
|
||||
}
|
||||
|
||||
mode = existing_stat.st_mode;
|
||||
}
|
||||
|
||||
if (::mkdir(p.c_str(), mode) == 0)
|
||||
return true;
|
||||
|
||||
#else // defined(BOOST_POSIX_API)
|
||||
|
||||
BOOL res;
|
||||
if (existing)
|
||||
res = ::CreateDirectoryExW(existing->c_str(), p.c_str(), NULL);
|
||||
else
|
||||
res = ::CreateDirectoryW(p.c_str(), NULL);
|
||||
|
||||
if (res)
|
||||
return true;
|
||||
|
||||
#endif // defined(BOOST_POSIX_API)
|
||||
|
||||
// attempt to create directory failed
|
||||
int errval(BOOST_ERRNO); // save reason for failure
|
||||
err_t errval = BOOST_ERRNO; // save reason for failure
|
||||
error_code dummy;
|
||||
|
||||
if (is_directory(p, dummy))
|
||||
{
|
||||
if (ec != 0)
|
||||
ec->clear();
|
||||
return false;
|
||||
}
|
||||
|
||||
// attempt to create directory failed && it doesn't already exist
|
||||
if (ec == 0)
|
||||
BOOST_FILESYSTEM_THROW(filesystem_error("boost::filesystem::create_directory",
|
||||
p, error_code(errval, system_category())));
|
||||
else
|
||||
ec->assign(errval, system_category());
|
||||
|
||||
emit_error(errval, p, ec, "boost::filesystem::create_directory");
|
||||
return false;
|
||||
}
|
||||
|
||||
// Deprecated, to be removed in a future release
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void copy_directory(const path& from, const path& to, system::error_code* ec)
|
||||
{
|
||||
# ifdef BOOST_POSIX_API
|
||||
struct stat from_stat;
|
||||
# endif
|
||||
error(!BOOST_COPY_DIRECTORY(from.c_str(), to.c_str()) ? BOOST_ERRNO : 0,
|
||||
from, to, ec, "boost::filesystem::copy_directory");
|
||||
}
|
||||
|
||||
BOOST_FILESYSTEM_DECL
|
||||
void create_directory_symlink(const path& to, const path& from,
|
||||
system::error_code* ec)
|
||||
@ -1401,11 +1535,7 @@ path current_path(error_code* ec)
|
||||
{
|
||||
if (BOOST_UNLIKELY(path_max > absolute_path_max))
|
||||
{
|
||||
if (ec == 0)
|
||||
BOOST_FILESYSTEM_THROW(filesystem_error("boost::filesystem::current_path",
|
||||
error_code(ENAMETOOLONG, system_category())));
|
||||
else
|
||||
ec->assign(ENAMETOOLONG, system_category());
|
||||
emit_error(ENAMETOOLONG, ec, "boost::filesystem::current_path");
|
||||
break;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user