1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
// file : libbuild2/filesystem.txx -*- C++ -*-
// license : MIT; see accompanying LICENSE file
#include <type_traits> // is_base_of
#include <libbuild2/diagnostics.hxx>
namespace build2
{
template <typename T>
fs_status<butl::rmfile_status>
rmfile (context& ctx, const path& f, const T& t, uint16_t v)
{
using namespace butl;
// We don't want to print the command if we couldn't remove the file
// because it does not exist (just like we don't print the update command
// if the file is up to date). This makes the below code a bit ugly.
//
auto print = [&f, &t, v] ()
{
if (verb >= v)
{
if (verb >= 2)
text << "rm " << f;
else if (verb)
text << "rm " << t;
}
};
rmfile_status rs;
try
{
rs = ctx.dry_run
? file_exists (f) ? rmfile_status::success : rmfile_status::not_exist
: try_rmfile (f);
}
catch (const system_error& e)
{
print ();
fail << "unable to remove file " << f << ": " << e << endf;
}
if (rs == rmfile_status::success)
print ();
return rs;
}
template <typename T>
fs_status<butl::rmdir_status>
rmdir (context& ctx, const dir_path& d, const T& t, uint16_t v)
{
using namespace butl;
// We don't want to print the command if we couldn't remove the directory
// because it does not exist (just like we don't print mkdir if it already
// exists) or if it is not empty. This makes the below code a bit ugly.
//
auto print = [&d, &t, v] ()
{
if (verb >= v)
{
if (verb >= 2)
text << "rmdir " << d;
else if (verb)
text << (std::is_base_of<dir_path, T>::value ? "rmdir " : "rm ") << t;
}
};
bool w (false); // Don't try to remove working directory.
rmdir_status rs;
try
{
rs = ctx.dry_run
? dir_exists (d) ? rmdir_status::success : rmdir_status::not_exist
: !(w = work.sub (d)) ? try_rmdir (d) : rmdir_status::not_empty;
}
catch (const system_error& e)
{
print ();
fail << "unable to remove directory " << d << ": " << e << endf;
}
switch (rs)
{
case rmdir_status::success:
{
print ();
break;
}
case rmdir_status::not_empty:
{
if (verb >= v && verb >= 2)
{
text << d << " is "
<< (w ? "current working directory" : "not empty")
<< ", not removing";
}
break;
}
case rmdir_status::not_exist:
break;
}
return rs;
}
}
|