XTL  0.1
eXtended Template Library
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
filesystem.hpp
Go to the documentation of this file.
1 
6 #pragma once
7 
8 #include <xtd/xtd.hpp>
9 
10 #include <xtd/string.hpp>
11 
12 #if ((XTD_OS_CYGWIN | XTD_OS_MSYS | XTD_OS_LINUX) & XTD_OS)
13  #include <sys/stat.h>
14  #include <paths.h>
15 #else
16  #include <xtd/exception.hpp>
17  #include <xtd/meta.hpp>
18 #endif
19 
20 #if (XTD_HAS_FILESYSTEM)
21  #include <filesystem>
22 #elif (XTD_HAS_EXP_FILESYSTEM)
23  #include <experimental/filesystem>
24 #endif
25 
26 namespace xtd{
27  namespace filesystem{
28 
29  namespace _{
30  template <typename, typename> struct path_adder;
31  }
32  }
33 }
34 
35 #if (XTD_HAS_EXP_FILESYSTEM)
36 namespace xtd{
37  namespace filesystem{
38 
39  struct path : std::experimental::filesystem::path{
40  using _super_t = std::experimental::filesystem::path;
41  template <typename ... _ArgTs> path(_ArgTs&&...oArgs) : _super_t(std::forward<_ArgTs>(oArgs)...){}
42 
43  template <typename _Ty> inline path operator+(const _Ty& src) const{
44  path oRet(*this);
45  oRet /= src;
46  return oRet;
47  }
48 
49  path filename() const{ return path(_super_t::filename()); }
50 
51  };
52 
53  static inline bool remove(const path& src){ return std::experimental::filesystem::remove(src); }
54 
55  static inline path temp_directory_path(){ return path(std::experimental::filesystem::temp_directory_path()); }
56  #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
57  static inline path home_directory_path(){
58  PWSTR sTemp;
59  xtd::windows::exception::throw_if(SHGetKnownFolderPath(FOLDERID_Profile, 0, nullptr, &sTemp), [](HRESULT hr){ return FAILED(hr); });
60  RAII(CoTaskMemFree(sTemp));
61  return xtd::string::format(static_cast<const wchar_t*>(sTemp));
62  }
63  #else
64  static inline path home_directory_path() { return path(getenv("HOME")); }
65  #endif
66 
67  }
68 }
69 #elif (XTD_HAS_FILESYSTEM)
70 namespace xtd{
71  namespace filesystem{
72  using namespace std::filesystem;
73  }
74 }
75 #else
76 
77 namespace xtd{
78  namespace filesystem{
79 
80 
81  class path_base : public xtd::string{
82  public:
83 
84  xtd::string& string() { return static_cast<xtd::string&>(*this);}
85  const xtd::string& string() const { return static_cast<const xtd::string&>(*this);}
86  template <typename ... _ArgTs> path_base(_ArgTs&&...oArgs) : xtd::string(std::forward<_ArgTs>(oArgs)...){}
87  };
88  class path : public path_base{
89  public:
90 #if ((XTD_OS_MINGW|XTD_OS_WINDOWS) & XTD_OS)
91  static constexpr value_type preferred_separator = '\\';
92  static constexpr value_type non_preferred_separator = '/';
93 #else
94  static constexpr value_type preferred_separator = '/';
95  static constexpr value_type non_preferred_separator = '\\';
96 #endif
97 
98  template <typename ... _ArgTs> path(_ArgTs&&...oArgs) : path_base(std::forward<_ArgTs>(oArgs)...){}
99 
100 
101  template <typename _RHST>
102  path& operator /= (const _RHST& rhs){
103  append(rhs);
104  return *this;
105  }
106 
107  template <typename _Ty> inline path operator+(const _Ty& src) const{
108  path oRet(*this);
109  oRet /= src;
110  return oRet;
111  }
112 
113  path& remove_filename(){
114  if (preferred_separator == string().back()){
115  return *this;
116  }
117  auto iEnd = --end();
118  for (; iEnd > begin() && preferred_separator != *iEnd && non_preferred_separator != *iEnd; --iEnd);
119  if (iEnd > begin()){
120  *this = substr(0, iEnd-begin());
121  }
122  return *this;
123  }
124 
125  path& replace_filename(const path& newval){
126  remove_filename();
127  *this /= newval;
128  return *this;
129  }
130 
131  path filename() const {
132  auto iEndSep = string().find_last_of(preferred_separator);
133  return string().substr(1+iEndSep);
134  }
135 
136  path& make_preferred(){
137  for (auto & oCh : *this){
138  if (oCh == non_preferred_separator){
139  oCh = preferred_separator;
140  }
141  }
142  return *this;
143  }
144 
145  path& append(const path& rhs){
146  if (preferred_separator == back() || non_preferred_separator==back()) pop_back();
147  if (preferred_separator != rhs.string().front() && non_preferred_separator != rhs.string().front()) push_back(preferred_separator);
148  _super_t::append(rhs.string());
149  return *this;
150  }
151 
152  path& operator+=(const path& rhs){ return append(rhs); }
153 
154  };
155 
156 
157 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
158  inline path temp_directory_path() {
159  xtd::string sTemp(1 + MAX_PATH, 0);
160  sTemp.resize(xtd::windows::exception::throw_if(GetTempPath(MAX_PATH, &sTemp[0]), [](DWORD d){return 0 == d;}));
161  return path(sTemp);
162  }
163 #else
164 
165  static inline size_t file_size(const path& oPath){
166  struct ::stat oStat;
167  if (0==::stat(oPath.string().c_str(), &oStat)){
168  return oStat.st_size;
169  }
170  return -1;
171  }
172 
173  static inline bool exists(const path& oPath){
174  FILE * pFile = fopen(oPath.string().c_str(), "r");
175  if (pFile) fclose(pFile);
176  return (pFile ? true : false);
177  }
178 
179  static inline path home_directory_path(){ return path(getenv("HOME")); }
180 
181 
182  inline path temp_directory_path(){
183  const char * cTempEnv = getenv("TMPDIR");
184  if (!cTempEnv || 0==strlen(cTempEnv)) cTempEnv = getenv("TEMP");
185 #if defined(P_tmpdir)
186  if (!cTempEnv || 0==strlen(cTempEnv)) cTempEnv = P_tmpdir;
187 #endif
188  if (!cTempEnv || 0==strlen(cTempEnv)) cTempEnv = _PATH_TMP;
189  return path(cTempEnv);
190  }
191 #endif
192 
193 
194  inline bool remove(const path& oPath){
195  return 0==::remove(oPath.string().c_str());
196  }
197  }
198 }
199 
200 #endif
201 
202 
203 namespace xtd{
204  namespace filesystem{
205  namespace _{
206 
207 
208  template <typename _ValueT, typename _Ch2, size_t _Dims> struct path_adder<_ValueT, const _Ch2(&)[_Dims]>{
209  inline static path add(const path& dest, const _Ch2(&src)[_Dims]){
210  auto str = xtd::xstring<_ValueT>::format(src);
211  path oRet(dest);
212  oRet.append(str);
213  return oRet;
214  }
215  };
216  template <typename _ValueT, typename _Ch2> struct path_adder<_ValueT, const _Ch2*>{
217  inline static path add(const path& dest, const _Ch2* src){
218  auto str = xtd::xstring<_ValueT>::format(src);
219  path oRet(dest);
220  oRet.append(str);
221  return oRet;
222  }
223  };
224  template <typename _ValueT> struct path_adder<_ValueT, const path&>{
225  inline static path add(const path& dest, const path& src){
226  path oRet(dest);
227  oRet.append(src);
228  return oRet;
229  }
230  };
231  template <typename _ValueT> struct path_adder<_ValueT, const xtd::xstring<_ValueT>&>{
232  inline static path add(const path& dest, const xtd::xstring<_ValueT>& src){
233  return path_adder<_ValueT, const _ValueT*>::add(dest, src.c_str());
234  }
235  };
236  }
237 
238 
239 
240  }
241 }
template meta-programming utilities
specializations of std::basic_string for advanced and common string handling
host, target and build configurations and settings Various components are purpose built for specific ...
#define throw_if(_test, _expression)
Simplifies use of exception::_throw_if.
Definition: exception.hpp:22
generic and special purpose exceptions
static xstring format()
Type safe formatting Appends each item in the parameter list together performing type-safe verificati...
Definition: string.hpp:62