/Library/Source/Nano/Files/NFileInfo.cpp

http://github.com/refnum/nano · C++ · 603 lines · 159 code · 263 blank · 181 comment · 3 complexity · d17cbf7847eb138968752bee6393bdb2 MD5 · raw file

  1. /* NAME:
  2. NFileInfo.cpp
  3. DESCRIPTION:
  4. File information.
  5. COPYRIGHT:
  6. Copyright (c) 2006-2021, refNum Software
  7. All rights reserved.
  8. Redistribution and use in source and binary forms, with or without
  9. modification, are permitted provided that the following conditions
  10. are met:
  11. 1. Redistributions of source code must retain the above copyright
  12. notice, this list of conditions and the following disclaimer.
  13. 2. Redistributions in binary form must reproduce the above copyright
  14. notice, this list of conditions and the following disclaimer in the
  15. documentation and/or other materials provided with the distribution.
  16. 3. Neither the name of the copyright holder nor the names of its
  17. contributors may be used to endorse or promote products derived from
  18. this software without specific prior written permission.
  19. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  20. "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  21. LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  22. A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  23. HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  24. SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  25. LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  26. DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  27. THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  28. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  29. OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  30. ___________________________________________________________________________
  31. */
  32. //=============================================================================
  33. // Includes
  34. //-----------------------------------------------------------------------------
  35. #include "NFileInfo.h"
  36. //=============================================================================
  37. // Internal Constants
  38. //-----------------------------------------------------------------------------
  39. inline constexpr NFileStateFlags kNFileStateMaskError =
  40. NFileState::Exists | NFileState::IsFile | NFileState::IsDirectory | NFileState::CanRead |
  41. NFileState::CanWrite | NFileState::CanExecute;
  42. //=============================================================================
  43. // NFileInfo::NFileInfo : Constructor.
  44. //-----------------------------------------------------------------------------
  45. NFileInfo::NFileInfo(const NFilePath& thePath)
  46. : mPath(thePath)
  47. , mValid{}
  48. , mState{}
  49. {
  50. // Validate our parameters
  51. NN_REQUIRE(thePath.IsValid());
  52. }
  53. //=============================================================================
  54. // NFileInfo::NFileInfo : Constructor.
  55. //-----------------------------------------------------------------------------
  56. NFileInfo::NFileInfo()
  57. : mPath{}
  58. , mValid{}
  59. , mState{}
  60. {
  61. }
  62. //=============================================================================
  63. // NFileInfo::IsValid : Is the object valid?
  64. //-----------------------------------------------------------------------------
  65. bool NFileInfo::IsValid() const
  66. {
  67. // Check our state
  68. return mPath.IsValid();
  69. }
  70. //=============================================================================
  71. // NFileInfo::Clear : Clear the object.
  72. //-----------------------------------------------------------------------------
  73. void NFileInfo::Clear()
  74. {
  75. // Clear the object
  76. mPath.Clear();
  77. Refresh();
  78. }
  79. //=============================================================================
  80. // NFileInfo::GetPath : Get the path.
  81. //-----------------------------------------------------------------------------
  82. NFilePath NFileInfo::GetPath() const
  83. {
  84. // Get the path
  85. return mPath;
  86. }
  87. //=============================================================================
  88. // NFileInfo::SetPath : Set the path.
  89. //-----------------------------------------------------------------------------
  90. void NFileInfo::SetPath(const NFilePath& thePath)
  91. {
  92. // Validate our parameters
  93. NN_REQUIRE(thePath.IsValid());
  94. // Set the path
  95. mPath = thePath;
  96. Refresh();
  97. }
  98. //=============================================================================
  99. // NFileInfo::Refresh : Refresh the info.
  100. //-----------------------------------------------------------------------------
  101. void NFileInfo::Refresh()
  102. {
  103. // Update our state
  104. mValid = {};
  105. mState = {};
  106. }
  107. //=============================================================================
  108. // NFileInfo::Exists : Does the path exist?
  109. //-----------------------------------------------------------------------------
  110. bool NFileInfo::Exists() const
  111. {
  112. // Validate our state
  113. NN_REQUIRE(IsValid());
  114. // Get the state
  115. UpdateState(NFileState::Exists);
  116. return TestFlag(NFileState::Exists);
  117. }
  118. //=============================================================================
  119. // NFileInfo::IsFile : Is the path a file?
  120. //-----------------------------------------------------------------------------
  121. bool NFileInfo::IsFile() const
  122. {
  123. // Validate our state
  124. NN_REQUIRE(IsValid());
  125. // Get the state
  126. UpdateState(NFileState::IsFile);
  127. return TestFlag(NFileState::IsFile);
  128. }
  129. //=============================================================================
  130. // NFileInfo::IsDirectory : Is the path a directory?
  131. //-----------------------------------------------------------------------------
  132. bool NFileInfo::IsDirectory() const
  133. {
  134. // Validate our state
  135. NN_REQUIRE(IsValid());
  136. // Get the state
  137. UpdateState(NFileState::IsDirectory);
  138. return TestFlag(NFileState::IsDirectory);
  139. }
  140. //=============================================================================
  141. // NFileInfo::CanRead : Can the path be read?
  142. //-----------------------------------------------------------------------------
  143. bool NFileInfo::CanRead() const
  144. {
  145. // Validate our state
  146. NN_REQUIRE(IsValid());
  147. // Get the state
  148. UpdateState(NFileState::CanRead);
  149. return TestFlag(NFileState::CanRead);
  150. }
  151. //=============================================================================
  152. // NFileInfo::CanWrite : Can the path be written?
  153. //-----------------------------------------------------------------------------
  154. bool NFileInfo::CanWrite() const
  155. {
  156. // Validate our state
  157. NN_REQUIRE(IsValid());
  158. // Get the state
  159. UpdateState(NFileState::CanWrite);
  160. return TestFlag(NFileState::CanWrite);
  161. }
  162. //=============================================================================
  163. // NFileInfo::CanExecute : Can the path be executed / searched?
  164. //-----------------------------------------------------------------------------
  165. bool NFileInfo::CanExecute() const
  166. {
  167. // Validate our state
  168. NN_REQUIRE(IsValid());
  169. // Get the state
  170. UpdateState(NFileState::CanExecute);
  171. return TestFlag(NFileState::CanExecute);
  172. }
  173. //=============================================================================
  174. // NFileInfo::GetCreationTime : Get the creation time.
  175. //-----------------------------------------------------------------------------
  176. NTime NFileInfo::GetCreationTime() const
  177. {
  178. // Validate our state
  179. NN_REQUIRE(IsValid());
  180. // Get the state
  181. UpdateState(NFileState::CreationTime);
  182. return mState.creationTime;
  183. }
  184. //=============================================================================
  185. // NFileInfo::GetModifiedTime : Get the last-modified time.
  186. //-----------------------------------------------------------------------------
  187. NTime NFileInfo::GetModifiedTime() const
  188. {
  189. // Validate our state
  190. NN_REQUIRE(IsValid());
  191. // Get the state
  192. UpdateState(NFileState::ModifiedTime);
  193. return mState.modifiedTime;
  194. }
  195. //=============================================================================
  196. // NFileInfo::GetFileSize : Get the file size.
  197. //-----------------------------------------------------------------------------
  198. uint64_t NFileInfo::GetFileSize() const
  199. {
  200. // Validate our state
  201. NN_REQUIRE(IsValid());
  202. // Get the state
  203. UpdateState(NFileState::FileSize);
  204. NN_REQUIRE(IsFile());
  205. return mState.fileSize;
  206. }
  207. //=============================================================================
  208. // NFileInfo::Exists : Does the path exist?
  209. //-----------------------------------------------------------------------------
  210. bool NFileInfo::Exists(const NFilePath& thePath)
  211. {
  212. // Validate our parameters
  213. NN_REQUIRE(thePath.IsValid());
  214. // Get the state
  215. return NFileInfo(thePath).Exists();
  216. }
  217. //=============================================================================
  218. // NFileInfo::IsFile : Is the path a file?
  219. //-----------------------------------------------------------------------------
  220. bool NFileInfo::IsFile(const NFilePath& thePath)
  221. {
  222. // Validate our parameters
  223. NN_REQUIRE(thePath.IsValid());
  224. // Get the state
  225. return NFileInfo(thePath).IsFile();
  226. }
  227. //=============================================================================
  228. // NFileInfo::IsDirectory : Is the path a directory?
  229. //-----------------------------------------------------------------------------
  230. bool NFileInfo::IsDirectory(const NFilePath& thePath)
  231. {
  232. // Validate our parameters
  233. NN_REQUIRE(thePath.IsValid());
  234. // Get the state
  235. return NFileInfo(thePath).IsDirectory();
  236. }
  237. //=============================================================================
  238. // NFileInfo::CanRead : Can the path be read?
  239. //-----------------------------------------------------------------------------
  240. bool NFileInfo::CanRead(const NFilePath& thePath)
  241. {
  242. // Validate our parameters
  243. NN_REQUIRE(thePath.IsValid());
  244. // Get the state
  245. return NFileInfo(thePath).CanRead();
  246. }
  247. //=============================================================================
  248. // NFileInfo::CanWrite : Can the path be written?
  249. //-----------------------------------------------------------------------------
  250. bool NFileInfo::CanWrite(const NFilePath& thePath)
  251. {
  252. // Validate our parameters
  253. NN_REQUIRE(thePath.IsValid());
  254. // Get the state
  255. return NFileInfo(thePath).CanWrite();
  256. }
  257. //=============================================================================
  258. // NFileInfo::CanExecute : Can the path be executed / searched?
  259. //-----------------------------------------------------------------------------
  260. bool NFileInfo::CanExecute(const NFilePath& thePath)
  261. {
  262. // Validate our parameters
  263. NN_REQUIRE(thePath.IsValid());
  264. // Get the state
  265. return NFileInfo(thePath).CanExecute();
  266. }
  267. //=============================================================================
  268. // NFileInfo::GetCreationTime : Get the creation time.
  269. //-----------------------------------------------------------------------------
  270. NTime NFileInfo::GetCreationTime(const NFilePath& thePath)
  271. {
  272. // Validate our parameters
  273. NN_REQUIRE(thePath.IsValid());
  274. // Get the state
  275. return NFileInfo(thePath).GetCreationTime();
  276. }
  277. //=============================================================================
  278. // NFileInfo::GetModifiedTime : Get the last-modified time.
  279. //-----------------------------------------------------------------------------
  280. NTime NFileInfo::GetModifiedTime(const NFilePath& thePath)
  281. {
  282. // Validate our parameters
  283. NN_REQUIRE(thePath.IsValid());
  284. // Get the state
  285. return NFileInfo(thePath).GetModifiedTime();
  286. }
  287. //=============================================================================
  288. // NFileInfo::GetFileSize : Get the file size.
  289. //-----------------------------------------------------------------------------
  290. uint64_t NFileInfo::GetFileSize(const NFilePath& thePath)
  291. {
  292. // Validate our parameters
  293. NN_REQUIRE(thePath.IsValid());
  294. // Get the state
  295. return NFileInfo(thePath).GetFileSize();
  296. }
  297. #pragma mark private
  298. //=============================================================================
  299. // NFileInfo::TestFlag : Test a flag.
  300. //-----------------------------------------------------------------------------
  301. bool NFileInfo::TestFlag(NFileStateFlags theFlag) const
  302. {
  303. // Validate our state
  304. NN_REQUIRE(mValid & theFlag);
  305. // Test the flag
  306. return mState.theFlags & theFlag;
  307. }
  308. //=============================================================================
  309. // NFileInfo::UpdateState : Update the file state.
  310. //-----------------------------------------------------------------------------
  311. void NFileInfo::UpdateState(NFileStateFlags theFlags) const
  312. {
  313. // Update the state
  314. //
  315. // We can cast away const as our state is immutable once fetched.
  316. if ((mValid & theFlags) != theFlags)
  317. {
  318. NFileInfo* thisPtr = const_cast<NFileInfo*>(this);
  319. if (!thisPtr->FetchState(theFlags))
  320. {
  321. // Handle failure
  322. //
  323. // If we fail to acquire any information we assume we can't trust
  324. // any of the state, to avoid the platform-specific code from having
  325. // to return partial results.
  326. //
  327. // However, if an error does occur, the one thing we can say is that
  328. // the is-this / can-this states can be reported as false.
  329. thisPtr->mValid = kNFileStateMaskError;
  330. thisPtr->mState = {};
  331. }
  332. }
  333. }