os/ossrv/ossrv_pub/boost_apis/boost/filesystem/convenience.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
//  boost/filesystem/convenience.hpp  ----------------------------------------//
sl@0
     2
sl@0
     3
//  Copyright Beman Dawes, 2002-2005
sl@0
     4
//  Copyright Vladimir Prus, 2002
sl@0
     5
//  Use, modification, and distribution is subject to the Boost Software
sl@0
     6
//  License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
sl@0
     7
//  http://www.boost.org/LICENSE_1_0.txt)
sl@0
     8
sl@0
     9
//  See library home page at http://www.boost.org/libs/filesystem
sl@0
    10
sl@0
    11
//----------------------------------------------------------------------------// 
sl@0
    12
sl@0
    13
#ifndef BOOST_FILESYSTEM_CONVENIENCE_HPP
sl@0
    14
#define BOOST_FILESYSTEM_CONVENIENCE_HPP
sl@0
    15
sl@0
    16
#include <boost/filesystem/operations.hpp>
sl@0
    17
#include <vector>
sl@0
    18
#include <stack>
sl@0
    19
sl@0
    20
#include <boost/config/abi_prefix.hpp> // must be the last #include
sl@0
    21
sl@0
    22
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
    23
#   define BOOST_FS_FUNC(BOOST_FS_TYPE) \
sl@0
    24
      template<class Path> typename boost::enable_if<is_basic_path<Path>, \
sl@0
    25
      BOOST_FS_TYPE>::type
sl@0
    26
#   define BOOST_FS_FUNC_STRING BOOST_FS_FUNC(typename Path::string_type)
sl@0
    27
#   define BOOST_FS_TYPENAME typename
sl@0
    28
# else
sl@0
    29
#   define BOOST_FS_FUNC(BOOST_FS_TYPE) inline BOOST_FS_TYPE 
sl@0
    30
    typedef boost::filesystem::path Path;
sl@0
    31
#   define BOOST_FS_FUNC_STRING std::string
sl@0
    32
#   define BOOST_FS_TYPENAME
sl@0
    33
# endif
sl@0
    34
sl@0
    35
namespace boost
sl@0
    36
{
sl@0
    37
  namespace filesystem
sl@0
    38
  {
sl@0
    39
sl@0
    40
    BOOST_FS_FUNC(bool) create_directories(const Path& ph)
sl@0
    41
    {
sl@0
    42
         if (ph.empty() || exists(ph))
sl@0
    43
         {
sl@0
    44
           if ( !ph.empty() && !is_directory(ph) )
sl@0
    45
               boost::throw_exception( basic_filesystem_error<Path>(
sl@0
    46
                 "boost::filesystem::create_directories", ph, -1 ) );
sl@0
    47
           return false;
sl@0
    48
         }
sl@0
    49
sl@0
    50
         // First create branch, by calling ourself recursively
sl@0
    51
         create_directories(ph.branch_path());
sl@0
    52
         // Now that parent's path exists, create the directory
sl@0
    53
         create_directory(ph);
sl@0
    54
         return true;
sl@0
    55
     }
sl@0
    56
sl@0
    57
    BOOST_FS_FUNC_STRING extension(const Path& ph)
sl@0
    58
    {
sl@0
    59
      typedef BOOST_FS_TYPENAME Path::string_type string_type;
sl@0
    60
      string_type leaf = ph.leaf();
sl@0
    61
sl@0
    62
      BOOST_FS_TYPENAME string_type::size_type n = leaf.rfind('.');
sl@0
    63
      if (n != string_type::npos)
sl@0
    64
        return leaf.substr(n);
sl@0
    65
      else
sl@0
    66
        return string_type();
sl@0
    67
    }
sl@0
    68
sl@0
    69
    BOOST_FS_FUNC_STRING basename(const Path& ph)
sl@0
    70
    {
sl@0
    71
      typedef BOOST_FS_TYPENAME Path::string_type string_type;
sl@0
    72
      string_type leaf = ph.leaf();
sl@0
    73
      BOOST_FS_TYPENAME string_type::size_type n = leaf.rfind('.');
sl@0
    74
      return leaf.substr(0, n);
sl@0
    75
    }
sl@0
    76
sl@0
    77
    BOOST_FS_FUNC(Path) change_extension( const Path & ph,
sl@0
    78
      const BOOST_FS_TYPENAME Path::string_type & new_extension )
sl@0
    79
      { return ph.branch_path() / (basename(ph) + new_extension); }
sl@0
    80
sl@0
    81
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
    82
sl@0
    83
    // "do-the-right-thing" overloads  ---------------------------------------//
sl@0
    84
sl@0
    85
    inline bool create_directories(const path& ph)
sl@0
    86
      { return create_directories<path>(ph); }
sl@0
    87
    inline bool create_directories(const wpath& ph)
sl@0
    88
      { return create_directories<wpath>(ph); }
sl@0
    89
sl@0
    90
    inline std::string extension(const path& ph)
sl@0
    91
      { return extension<path>(ph); }
sl@0
    92
    inline std::wstring extension(const wpath& ph)
sl@0
    93
      { return extension<wpath>(ph); }
sl@0
    94
sl@0
    95
    inline std::string basename(const path& ph)
sl@0
    96
      { return basename<path>( ph ); }
sl@0
    97
    inline std::wstring basename(const wpath& ph)
sl@0
    98
      { return basename<wpath>( ph ); }
sl@0
    99
sl@0
   100
    inline path change_extension( const path & ph, const std::string& new_ex )
sl@0
   101
      { return change_extension<path>( ph, new_ex ); }
sl@0
   102
    inline wpath change_extension( const wpath & ph, const std::wstring& new_ex )
sl@0
   103
      { return change_extension<wpath>( ph, new_ex ); }
sl@0
   104
sl@0
   105
# endif
sl@0
   106
sl@0
   107
sl@0
   108
    //  basic_recursive_directory_iterator helpers  --------------------------//
sl@0
   109
sl@0
   110
    namespace detail
sl@0
   111
    {
sl@0
   112
      template< class Path >
sl@0
   113
      struct recur_dir_itr_imp
sl@0
   114
      {
sl@0
   115
        typedef basic_directory_iterator< Path > element_type;
sl@0
   116
        std::stack< element_type, std::vector< element_type > > m_stack;
sl@0
   117
        int  m_level;
sl@0
   118
        bool m_no_push;
sl@0
   119
        bool m_no_throw;
sl@0
   120
sl@0
   121
        recur_dir_itr_imp() : m_level(0), m_no_push(false), m_no_throw(false) {}
sl@0
   122
      };
sl@0
   123
sl@0
   124
    } // namespace detail
sl@0
   125
sl@0
   126
    //  basic_recursive_directory_iterator  ----------------------------------//
sl@0
   127
sl@0
   128
    template< class Path >
sl@0
   129
    class basic_recursive_directory_iterator
sl@0
   130
      : public boost::iterator_facade<
sl@0
   131
          basic_recursive_directory_iterator<Path>,
sl@0
   132
          basic_directory_entry<Path>,
sl@0
   133
          boost::single_pass_traversal_tag >
sl@0
   134
    {
sl@0
   135
    public:
sl@0
   136
      typedef Path path_type;
sl@0
   137
sl@0
   138
      basic_recursive_directory_iterator(){}  // creates the "end" iterator
sl@0
   139
sl@0
   140
      explicit basic_recursive_directory_iterator( const Path & dir_path );
sl@0
   141
      basic_recursive_directory_iterator( const Path & dir_path, system_error_type & ec );
sl@0
   142
sl@0
   143
      int level() const { return m_imp->m_level; }
sl@0
   144
sl@0
   145
      void pop();
sl@0
   146
      void no_push()
sl@0
   147
      {
sl@0
   148
        BOOST_ASSERT( m_imp.get() && "attempt to no_push() on end iterator" );
sl@0
   149
        m_imp->m_no_push = true;
sl@0
   150
      }
sl@0
   151
sl@0
   152
      file_status status() const
sl@0
   153
      {
sl@0
   154
        BOOST_ASSERT( m_imp.get()
sl@0
   155
          && "attempt to call status() on end recursive_iterator" );
sl@0
   156
        return m_imp->m_stack.top()->status();
sl@0
   157
      }
sl@0
   158
sl@0
   159
      file_status symlink_status() const
sl@0
   160
      {
sl@0
   161
        BOOST_ASSERT( m_imp.get()
sl@0
   162
          && "attempt to call symlink_status() on end recursive_iterator" );
sl@0
   163
        return m_imp->m_stack.top()->symlink_status();
sl@0
   164
      }
sl@0
   165
sl@0
   166
    private:
sl@0
   167
sl@0
   168
      // shared_ptr provides shallow-copy semantics required for InputIterators.
sl@0
   169
      // m_imp.get()==0 indicates the end iterator.
sl@0
   170
      boost::shared_ptr< detail::recur_dir_itr_imp< Path > >  m_imp;
sl@0
   171
sl@0
   172
      friend class boost::iterator_core_access;
sl@0
   173
sl@0
   174
      typename boost::iterator_facade< 
sl@0
   175
        basic_recursive_directory_iterator<Path>,
sl@0
   176
        basic_directory_entry<Path>,
sl@0
   177
        boost::single_pass_traversal_tag >::reference
sl@0
   178
      dereference() const 
sl@0
   179
      {
sl@0
   180
        BOOST_ASSERT( m_imp.get() && "attempt to dereference end iterator" );
sl@0
   181
        return *m_imp->m_stack.top();
sl@0
   182
      }
sl@0
   183
sl@0
   184
      void increment();
sl@0
   185
sl@0
   186
      bool equal( const basic_recursive_directory_iterator & rhs ) const
sl@0
   187
        { return m_imp == rhs.m_imp; }
sl@0
   188
    };
sl@0
   189
sl@0
   190
    typedef basic_recursive_directory_iterator<path> recursive_directory_iterator;
sl@0
   191
# ifndef BOOST_FILESYSTEM_NARROW_ONLY
sl@0
   192
    typedef basic_recursive_directory_iterator<wpath> wrecursive_directory_iterator;
sl@0
   193
# endif
sl@0
   194
sl@0
   195
    //  basic_recursive_directory_iterator implementation  -------------------//
sl@0
   196
sl@0
   197
    //  constructors
sl@0
   198
    template<class Path>
sl@0
   199
    basic_recursive_directory_iterator<Path>::
sl@0
   200
      basic_recursive_directory_iterator( const Path & dir_path )
sl@0
   201
      : m_imp( new detail::recur_dir_itr_imp<Path> )
sl@0
   202
    {
sl@0
   203
      m_imp->m_stack.push( basic_directory_iterator<Path>( dir_path ) );
sl@0
   204
    }
sl@0
   205
sl@0
   206
    template<class Path>
sl@0
   207
    basic_recursive_directory_iterator<Path>::
sl@0
   208
      basic_recursive_directory_iterator( const Path & dir_path, system_error_type & ec )
sl@0
   209
      : m_imp( new detail::recur_dir_itr_imp<Path> )
sl@0
   210
    {
sl@0
   211
      m_imp->m_stack.push( basic_directory_iterator<Path>( dir_path, std::nothrow ) );
sl@0
   212
      m_imp->m_no_throw = true;
sl@0
   213
    }
sl@0
   214
sl@0
   215
    //  increment
sl@0
   216
    template<class Path>
sl@0
   217
    void basic_recursive_directory_iterator<Path>::increment()
sl@0
   218
    {
sl@0
   219
      BOOST_ASSERT( m_imp.get() && "increment on end iterator" );
sl@0
   220
      
sl@0
   221
      static const basic_directory_iterator<Path> end_itr;
sl@0
   222
sl@0
   223
      if ( m_imp->m_no_push ) m_imp->m_no_push = false;
sl@0
   224
      else if ( is_directory( m_imp->m_stack.top()->status() ) )
sl@0
   225
      {
sl@0
   226
        system_error_type ec;
sl@0
   227
        m_imp->m_stack.push(
sl@0
   228
          m_imp->m_no_throw
sl@0
   229
            ? basic_directory_iterator<Path>( *m_imp->m_stack.top(), ec )
sl@0
   230
            : basic_directory_iterator<Path>( *m_imp->m_stack.top() )
sl@0
   231
          );
sl@0
   232
        if ( m_imp->m_stack.top() != end_itr )
sl@0
   233
        {
sl@0
   234
          ++m_imp->m_level;
sl@0
   235
          return;
sl@0
   236
        }
sl@0
   237
        m_imp->m_stack.pop();
sl@0
   238
      }
sl@0
   239
sl@0
   240
      while ( !m_imp->m_stack.empty()
sl@0
   241
        && ++m_imp->m_stack.top() == end_itr )
sl@0
   242
      {
sl@0
   243
        m_imp->m_stack.pop();
sl@0
   244
        --m_imp->m_level;
sl@0
   245
      }
sl@0
   246
sl@0
   247
      if ( m_imp->m_stack.empty() ) m_imp.reset(); // done, so make end iterator
sl@0
   248
    }
sl@0
   249
sl@0
   250
    //  pop
sl@0
   251
    template<class Path>
sl@0
   252
    void basic_recursive_directory_iterator<Path>::pop()
sl@0
   253
    {
sl@0
   254
      BOOST_ASSERT( m_imp.get() && "pop on end iterator" );
sl@0
   255
      BOOST_ASSERT( m_imp->m_level > 0 && "pop with level < 1" );
sl@0
   256
sl@0
   257
      m_imp->m_stack.pop();
sl@0
   258
      --m_imp->m_level;
sl@0
   259
    }
sl@0
   260
sl@0
   261
    //  what() basic_filesystem_error_decoder  -------------------------------//
sl@0
   262
sl@0
   263
    namespace detail
sl@0
   264
    {
sl@0
   265
sl@0
   266
#   if BOOST_WORKAROUND(__BORLANDC__,BOOST_TESTED_AT(0x581))
sl@0
   267
      using boost::filesystem::system_message;
sl@0
   268
#   endif
sl@0
   269
sl@0
   270
      inline void decode_system_message( system_error_type ec, std::string & target )
sl@0
   271
      {
sl@0
   272
        system_message( ec, target );
sl@0
   273
      }
sl@0
   274
sl@0
   275
#   if defined(BOOST_WINDOWS_API) && !defined(BOOST_FILESYSTEM_NARROW_ONLY)
sl@0
   276
      inline void decode_system_message( system_error_type ec, std::wstring & target )
sl@0
   277
      {
sl@0
   278
        system_message( ec, target );
sl@0
   279
      }
sl@0
   280
#   endif
sl@0
   281
sl@0
   282
      template<class String>
sl@0
   283
      void decode_system_message( system_error_type ec, String & target )
sl@0
   284
      {
sl@0
   285
        std::string temp;
sl@0
   286
        system_message( ec, temp );
sl@0
   287
        for ( const char * p = temp.c_str(); *p != 0; ++p )
sl@0
   288
          { target += static_cast<typename String::value_type>( *p ); }
sl@0
   289
      }
sl@0
   290
    }
sl@0
   291
sl@0
   292
    template<class Path>
sl@0
   293
    typename Path::string_type what( const basic_filesystem_error<Path> & ex )
sl@0
   294
    {
sl@0
   295
      typename Path::string_type s;
sl@0
   296
      for ( const char * p = ex.what(); *p != 0; ++p )
sl@0
   297
        { s += static_cast<typename Path::string_type::value_type>( *p ); }
sl@0
   298
sl@0
   299
      if ( !ex.path1().empty() )
sl@0
   300
      {
sl@0
   301
        s += static_cast<typename Path::string_type::value_type>( ':' );
sl@0
   302
        s += static_cast<typename Path::string_type::value_type>( ' ' );
sl@0
   303
        s += static_cast<typename Path::string_type::value_type>( '\"' );
sl@0
   304
        s += ex.path1().file_string();
sl@0
   305
        s += static_cast<typename Path::string_type::value_type>( '\"' );
sl@0
   306
      }
sl@0
   307
      if ( !ex.path2().empty() )
sl@0
   308
      {
sl@0
   309
        s += static_cast<typename Path::string_type::value_type>( ',' );
sl@0
   310
        s += static_cast<typename Path::string_type::value_type>( ' ' );
sl@0
   311
        s += static_cast<typename Path::string_type::value_type>( '\"' );
sl@0
   312
        s += ex.path2().file_string();
sl@0
   313
        s += static_cast<typename Path::string_type::value_type>( '\"' );
sl@0
   314
      }
sl@0
   315
      if ( ex.system_error() )
sl@0
   316
      {
sl@0
   317
        s += static_cast<typename Path::string_type::value_type>( ' ' );
sl@0
   318
sl@0
   319
        detail::decode_system_message( ex.system_error(), s );
sl@0
   320
      }
sl@0
   321
      return s;
sl@0
   322
    }
sl@0
   323
sl@0
   324
  } // namespace filesystem
sl@0
   325
} // namespace boost
sl@0
   326
sl@0
   327
#undef BOOST_FS_FUNC_STRING
sl@0
   328
#undef BOOST_FS_FUNC
sl@0
   329
sl@0
   330
#include <boost/config/abi_suffix.hpp> // pops abi_prefix.hpp pragmas
sl@0
   331
#endif // BOOST_FILESYSTEM_CONVENIENCE_HPP