local-store.hh 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. #pragma once
  2. #include "sqlite.hh"
  3. #include <string>
  4. #include <unordered_set>
  5. #include "pathlocks.hh"
  6. #include "store-api.hh"
  7. #include "util.hh"
  8. namespace nix {
  9. /* Nix store and database schema version. Version 1 (or 0) was Nix <=
  10. 0.7. Version 2 was Nix 0.8 and 0.9. Version 3 is Nix 0.10.
  11. Version 4 is Nix 0.11. Version 5 is Nix 0.12-0.16. Version 6 is
  12. Nix 1.0. Version 7 is Nix 1.3. Guix has always used version 7. */
  13. const int nixSchemaVersion = 7;
  14. extern string drvsLogDir;
  15. struct Derivation;
  16. struct OptimiseStats
  17. {
  18. unsigned long filesLinked;
  19. unsigned long long bytesFreed;
  20. unsigned long long blocksFreed;
  21. OptimiseStats()
  22. {
  23. filesLinked = 0;
  24. bytesFreed = blocksFreed = 0;
  25. }
  26. };
  27. struct RunningSubstituter
  28. {
  29. Pid pid;
  30. AutoCloseFD to, from, error;
  31. FdSource fromBuf;
  32. bool disabled;
  33. RunningSubstituter() : disabled(false) { };
  34. };
  35. class LocalStore : public StoreAPI
  36. {
  37. private:
  38. /* The currently running substituter or empty. */
  39. std::unique_ptr<RunningSubstituter> runningSubstituter;
  40. Path linksDir;
  41. public:
  42. /* Initialise the local store, upgrading the schema if
  43. necessary. */
  44. LocalStore(bool reserveSpace = true);
  45. ~LocalStore();
  46. /* Implementations of abstract store API methods. */
  47. bool isValidPath(const Path & path);
  48. PathSet queryValidPaths(const PathSet & paths);
  49. PathSet queryAllValidPaths();
  50. ValidPathInfo queryPathInfo(const Path & path);
  51. Hash queryPathHash(const Path & path);
  52. void queryReferences(const Path & path, PathSet & references);
  53. void queryReferrers(const Path & path, PathSet & referrers);
  54. Path queryDeriver(const Path & path);
  55. PathSet queryValidDerivers(const Path & path);
  56. PathSet queryDerivationOutputs(const Path & path);
  57. StringSet queryDerivationOutputNames(const Path & path);
  58. Path queryPathFromHashPart(const string & hashPart);
  59. PathSet querySubstitutablePaths(const PathSet & paths);
  60. void querySubstitutablePathInfos(PathSet & paths,
  61. SubstitutablePathInfos & infos);
  62. void querySubstitutablePathInfos(const PathSet & paths,
  63. SubstitutablePathInfos & infos);
  64. Path addToStore(const string & name, const Path & srcPath,
  65. bool recursive = true, HashType hashAlgo = htSHA256,
  66. PathFilter & filter = defaultPathFilter, bool repair = false);
  67. /* Like addToStore(), but the contents of the path are contained
  68. in `dump', which is either a NAR serialisation (if recursive ==
  69. true) or simply the contents of a regular file (if recursive ==
  70. false). */
  71. Path addToStoreFromDump(const string & dump, const string & name,
  72. bool recursive = true, HashType hashAlgo = htSHA256, bool repair = false);
  73. Path addTextToStore(const string & name, const string & s,
  74. const PathSet & references, bool repair = false);
  75. void exportPath(const Path & path, bool sign,
  76. Sink & sink);
  77. Paths importPaths(bool requireSignature, Source & source);
  78. void buildPaths(const PathSet & paths, BuildMode buildMode);
  79. void ensurePath(const Path & path);
  80. void addTempRoot(const Path & path);
  81. void addIndirectRoot(const Path & path);
  82. void syncWithGC();
  83. Roots findRoots();
  84. void collectGarbage(const GCOptions & options, GCResults & results);
  85. /* Optimise the disk space usage of the Nix store by hard-linking
  86. files with the same contents. */
  87. void optimiseStore(OptimiseStats & stats);
  88. /* Generic variant of the above method. */
  89. void optimiseStore();
  90. /* Optimise a single store path. */
  91. void optimisePath(const Path & path);
  92. /* Check the integrity of the Nix store. Returns true if errors
  93. remain. */
  94. bool verifyStore(bool checkContents, bool repair);
  95. /* Register the validity of a path, i.e., that `path' exists, that
  96. the paths referenced by it exists, and in the case of an output
  97. path of a derivation, that it has been produced by a successful
  98. execution of the derivation (or something equivalent). Also
  99. register the hash of the file system contents of the path. The
  100. hash must be a SHA-256 hash. */
  101. void registerValidPath(const ValidPathInfo & info);
  102. void registerValidPaths(const ValidPathInfos & infos);
  103. /* Register that the build of a derivation with output `path' has
  104. failed. */
  105. void registerFailedPath(const Path & path);
  106. /* Query whether `path' previously failed to build. */
  107. bool hasPathFailed(const Path & path);
  108. PathSet queryFailedPaths();
  109. void clearFailedPaths(const PathSet & paths);
  110. void vacuumDB();
  111. /* Repair the contents of the given path by redownloading it using
  112. a substituter (if available). */
  113. void repairPath(const Path & path);
  114. /* Check whether the given valid path exists and has the right
  115. contents. */
  116. bool pathContentsGood(const Path & path);
  117. void markContentsGood(const Path & path);
  118. void setSubstituterEnv();
  119. void createUser(const std::string & userName, uid_t userId);
  120. private:
  121. Path schemaPath;
  122. /* Lock file used for upgrading. */
  123. AutoCloseFD globalLock;
  124. /* The SQLite database object. */
  125. SQLite db;
  126. /* Some precompiled SQLite statements. */
  127. SQLiteStmt stmtRegisterValidPath;
  128. SQLiteStmt stmtUpdatePathInfo;
  129. SQLiteStmt stmtAddReference;
  130. SQLiteStmt stmtQueryPathInfo;
  131. SQLiteStmt stmtQueryReferences;
  132. SQLiteStmt stmtQueryReferrers;
  133. SQLiteStmt stmtInvalidatePath;
  134. SQLiteStmt stmtRegisterFailedPath;
  135. SQLiteStmt stmtHasPathFailed;
  136. SQLiteStmt stmtQueryFailedPaths;
  137. SQLiteStmt stmtClearFailedPath;
  138. SQLiteStmt stmtAddDerivationOutput;
  139. SQLiteStmt stmtQueryValidDerivers;
  140. SQLiteStmt stmtQueryDerivationOutputs;
  141. SQLiteStmt stmtQueryPathFromHashPart;
  142. SQLiteStmt stmtQueryValidPaths;
  143. /* Cache for pathContentsGood(). */
  144. std::map<Path, bool> pathContentsGoodCache;
  145. bool didSetSubstituterEnv;
  146. /* The file to which we write our temporary roots. */
  147. Path fnTempRoots;
  148. AutoCloseFD fdTempRoots;
  149. int getSchema();
  150. void openDB(bool create);
  151. void makeStoreWritable();
  152. uint64_t queryValidPathId(const Path & path);
  153. uint64_t addValidPath(const ValidPathInfo & info, bool checkOutputs = true);
  154. void addReference(uint64_t referrer, uint64_t reference);
  155. void appendReferrer(const Path & from, const Path & to, bool lock);
  156. void rewriteReferrers(const Path & path, bool purge, PathSet referrers);
  157. void invalidatePath(const Path & path);
  158. /* Delete a path from the Nix store. */
  159. void invalidatePathChecked(const Path & path);
  160. void verifyPath(const Path & path, const PathSet & store,
  161. PathSet & done, PathSet & validPaths, bool repair, bool & errors);
  162. void updatePathInfo(const ValidPathInfo & info);
  163. struct GCState;
  164. void deleteGarbage(GCState & state, const Path & path);
  165. void tryToDelete(GCState & state, const Path & path);
  166. bool canReachRoot(GCState & state, PathSet & visited, const Path & path);
  167. void deletePathRecursive(GCState & state, const Path & path);
  168. bool isActiveTempFile(const GCState & state,
  169. const Path & path, const string & suffix);
  170. int openGCLock(LockType lockType);
  171. void removeUnusedLinks(const GCState & state);
  172. void startSubstituter(RunningSubstituter & runningSubstituter);
  173. string getLineFromSubstituter(RunningSubstituter & run);
  174. template<class T> T getIntLineFromSubstituter(RunningSubstituter & run);
  175. Path createTempDirInStore();
  176. Path importPath(bool requireSignature, Source & source);
  177. void checkDerivationOutputs(const Path & drvPath, const Derivation & drv);
  178. typedef std::unordered_set<ino_t> InodeHash;
  179. InodeHash loadInodeHash();
  180. Strings readDirectoryIgnoringInodes(const Path & path, const InodeHash & inodeHash);
  181. void optimisePath_(OptimiseStats & stats, const Path & path, InodeHash & inodeHash);
  182. // Internal versions that are not wrapped in retry_sqlite.
  183. bool isValidPath_(const Path & path);
  184. void queryReferrers_(const Path & path, PathSet & referrers);
  185. };
  186. typedef std::pair<dev_t, ino_t> Inode;
  187. typedef set<Inode> InodesSeen;
  188. /* "Fix", or canonicalise, the meta-data of the files in a store path
  189. after it has been built. In particular:
  190. - the last modification date on each file is set to 1 (i.e.,
  191. 00:00:01 1/1/1970 UTC)
  192. - the permissions are set of 444 or 555 (i.e., read-only with or
  193. without execute permission; setuid bits etc. are cleared)
  194. - the owner and group are set to the Nix user and group, if we're
  195. running as root. */
  196. void canonicalisePathMetaData(const Path & path, uid_t fromUid, InodesSeen & inodesSeen);
  197. void canonicalisePathMetaData(const Path & path, uid_t fromUid);
  198. void canonicaliseTimestampAndPermissions(const Path & path);
  199. MakeError(PathInUse, Error);
  200. }