Asio Extensions
Additional functionality built on top of (Boost.)Asio
unique_file_handle.hpp
1 /// @file
2 /// Defines the unique_file_handle class
3 ///
4 /// @copyright Copyright (c) 2015 Tim Niederhausen (tim@rnc-ag.de)
5 /// Distributed under the Boost Software License, Version 1.0.
6 /// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 
8 #ifndef ASIOEXT_UNIQUEFILEHANDLE_HPP
9 #define ASIOEXT_UNIQUEFILEHANDLE_HPP
10 
11 #include "asioext/detail/config.hpp"
12 
13 #if ASIOEXT_HAS_PRAGMA_ONCE
14 # pragma once
15 #endif
16 
17 #include "asioext/file_handle.hpp"
18 #include "asioext/file_perms.hpp"
19 #include "asioext/file_attrs.hpp"
20 #include "asioext/seek_origin.hpp"
21 #include "asioext/error_code.hpp"
22 
23 ASIOEXT_NS_BEGIN
24 
25 /// @ingroup files_handle
26 /// @brief RAII wrapper around file_handle.
27 ///
28 /// The unique_file_handle class provides a std::unique_ptr<>-like wrapper
29 /// around a file_handle. The unique_file_handle owns the contained native
30 /// handle.
31 ///
32 /// unique_file_handle objects cannot be copied, but are move-constructible/
33 /// move-assignable if a compiler with C++11 support is used.
34 ///
35 /// unique_file_handle models the following asio concepts:
36 /// * SyncRandomAccessReadDevice
37 /// * SyncRandomAccessWriteDevice
38 /// * SyncReadStream
39 /// * SyncWriteStream
40 ///
41 /// @par Thread Safety:
42 /// @e Distinct @e objects: Safe.@n
43 /// @e Shared @e objects: Unsafe.
44 ///
45 /// @note If copying a handle (which is a costly operation) is really necessary,
46 /// the @ref duplicate function can be used.
47 ///
48 /// @par Example
49 /// @code
50 /// asioext::unique_file_handle file = asioext::open(
51 /// "myfile.txt", asioext::open_flags::access_read_write |
52 /// asioext::open_flags::create_always);
53 ///
54 /// // Write a string
55 /// const std::string content = "Hello world";
56 /// asio::write(file, asio::buffer(content));
57 ///
58 /// // Read it back
59 /// // Note that we could avoid seek() by using write_at/read_t!
60 /// file.seek(asioext::seek_origin::from_begin, 0);
61 /// std::string content_returned(content.size());
62 /// asio::read(file, asio::buffer(content));
63 /// @endcode
65 {
66 public:
67  /// A unique_file_handle is always the lowest layer.
69 
70  /// @brief Construct an empy unique_file_handle.
71  ///
72  /// This constructor initializes the unique_file_handle to an empty state.
73  ASIOEXT_DECL unique_file_handle() ASIOEXT_NOEXCEPT;
74 
75  /// Destroy a unique_file_handle.
76  ///
77  /// This destructor attempts to close the currently owned file handle.
78  /// Failures are silently ignored.
79  ASIOEXT_DECL ~unique_file_handle();
80 
81  /// @brief Construct a unique_file_handle using a file_handle object.
82  ///
83  /// This constructor takes ownership of the given file_handle.
84  ///
85  /// @param handle The file_handle object which shall be assigned to this
86  /// unique_file_handle object.
87  ASIOEXT_DECL explicit unique_file_handle(file_handle handle) ASIOEXT_NOEXCEPT;
88 
89  /// @brief Construct a unique_file_handle using a native handle object.
90  ///
91  /// This constructor takes ownership of the given native handle.
92  ///
93  /// @param handle The native handle object which shall be assigned to this
94  /// unique_file_handle object.
95  ASIOEXT_DECL explicit unique_file_handle(
96  const file_handle::native_handle_type& handle) ASIOEXT_NOEXCEPT;
97 
98 #ifdef ASIOEXT_HAS_MOVE
99  /// @brief Move-construct a unique_file_handle from another.
100  ///
101  /// This constructor moves a handle from one object to another.
102  ///
103  /// @param other The other unique_file_handle object from which the move will
104  /// occur.
105  ///
106  /// @note Following the move, the moved-from object is in the same state as if
107  /// constructed using the @c unique_file_handle() constructor.
108  ASIOEXT_DECL unique_file_handle(unique_file_handle&& other) ASIOEXT_NOEXCEPT;
109 
110  /// @brief Move-assign a unique_file_handle from another.
111  ///
112  /// This assignment operator moves a handle from one object to another.
113  /// If this object already owns a file handle, the current handle will be
114  /// closed. This operation is potentially throwing.
115  ///
116  /// @param other The other unique_file_handle object from which the move will
117  /// occur.
118  ///
119  /// @note Following the move, the moved-from object is in the same state as if
120  /// constructed using the @c unique_file_handle() constructor.
121  ASIOEXT_DECL unique_file_handle& operator=(unique_file_handle&& other);
122 #endif
123 
124  /// @brief Get a reference to the lowest layer.
125  ///
126  /// This function returns a reference to the lowest layer in a stack of
127  /// layers. Since a unique_file_handle cannot contain any further layers, it
128  /// simply returns a reference to itself.
129  ///
130  /// @return A reference to the lowest layer in the stack of layers. Ownership
131  /// is not transferred to the caller.
132  lowest_layer_type& lowest_layer() ASIOEXT_NOEXCEPT
133  {
134  return *this;
135  }
136 
137  /// @brief Get a const reference to the lowest layer.
138  ///
139  /// This function returns a const reference to the lowest layer in a stack of
140  /// layers. Since a unique_file_handle cannot contain any further layers, it
141  /// simply returns a reference to itself.
142  ///
143  /// @return A const reference to the lowest layer in the stack of layers.
144  /// Ownership is not transferred to the caller.
145  const lowest_layer_type& lowest_layer() const ASIOEXT_NOEXCEPT
146  {
147  return *this;
148  }
149 
150  /// @brief Get the managed file_handle.
151  ///
152  /// This function returns a copy of the managed file_handle.
153  ///
154  /// @return A copy of the contained file_handle object.
155  /// Ownership is not transferred to the caller.
156  file_handle get() ASIOEXT_NOEXCEPT
157  {
158  return handle_;
159  }
160 
161  /// @copydoc file_handle::is_open()
162  bool is_open() const ASIOEXT_NOEXCEPT
163  {
164  return handle_.is_open();
165  }
166 
167  /// @copydoc file_handle::close()
168  void close()
169  {
170  handle_.close();
171  }
172 
173  /// @copydoc file_handle::close(error_code&)
174  void close(error_code& ec) ASIOEXT_NOEXCEPT
175  {
176  handle_.close(ec);
177  }
178 
179  /// @brief Take ownership of the contained file_handle.
180  ///
181  /// This function transfers ownership of the contained file_handle to
182  /// the caller. The unique_file_handle object is reset to an empty state
183  /// (i.e. no file opened).
184  ///
185  /// @return The contained file_handle.
186  /// The caller is now responsible to close it.
187  ///
188  /// @warning This function is dangerous. It is highly unlikely that you'll
189  /// ever need to use this.
190  ASIOEXT_DECL file_handle release() ASIOEXT_NOEXCEPT;
191 
192  /// @brief Replace the managed file_handle.
193  ///
194  /// This function closes the currently managed handle (if any),
195  /// and replaces it with the given one.
196  ///
197  /// @param handle The new file_handle to manage.
198  /// Ownership is transferred to the unique_file_handle.
199  ///
200  /// @throws asio::system_error Thrown on failure.
201  ASIOEXT_DECL void reset(file_handle handle);
202 
203  /// @brief Replace the managed file_handle.
204  ///
205  /// This function closes the currently managed handle (if any),
206  /// and replaces it with the given one.
207  ///
208  /// @param handle The new file_handle to manage.
209  /// Ownership is transferred to the unique_file_handle.
210  ///
211  /// @param ec Set to indicate what error occurred. If no error occurred,
212  /// the object is reset.
213  ASIOEXT_DECL void reset(file_handle handle, error_code& ec) ASIOEXT_NOEXCEPT;
214 
215  /// @name Positioning functions
216  /// @{
217 
218  /// @copydoc file_handle::position()
219  uint64_t position()
220  {
221  return handle_.position();
222  }
223 
224  /// @copydoc file_handle::position(error_code&)
225  uint64_t position(error_code& ec) ASIOEXT_NOEXCEPT
226  {
227  return handle_.position(ec);
228  }
229 
230  /// @copydoc file_handle::seek(seek_origin,int64_t)
231  uint64_t seek(seek_origin origin, int64_t offset)
232  {
233  return handle_.seek(origin, offset);
234  }
235 
236  /// @copydoc file_handle::seek(seek_origin,int64_t,error_code&)
237  uint64_t seek(seek_origin origin,
238  int64_t offset,
239  error_code& ec) ASIOEXT_NOEXCEPT
240  {
241  return handle_.seek(origin, offset, ec);
242  }
243 
244  /// @}
245 
246  /// @name Metadata functions
247  /// @{
248 
249  /// @copydoc file_handle::size()
250  uint64_t size()
251  {
252  return handle_.size();
253  }
254 
255  /// @copydoc file_handle::size(error_code&)
256  uint64_t size(error_code& ec) ASIOEXT_NOEXCEPT
257  {
258  return handle_.size(ec);
259  }
260 
261  /// @copydoc file_handle::size(uint64_t)
262  void size(uint64_t new_size)
263  {
264  handle_.size(new_size);
265  }
266 
267  /// @copydoc file_handle::size(uint64_t,error_code&)
268  void size(uint64_t new_size, error_code& ec) ASIOEXT_NOEXCEPT
269  {
270  handle_.size(new_size, ec);
271  }
272 
273  /// @copydoc file_handle::permissions()
274  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
276  {
277  return handle_.permissions();
278  }
279 
280  /// @copydoc file_handle::permissions(error_code&)
281  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
282  file_perms permissions(error_code& ec) ASIOEXT_NOEXCEPT
283  {
284  return handle_.permissions(ec);
285  }
286 
287  /// @copydoc file_handle::permissions(file_perms,file_perm_options)
288  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
290  file_perm_options::replace)
291  {
292  handle_.permissions(perms, opts);
293  }
294 
295  /// @copydoc file_handle::permissions(file_perms,error_code&)
296  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
297  void permissions(file_perms perms, error_code& ec) ASIOEXT_NOEXCEPT
298  {
299  handle_.permissions(perms, ec);
300  }
301 
302  /// @copydoc file_handle::permissions(file_perms,file_perm_options,error_code&)
303  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
305  error_code& ec) ASIOEXT_NOEXCEPT
306  {
307  handle_.permissions(perms, opts, ec);
308  }
309 
310  /// @copydoc file_handle::attributes()
311  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
313  {
314  return handle_.attributes();
315  }
316 
317  /// @copydoc file_handle::attributes(error_code&)
318  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
319  file_attrs attributes(error_code& ec) ASIOEXT_NOEXCEPT
320  {
321  return handle_.attributes(ec);
322  }
323 
324  /// @copydoc file_handle::attributes(file_attrs,file_attr_options)
325  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
327  file_attr_options::replace)
328  {
329  handle_.attributes(attrs, opts);
330  }
331 
332  /// @copydoc file_handle::attributes(file_attrs,file_attr_options,error_code&)
333  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
335  error_code& ec) ASIOEXT_NOEXCEPT
336  {
337  handle_.attributes(attrs, opts, ec);
338  }
339 
340  /// @copydoc file_handle::attributes(file_attrs,error_code&)
341  ASIOEXT_WINDOWS_NO_HANDLEINFO_WARNING
342  void attributes(file_attrs attrs, error_code& ec) ASIOEXT_NOEXCEPT
343  {
344  handle_.attributes(attrs, ec);
345  }
346 
347  /// @copydoc file_handle::times()
349  {
350  return handle_.times();
351  }
352 
353  /// @copydoc file_handle::times(error_code&)
354  file_times times(error_code& ec) ASIOEXT_NOEXCEPT
355  {
356  return handle_.times(ec);
357  }
358 
359  /// @copydoc file_handle::times(const file_times&)
360  void times(const file_times& new_times)
361  {
362  handle_.times(new_times);
363  }
364 
365  /// @copydoc file_handle::times(const file_times&,error_code&)
366  void times(const file_times& new_times, error_code& ec) ASIOEXT_NOEXCEPT
367  {
368  handle_.times(new_times, ec);
369  }
370 
371  /// @}
372 
373  /// @name SyncReadStream functions
374  /// @{
375 
376  /// @copydoc file_handle::read_some(const MutableBufferSequence&)
377  template <typename MutableBufferSequence>
379  {
380  return handle_.read_some(buffers);
381  }
382 
383  /// @copydoc file_handle::read_some(const MutableBufferSequence&,error_code&)
384  template <typename MutableBufferSequence>
386  error_code& ec) ASIOEXT_NOEXCEPT
387  {
388  return handle_.read_some(buffers, ec);
389  }
390 
391  /// @}
392 
393  /// @name SyncWriteStream functions
394  /// @{
395 
396  /// @copydoc file_handle::write_some(const ConstBufferSequence&)
397  template <typename ConstBufferSequence>
399  {
400  return handle_.write_some(buffers);
401  }
402 
403  /// @copydoc file_handle::write_some(const ConstBufferSequence&,error_code&)
404  template <typename ConstBufferSequence>
406  error_code& ec) ASIOEXT_NOEXCEPT
407  {
408  return handle_.write_some(buffers, ec);
409  }
410 
411  /// @}
412 
413  /// @name SyncRandomAccessReadDevice functions
414  /// @{
415 
416  /// @copydoc file_handle::read_some_at(uint64_t,const MutableBufferSequence&)
417  template <typename MutableBufferSequence>
418  std::size_t read_some_at(uint64_t offset,
419  const MutableBufferSequence& buffers)
420  {
421  return handle_.read_some_at(offset, buffers);
422  }
423 
424  /// @copydoc file_handle::read_some_at(uint64_t,const MutableBufferSequence&,error_code&)
425  template <typename MutableBufferSequence>
426  std::size_t read_some_at(uint64_t offset,
427  const MutableBufferSequence& buffers,
428  error_code& ec) ASIOEXT_NOEXCEPT
429  {
430  return handle_.read_some_at(offset, buffers, ec);
431  }
432 
433  /// @}
434 
435  /// @name SyncRandomAccessWriteDevice functions
436  /// @{
437 
438  /// @copydoc file_handle::write_some_at(uint64_t,const ConstBufferSequence&)
439  template <typename ConstBufferSequence>
440  std::size_t write_some_at(uint64_t offset, const ConstBufferSequence& buffers)
441  {
442  return handle_.write_some_at(offset, buffers);
443  }
444 
445  /// @copydoc file_handle::write_some_at(uint64_t,const ConstBufferSequence&,error_code&)
446  template <typename ConstBufferSequence>
447  std::size_t write_some_at(uint64_t offset,
448  const ConstBufferSequence& buffers,
449  error_code& ec) ASIOEXT_NOEXCEPT
450  {
451  return handle_.write_some_at(offset, buffers, ec);
452  }
453 
454  /// @}
455 
456 private:
457  // Prevent copying
458  unique_file_handle(const unique_file_handle&) ASIOEXT_DELETED;
459  unique_file_handle& operator=(const unique_file_handle&) ASIOEXT_DELETED;
460 
461  file_handle handle_;
462 };
463 
464 ASIOEXT_NS_END
465 
466 #if defined(ASIOEXT_HEADER_ONLY)
467 # include "asioext/impl/unique_file_handle.cpp"
468 #endif
469 
470 #endif
void times(const file_times &new_times, error_code &ec) noexcept
Change a file&#39;s time data.
Definition: unique_file_handle.hpp:366
void permissions(file_perms perms, file_perm_options opts=file_perm_options::replace)
Change file access permissions.
Definition: unique_file_handle.hpp:289
uint64_t seek(seek_origin origin, int64_t offset, error_code &ec) noexcept
Change the read/write position.
Definition: unique_file_handle.hpp:237
std::size_t write_some(const ConstBufferSequence &buffers)
Write some data to the file.
Definition: unique_file_handle.hpp:398
Container for various times associated with a file.
Definition: file_handle.hpp:34
file_perms
Names for permissions.
Definition: file_perms.hpp:31
std::size_t read_some_at(uint64_t offset, const MutableBufferSequence &buffers)
Read some data from the file at the specified offset.
Definition: unique_file_handle.hpp:418
file_perm_options
Options to control the behavior of asioext::file_handle::permissions(file_perms,file_perm_options) ...
Definition: file_perms.hpp:119
const lowest_layer_type & lowest_layer() const noexcept
Get a const reference to the lowest layer.
Definition: unique_file_handle.hpp:145
void close(error_code &ec) noexcept
Close the handle.
Definition: unique_file_handle.hpp:174
void permissions(file_perms perms, file_perm_options opts, error_code &ec) noexcept
Change file access permissions.
Definition: unique_file_handle.hpp:304
file_times times(error_code &ec) noexcept
Get the file&#39;s time data.
Definition: unique_file_handle.hpp:354
void size(uint64_t new_size, error_code &ec) noexcept
Set the size of a file.
Definition: unique_file_handle.hpp:268
std::size_t write_some_at(uint64_t offset, const ConstBufferSequence &buffers)
Write some data to the file at the specified offset.
Definition: unique_file_handle.hpp:440
std::size_t read_some(const MutableBufferSequence &buffers)
Read some data from the file.
Definition: unique_file_handle.hpp:378
file_attrs
Names for file attributes.
Definition: file_attrs.hpp:32
uint64_t seek(seek_origin origin, int64_t offset)
Change the read/write position.
Definition: unique_file_handle.hpp:231
RAII wrapper around file_handle.
Definition: unique_file_handle.hpp:64
bool is_open() const noexcept
Determine whether the handle is open.
Definition: unique_file_handle.hpp:162
void close()
Close the handle.
Definition: unique_file_handle.hpp:168
void attributes(file_attrs attrs, error_code &ec) noexcept
Change the file&#39;s attributes.
Definition: unique_file_handle.hpp:342
uint64_t size()
Get the size of a file.
Definition: unique_file_handle.hpp:250
file_times times()
Get the file&#39;s time data.
Definition: unique_file_handle.hpp:348
automatically_chosen error_code
Typedef for the error_code class used by this library.
Definition: error_code.hpp:37
uint64_t position(error_code &ec) noexcept
Get the current file position.
Definition: unique_file_handle.hpp:225
uint64_t position()
Get the current file position.
void times(const file_times &new_times)
Change a file&#39;s time data.
Definition: unique_file_handle.hpp:360
file_attrs attributes()
Get the file&#39;s attributes.
Definition: unique_file_handle.hpp:312
std::size_t write_some(const ConstBufferSequence &buffers, error_code &ec) noexcept
Write some data to the file.
Definition: unique_file_handle.hpp:405
seek_origin
Specifies the various position offset origins.
Definition: seek_origin.hpp:25
void size(uint64_t new_size)
Set the size of a file.
Definition: unique_file_handle.hpp:262
file_perms permissions(error_code &ec) noexcept
Get the file&#39;s current access permissions.
Definition: unique_file_handle.hpp:282
file_perms permissions()
Get the file&#39;s current access permissions.
Definition: unique_file_handle.hpp:275
unique_file_handle lowest_layer_type
A unique_file_handle is always the lowest layer.
Definition: unique_file_handle.hpp:68
std::size_t read_some(const MutableBufferSequence &buffers, error_code &ec) noexcept
Read some data from the file.
Definition: unique_file_handle.hpp:385
std::size_t read_some_at(uint64_t offset, const MutableBufferSequence &buffers, error_code &ec) noexcept
Read some data from the file at the specified offset.
Definition: unique_file_handle.hpp:426
void permissions(file_perms perms, error_code &ec) noexcept
Change file access permissions.
Definition: unique_file_handle.hpp:297
file_attr_options
Options to control the behavior of asioext::file_handle::attributes(file_attrs,file_attr_options) ...
Definition: file_attrs.hpp:99
void attributes(file_attrs attrs, file_attr_options opts, error_code &ec) noexcept
Change the file&#39;s attributes.
Definition: unique_file_handle.hpp:334
std::size_t write_some_at(uint64_t offset, const ConstBufferSequence &buffers, error_code &ec) noexcept
Write some data to the file at the specified offset.
Definition: unique_file_handle.hpp:447
uint64_t size(error_code &ec) noexcept
Get the size of a file.
Definition: unique_file_handle.hpp:256
file_attrs attributes(error_code &ec) noexcept
Get the file&#39;s attributes.
Definition: unique_file_handle.hpp:319
A thin and lightweight wrapper around a native file handle.
Definition: file_handle.hpp:73
lowest_layer_type & lowest_layer() noexcept
Get a reference to the lowest layer.
Definition: unique_file_handle.hpp:132
void attributes(file_attrs attrs, file_attr_options opts=file_attr_options::replace)
Change the file&#39;s attributes.
Definition: unique_file_handle.hpp:326