First public contribution.
1 // boost/filesystem/convenience.hpp ----------------------------------------//
3 // Copyright Beman Dawes, 2002-2005
4 // Copyright Vladimir Prus, 2002
5 // Use, modification, and distribution is subject to the Boost Software
6 // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
7 // http://www.boost.org/LICENSE_1_0.txt)
9 // See library home page at http://www.boost.org/libs/filesystem
11 //----------------------------------------------------------------------------//
13 #ifndef BOOST_FILESYSTEM_CONVENIENCE_HPP
14 #define BOOST_FILESYSTEM_CONVENIENCE_HPP
16 #include <boost/filesystem/operations.hpp>
20 #include <boost/config/abi_prefix.hpp> // must be the last #include
22 # ifndef BOOST_FILESYSTEM_NARROW_ONLY
23 # define BOOST_FS_FUNC(BOOST_FS_TYPE) \
24 template<class Path> typename boost::enable_if<is_basic_path<Path>, \
26 # define BOOST_FS_FUNC_STRING BOOST_FS_FUNC(typename Path::string_type)
27 # define BOOST_FS_TYPENAME typename
29 # define BOOST_FS_FUNC(BOOST_FS_TYPE) inline BOOST_FS_TYPE
30 typedef boost::filesystem::path Path;
31 # define BOOST_FS_FUNC_STRING std::string
32 # define BOOST_FS_TYPENAME
40 BOOST_FS_FUNC(bool) create_directories(const Path& ph)
42 if (ph.empty() || exists(ph))
44 if ( !ph.empty() && !is_directory(ph) )
45 boost::throw_exception( basic_filesystem_error<Path>(
46 "boost::filesystem::create_directories", ph, -1 ) );
50 // First create branch, by calling ourself recursively
51 create_directories(ph.branch_path());
52 // Now that parent's path exists, create the directory
57 BOOST_FS_FUNC_STRING extension(const Path& ph)
59 typedef BOOST_FS_TYPENAME Path::string_type string_type;
60 string_type leaf = ph.leaf();
62 BOOST_FS_TYPENAME string_type::size_type n = leaf.rfind('.');
63 if (n != string_type::npos)
64 return leaf.substr(n);
69 BOOST_FS_FUNC_STRING basename(const Path& ph)
71 typedef BOOST_FS_TYPENAME Path::string_type string_type;
72 string_type leaf = ph.leaf();
73 BOOST_FS_TYPENAME string_type::size_type n = leaf.rfind('.');
74 return leaf.substr(0, n);
77 BOOST_FS_FUNC(Path) change_extension( const Path & ph,
78 const BOOST_FS_TYPENAME Path::string_type & new_extension )
79 { return ph.branch_path() / (basename(ph) + new_extension); }
81 # ifndef BOOST_FILESYSTEM_NARROW_ONLY
83 // "do-the-right-thing" overloads ---------------------------------------//
85 inline bool create_directories(const path& ph)
86 { return create_directories<path>(ph); }
87 inline bool create_directories(const wpath& ph)
88 { return create_directories<wpath>(ph); }
90 inline std::string extension(const path& ph)
91 { return extension<path>(ph); }
92 inline std::wstring extension(const wpath& ph)
93 { return extension<wpath>(ph); }
95 inline std::string basename(const path& ph)
96 { return basename<path>( ph ); }
97 inline std::wstring basename(const wpath& ph)
98 { return basename<wpath>( ph ); }
100 inline path change_extension( const path & ph, const std::string& new_ex )
101 { return change_extension<path>( ph, new_ex ); }
102 inline wpath change_extension( const wpath & ph, const std::wstring& new_ex )
103 { return change_extension<wpath>( ph, new_ex ); }
108 // basic_recursive_directory_iterator helpers --------------------------//
112 template< class Path >
113 struct recur_dir_itr_imp
115 typedef basic_directory_iterator< Path > element_type;
116 std::stack< element_type, std::vector< element_type > > m_stack;
121 recur_dir_itr_imp() : m_level(0), m_no_push(false), m_no_throw(false) {}
124 } // namespace detail
126 // basic_recursive_directory_iterator ----------------------------------//
128 template< class Path >
129 class basic_recursive_directory_iterator
130 : public boost::iterator_facade<
131 basic_recursive_directory_iterator<Path>,
132 basic_directory_entry<Path>,
133 boost::single_pass_traversal_tag >
136 typedef Path path_type;
138 basic_recursive_directory_iterator(){} // creates the "end" iterator
140 explicit basic_recursive_directory_iterator( const Path & dir_path );
141 basic_recursive_directory_iterator( const Path & dir_path, system_error_type & ec );
143 int level() const { return m_imp->m_level; }
148 BOOST_ASSERT( m_imp.get() && "attempt to no_push() on end iterator" );
149 m_imp->m_no_push = true;
152 file_status status() const
154 BOOST_ASSERT( m_imp.get()
155 && "attempt to call status() on end recursive_iterator" );
156 return m_imp->m_stack.top()->status();
159 file_status symlink_status() const
161 BOOST_ASSERT( m_imp.get()
162 && "attempt to call symlink_status() on end recursive_iterator" );
163 return m_imp->m_stack.top()->symlink_status();
168 // shared_ptr provides shallow-copy semantics required for InputIterators.
169 // m_imp.get()==0 indicates the end iterator.
170 boost::shared_ptr< detail::recur_dir_itr_imp< Path > > m_imp;
172 friend class boost::iterator_core_access;
174 typename boost::iterator_facade<
175 basic_recursive_directory_iterator<Path>,
176 basic_directory_entry<Path>,
177 boost::single_pass_traversal_tag >::reference
180 BOOST_ASSERT( m_imp.get() && "attempt to dereference end iterator" );
181 return *m_imp->m_stack.top();
186 bool equal( const basic_recursive_directory_iterator & rhs ) const
187 { return m_imp == rhs.m_imp; }
190 typedef basic_recursive_directory_iterator<path> recursive_directory_iterator;
191 # ifndef BOOST_FILESYSTEM_NARROW_ONLY
192 typedef basic_recursive_directory_iterator<wpath> wrecursive_directory_iterator;
195 // basic_recursive_directory_iterator implementation -------------------//
199 basic_recursive_directory_iterator<Path>::
200 basic_recursive_directory_iterator( const Path & dir_path )
201 : m_imp( new detail::recur_dir_itr_imp<Path> )
203 m_imp->m_stack.push( basic_directory_iterator<Path>( dir_path ) );
207 basic_recursive_directory_iterator<Path>::
208 basic_recursive_directory_iterator( const Path & dir_path, system_error_type & ec )
209 : m_imp( new detail::recur_dir_itr_imp<Path> )
211 m_imp->m_stack.push( basic_directory_iterator<Path>( dir_path, std::nothrow ) );
212 m_imp->m_no_throw = true;
217 void basic_recursive_directory_iterator<Path>::increment()
219 BOOST_ASSERT( m_imp.get() && "increment on end iterator" );
221 static const basic_directory_iterator<Path> end_itr;
223 if ( m_imp->m_no_push ) m_imp->m_no_push = false;
224 else if ( is_directory( m_imp->m_stack.top()->status() ) )
226 system_error_type ec;
229 ? basic_directory_iterator<Path>( *m_imp->m_stack.top(), ec )
230 : basic_directory_iterator<Path>( *m_imp->m_stack.top() )
232 if ( m_imp->m_stack.top() != end_itr )
237 m_imp->m_stack.pop();
240 while ( !m_imp->m_stack.empty()
241 && ++m_imp->m_stack.top() == end_itr )
243 m_imp->m_stack.pop();
247 if ( m_imp->m_stack.empty() ) m_imp.reset(); // done, so make end iterator
252 void basic_recursive_directory_iterator<Path>::pop()
254 BOOST_ASSERT( m_imp.get() && "pop on end iterator" );
255 BOOST_ASSERT( m_imp->m_level > 0 && "pop with level < 1" );
257 m_imp->m_stack.pop();
261 // what() basic_filesystem_error_decoder -------------------------------//
266 # if BOOST_WORKAROUND(__BORLANDC__,BOOST_TESTED_AT(0x581))
267 using boost::filesystem::system_message;
270 inline void decode_system_message( system_error_type ec, std::string & target )
272 system_message( ec, target );
275 # if defined(BOOST_WINDOWS_API) && !defined(BOOST_FILESYSTEM_NARROW_ONLY)
276 inline void decode_system_message( system_error_type ec, std::wstring & target )
278 system_message( ec, target );
282 template<class String>
283 void decode_system_message( system_error_type ec, String & target )
286 system_message( ec, temp );
287 for ( const char * p = temp.c_str(); *p != 0; ++p )
288 { target += static_cast<typename String::value_type>( *p ); }
293 typename Path::string_type what( const basic_filesystem_error<Path> & ex )
295 typename Path::string_type s;
296 for ( const char * p = ex.what(); *p != 0; ++p )
297 { s += static_cast<typename Path::string_type::value_type>( *p ); }
299 if ( !ex.path1().empty() )
301 s += static_cast<typename Path::string_type::value_type>( ':' );
302 s += static_cast<typename Path::string_type::value_type>( ' ' );
303 s += static_cast<typename Path::string_type::value_type>( '\"' );
304 s += ex.path1().file_string();
305 s += static_cast<typename Path::string_type::value_type>( '\"' );
307 if ( !ex.path2().empty() )
309 s += static_cast<typename Path::string_type::value_type>( ',' );
310 s += static_cast<typename Path::string_type::value_type>( ' ' );
311 s += static_cast<typename Path::string_type::value_type>( '\"' );
312 s += ex.path2().file_string();
313 s += static_cast<typename Path::string_type::value_type>( '\"' );
315 if ( ex.system_error() )
317 s += static_cast<typename Path::string_type::value_type>( ' ' );
319 detail::decode_system_message( ex.system_error(), s );
324 } // namespace filesystem
327 #undef BOOST_FS_FUNC_STRING
330 #include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas
331 #endif // BOOST_FILESYSTEM_CONVENIENCE_HPP