GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/catalog.cc
Date: 2026-06-28 02:36:10
Exec Total Coverage
Lines: 407 439 92.7%
Branches: 324 606 53.5%

Line Branch Exec Source
1 /**
2 * This file is part of the CernVM File System.
3 */
4
5 #include "catalog.h"
6
7 #include <alloca.h>
8
9 #include <cassert>
10
11 #include "catalog_mgr.h"
12 #include "util/logging.h"
13 #include "util/smalloc.h"
14
15 using namespace std; // NOLINT
16
17 namespace catalog {
18
19 const shash::Md5 Catalog::kMd5PathEmpty("", 0);
20
21
22 /**
23 * Open a catalog outside the framework of a catalog manager.
24 */
25 663 Catalog *Catalog::AttachFreely(const string &imaginary_mountpoint,
26 const string &file,
27 const shash::Any &catalog_hash,
28 Catalog *parent,
29 const bool is_nested) {
30 Catalog *catalog = new Catalog(
31 663 PathString(imaginary_mountpoint.data(), imaginary_mountpoint.length()),
32 catalog_hash,
33 parent,
34
2/4
✓ Branch 1 taken 663 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 663 times.
✗ Branch 5 not taken.
663 is_nested);
35 663 const bool successful_init = catalog->InitStandalone(file);
36
2/2
✓ Branch 0 taken 11 times.
✓ Branch 1 taken 652 times.
663 if (!successful_init) {
37
1/2
✓ Branch 0 taken 11 times.
✗ Branch 1 not taken.
11 delete catalog;
38 11 return NULL;
39 }
40 652 return catalog;
41 }
42
43
44 4874 Catalog::Catalog(const PathString &mountpoint,
45 const shash::Any &catalog_hash,
46 Catalog *parent,
47 4874 const bool is_nested)
48 4874 : catalog_hash_(catalog_hash)
49
1/2
✓ Branch 1 taken 4874 times.
✗ Branch 2 not taken.
4874 , mountpoint_(mountpoint)
50
1/2
✓ Branch 1 taken 4874 times.
✗ Branch 2 not taken.
4874 , is_regular_mountpoint_(mountpoint_ == root_prefix_)
51 4874 , volatile_flag_(false)
52
3/4
✓ Branch 0 taken 2707 times.
✓ Branch 1 taken 2167 times.
✓ Branch 2 taken 2707 times.
✗ Branch 3 not taken.
4874 , is_root_(parent == NULL && !is_nested)
53 4874 , managed_database_(false)
54 4874 , parent_(parent)
55 4874 , nested_catalog_cache_dirty_(true)
56 4874 , voms_authz_status_(kVomsUnknown)
57
1/2
✓ Branch 8 taken 4874 times.
✗ Branch 9 not taken.
9748 , initialized_(false) {
58 4874 max_row_id_ = 0;
59 4874 inode_annotation_ = NULL;
60 4874 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
61 4874 const int retval = pthread_mutex_init(lock_, NULL);
62
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 4874 times.
4874 assert(retval == 0);
63
64 4874 database_ = NULL;
65 4874 uid_map_ = NULL;
66 4874 gid_map_ = NULL;
67 4874 sql_listing_ = NULL;
68 4874 sql_lookup_md5path_ = NULL;
69 4874 sql_lookup_nested_ = NULL;
70 4874 sql_list_nested_ = NULL;
71 4874 sql_own_list_nested_ = NULL;
72 4874 sql_all_chunks_ = NULL;
73 4874 sql_chunks_listing_ = NULL;
74 4874 sql_lookup_xattrs_ = NULL;
75 4874 }
76
77
78 12736 Catalog::~Catalog() {
79 9716 pthread_mutex_destroy(lock_);
80 9716 free(lock_);
81 9716 FinalizePreparedStatements();
82
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
9716 delete database_;
83 12736 }
84
85
86 /**
87 * InitPreparedStatement uses polymorphism in case of a r/w catalog.
88 * FinalizePreparedStatements is called in the destructor where
89 * polymorphism does not work any more and has to be called both in
90 * the WritableCatalog and the Catalog destructor
91 */
92 4786 void Catalog::InitPreparedStatements() {
93
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_listing_ = new SqlListing(database());
94
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_lookup_md5path_ = new SqlLookupPathHash(database());
95
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_lookup_nested_ = new SqlNestedCatalogLookup(database());
96
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_list_nested_ = new SqlNestedCatalogListing(database());
97
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_own_list_nested_ = new SqlOwnNestedCatalogListing(database());
98
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_all_chunks_ = new SqlAllChunks(database());
99
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_chunks_listing_ = new SqlChunksListing(database());
100
1/2
✓ Branch 3 taken 4786 times.
✗ Branch 4 not taken.
4786 sql_lookup_xattrs_ = new SqlLookupXattrs(database());
101 4786 }
102
103
104 4858 void Catalog::FinalizePreparedStatements() {
105
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_lookup_xattrs_;
106
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_chunks_listing_;
107
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_all_chunks_;
108
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_listing_;
109
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_lookup_md5path_;
110
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_lookup_nested_;
111
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_list_nested_;
112
2/2
✓ Branch 0 taken 4772 times.
✓ Branch 1 taken 86 times.
4858 delete sql_own_list_nested_;
113 4858 }
114
115
116 905 bool Catalog::InitStandalone(const std::string &database_file) {
117
1/2
✓ Branch 1 taken 905 times.
✗ Branch 2 not taken.
905 const bool retval = OpenDatabase(database_file);
118
2/2
✓ Branch 0 taken 11 times.
✓ Branch 1 taken 894 times.
905 if (!retval) {
119 11 return false;
120 }
121
122 894 InodeRange inode_range;
123 894 inode_range.MakeDummy();
124 894 set_inode_range(inode_range);
125 894 return true;
126 }
127
128
129 8205 bool Catalog::ReadCatalogCounters() {
130
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 8205 times.
8205 assert(database_ != NULL);
131 bool statistics_loaded;
132 8205 if (database().schema_version() < CatalogDatabase::kLatestSupportedSchema
133
2/2
✓ Branch 0 taken 22 times.
✓ Branch 1 taken 8183 times.
8205 - CatalogDatabase::kSchemaEpsilon) {
134 22 statistics_loaded = counters_.ReadFromDatabase(database(),
135 LegacyMode::kLegacy);
136
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 8183 times.
8183 } else if (database().schema_revision() < 2) {
137 statistics_loaded = counters_.ReadFromDatabase(database(),
138 LegacyMode::kNoXattrs);
139
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 8183 times.
8183 } else if (database().schema_revision() < 3) {
140 statistics_loaded = counters_.ReadFromDatabase(database(),
141 LegacyMode::kNoExternals);
142
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 8183 times.
8183 } else if (database().schema_revision() < 5) {
143 statistics_loaded = counters_.ReadFromDatabase(database(),
144 LegacyMode::kNoSpecials);
145 } else {
146 8183 statistics_loaded = counters_.ReadFromDatabase(database());
147 }
148 8205 return statistics_loaded;
149 }
150
151
152 /**
153 * Establishes the database structures and opens the sqlite database file.
154 * @param db_path the absolute path to the database file on local file system
155 * @return true on successful initialization otherwise false
156 */
157 4808 bool Catalog::OpenDatabase(const string &db_path) {
158
2/4
✓ Branch 1 taken 4808 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4808 times.
✗ Branch 5 not taken.
4808 database_ = CatalogDatabase::Open(db_path, DatabaseOpenMode());
159
2/2
✓ Branch 0 taken 22 times.
✓ Branch 1 taken 4786 times.
4808 if (NULL == database_) {
160 22 return false;
161 }
162
163
2/2
✓ Branch 2 taken 22 times.
✓ Branch 3 taken 4764 times.
4786 if (database_->IsEqualSchema(database_->schema_version(), 1.0)) {
164 // Possible fix-up for database layout lacking the content hash of
165 // nested catalogs
166 SqlCatalog sql_has_nested_sha1(
167 22 database(),
168 "SELECT count(*) FROM sqlite_master "
169
2/4
✓ Branch 2 taken 22 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 22 times.
✗ Branch 6 not taken.
66 "WHERE type='table' AND name='nested_catalogs' AND sql LIKE '%sha1%';");
170
1/2
✓ Branch 1 taken 22 times.
✗ Branch 2 not taken.
22 const bool retval = sql_has_nested_sha1.FetchRow();
171
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 22 times.
22 assert(retval == true);
172
1/2
✓ Branch 1 taken 22 times.
✗ Branch 2 not taken.
22 const bool has_nested_sha1 = sql_has_nested_sha1.RetrieveInt64(0);
173
2/2
✓ Branch 0 taken 11 times.
✓ Branch 1 taken 11 times.
22 if (!has_nested_sha1) {
174 11 database_->EnforceSchema(0.9, 0);
175 }
176 22 }
177
178
1/2
✓ Branch 1 taken 4786 times.
✗ Branch 2 not taken.
4786 InitPreparedStatements();
179
180 // Set the database file ownership if requested
181
2/2
✓ Branch 0 taken 226 times.
✓ Branch 1 taken 4560 times.
4786 if (managed_database_) {
182
1/2
✓ Branch 1 taken 226 times.
✗ Branch 2 not taken.
226 database_->TakeFileOwnership();
183 }
184
185 // Find out the maximum row id of this database file
186
2/4
✓ Branch 2 taken 4786 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 4786 times.
✗ Branch 7 not taken.
9572 SqlCatalog sql_max_row_id(database(), "SELECT MAX(rowid) FROM catalog;");
187
2/4
✓ Branch 1 taken 4786 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 4786 times.
4786 if (!sql_max_row_id.FetchRow()) {
188 LogCvmfs(kLogCatalog, kLogDebug,
189 "Cannot retrieve maximal row id for database file %s "
190 "(SqliteErrorcode: %d)",
191 db_path.c_str(), sql_max_row_id.GetLastError());
192 return false;
193 }
194
1/2
✓ Branch 1 taken 4786 times.
✗ Branch 2 not taken.
4786 max_row_id_ = sql_max_row_id.RetrieveInt64(0);
195
196 // Get root prefix
197
4/6
✓ Branch 2 taken 4786 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 4786 times.
✗ Branch 6 not taken.
✓ Branch 9 taken 2599 times.
✓ Branch 10 taken 2187 times.
4786 if (database_->HasProperty("root_prefix")) {
198 2599 const std::string root_prefix = database_->GetProperty<std::string>(
199
2/4
✓ Branch 2 taken 2599 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2599 times.
✗ Branch 6 not taken.
5198 "root_prefix");
200
1/2
✓ Branch 3 taken 2599 times.
✗ Branch 4 not taken.
2599 root_prefix_.Assign(root_prefix.data(), root_prefix.size());
201
1/2
✓ Branch 3 taken 2599 times.
✗ Branch 4 not taken.
2599 LogCvmfs(kLogCatalog, kLogDebug,
202 "found root prefix %s in root catalog file %s",
203 root_prefix_.c_str(), db_path.c_str());
204
1/2
✓ Branch 1 taken 2599 times.
✗ Branch 2 not taken.
2599 is_regular_mountpoint_ = (root_prefix_ == mountpoint_);
205 2599 } else {
206
1/2
✓ Branch 2 taken 2187 times.
✗ Branch 3 not taken.
2187 LogCvmfs(kLogCatalog, kLogDebug, "no root prefix for root catalog file %s",
207 db_path.c_str());
208 }
209
210 // Get volatile content flag
211
2/4
✓ Branch 1 taken 4786 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4786 times.
✗ Branch 5 not taken.
4786 volatile_flag_ = database_->GetPropertyDefault<bool>("volatile",
212 4786 volatile_flag_);
213
214 // Read Catalog Counter Statistics
215
2/4
✓ Branch 1 taken 4786 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 4786 times.
4786 if (!ReadCatalogCounters()) {
216 LogCvmfs(kLogCatalog, kLogStderr,
217 "failed to load statistics counters for catalog %s (file %s)",
218 mountpoint_.c_str(), db_path.c_str());
219 return false;
220 }
221
222
2/2
✓ Branch 1 taken 2167 times.
✓ Branch 2 taken 2619 times.
4786 if (HasParent()) {
223
1/2
✓ Branch 1 taken 2167 times.
✗ Branch 2 not taken.
2167 parent_->AddChild(this);
224 }
225
226 4786 initialized_ = true;
227 4786 return true;
228 4786 }
229
230
231 /**
232 * Removes the mountpoint and prepends the root prefix to path
233 */
234 29472 shash::Md5 Catalog::NormalizePath(const PathString &path) const {
235
2/2
✓ Branch 0 taken 29256 times.
✓ Branch 1 taken 216 times.
29472 if (is_regular_mountpoint_)
236
1/2
✓ Branch 3 taken 29256 times.
✗ Branch 4 not taken.
29256 return shash::Md5(path.GetChars(), path.GetLength());
237
238
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 216 times.
216 assert(path.GetLength() >= mountpoint_.GetLength());
239 // Piecewise hash calculation: root_prefix plus tail of path
240
1/2
✓ Branch 1 taken 216 times.
✗ Branch 2 not taken.
216 shash::Any result(shash::kMd5);
241
1/2
✓ Branch 1 taken 216 times.
✗ Branch 2 not taken.
216 shash::ContextPtr ctx(shash::kMd5);
242 216 ctx.buffer = alloca(ctx.size);
243
1/2
✓ Branch 1 taken 216 times.
✗ Branch 2 not taken.
216 shash::Init(ctx);
244
1/2
✓ Branch 2 taken 216 times.
✗ Branch 3 not taken.
432 shash::Update(
245 216 reinterpret_cast<const unsigned char *>(root_prefix_.GetChars()),
246 root_prefix_.GetLength(),
247 ctx);
248 216 shash::Update(reinterpret_cast<const unsigned char *>(path.GetChars())
249
1/2
✓ Branch 2 taken 216 times.
✗ Branch 3 not taken.
216 + mountpoint_.GetLength(),
250 216 path.GetLength() - mountpoint_.GetLength(),
251 ctx);
252
1/2
✓ Branch 1 taken 216 times.
✗ Branch 2 not taken.
216 shash::Final(ctx, &result);
253
1/2
✓ Branch 1 taken 216 times.
✗ Branch 2 not taken.
216 return result.CastToMd5();
254 }
255
256
257 /**
258 * Same as NormalizePath but returns a PathString instead of an Md5 hash.
259 */
260 2383 PathString Catalog::NormalizePath2(const PathString &path) const {
261
2/2
✓ Branch 0 taken 2339 times.
✓ Branch 1 taken 44 times.
2383 if (is_regular_mountpoint_)
262
1/2
✓ Branch 1 taken 2339 times.
✗ Branch 2 not taken.
2339 return path;
263
264
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 44 times.
44 assert(path.GetLength() >= mountpoint_.GetLength());
265
1/2
✓ Branch 1 taken 44 times.
✗ Branch 2 not taken.
44 PathString result = root_prefix_;
266
1/2
✓ Branch 2 taken 44 times.
✗ Branch 3 not taken.
44 const PathString suffix = path.Suffix(mountpoint_.GetLength());
267
1/2
✓ Branch 3 taken 44 times.
✗ Branch 4 not taken.
44 result.Append(suffix.GetChars(), suffix.GetLength());
268
1/2
✓ Branch 1 taken 44 times.
✗ Branch 2 not taken.
44 return result;
269 44 }
270
271
272 /**
273 * The opposite of NormalizePath: from a full path remove the root prefix and
274 * add the catalogs current mountpoint. Needed for normalized paths from the
275 * SQlite tables, such as nested catalog entry points.
276 */
277 2524 PathString Catalog::PlantPath(const PathString &path) const {
278
2/2
✓ Branch 0 taken 2480 times.
✓ Branch 1 taken 44 times.
2524 if (is_regular_mountpoint_)
279
1/2
✓ Branch 1 taken 2480 times.
✗ Branch 2 not taken.
2480 return path;
280
281
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 44 times.
44 assert(path.GetLength() >= root_prefix_.GetLength());
282
1/2
✓ Branch 1 taken 44 times.
✗ Branch 2 not taken.
44 PathString result = mountpoint_;
283
1/2
✓ Branch 2 taken 44 times.
✗ Branch 3 not taken.
44 const PathString suffix = path.Suffix(root_prefix_.GetLength());
284
1/2
✓ Branch 3 taken 44 times.
✗ Branch 4 not taken.
44 result.Append(suffix.GetChars(), suffix.GetLength());
285
1/2
✓ Branch 1 taken 44 times.
✗ Branch 2 not taken.
44 return result;
286 44 }
287
288
289 /**
290 * Performs a lookup on this Catalog for a given MD5 path hash.
291 * @param md5path the MD5 hash of the searched path
292 * @param expand_symlink indicates if variables in symlink should be resolved
293 * @param dirent will be set to the found DirectoryEntry
294 * @return true if DirectoryEntry was successfully found, false otherwise
295 */
296 26607 bool Catalog::LookupEntry(const shash::Md5 &md5path, const bool expand_symlink,
297 DirectoryEntry *dirent) const {
298
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 26607 times.
26607 assert(IsInitialized());
299
300 26607 const MutexLockGuard m(lock_);
301
1/2
✓ Branch 1 taken 26607 times.
✗ Branch 2 not taken.
26607 sql_lookup_md5path_->BindPathHash(md5path);
302
1/2
✓ Branch 1 taken 26607 times.
✗ Branch 2 not taken.
26607 const bool found = sql_lookup_md5path_->FetchRow();
303
4/4
✓ Branch 0 taken 25705 times.
✓ Branch 1 taken 902 times.
✓ Branch 2 taken 25661 times.
✓ Branch 3 taken 44 times.
26607 if (found && (dirent != NULL)) {
304
2/4
✓ Branch 1 taken 25661 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 25661 times.
✗ Branch 5 not taken.
25661 *dirent = sql_lookup_md5path_->GetDirent(this, expand_symlink);
305
1/2
✓ Branch 1 taken 25661 times.
✗ Branch 2 not taken.
25661 FixTransitionPoint(md5path, dirent);
306 }
307
1/2
✓ Branch 1 taken 26607 times.
✗ Branch 2 not taken.
26607 sql_lookup_md5path_->Reset();
308
309 26607 return found;
310 26607 }
311
312
313 /**
314 * Performs a lookup on this Catalog for a given MD5 path hash.
315 * @param md5path the MD5 hash of the searched path
316 * @param dirent will be set to the found DirectoryEntry
317 * @return true if DirectoryEntry was successfully found, false otherwise
318 */
319 26598 bool Catalog::LookupMd5Path(const shash::Md5 &md5path,
320 DirectoryEntry *dirent) const {
321 26598 return LookupEntry(md5path, true, dirent);
322 }
323
324
325 9 bool Catalog::LookupRawSymlink(const PathString &path,
326 LinkString *raw_symlink) const {
327
1/2
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
9 DirectoryEntry dirent;
328
2/4
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 9 times.
✗ Branch 5 not taken.
9 const bool result = (LookupEntry(NormalizePath(path), false, &dirent));
329
1/2
✓ Branch 0 taken 9 times.
✗ Branch 1 not taken.
9 if (result)
330
2/4
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 9 times.
✗ Branch 5 not taken.
9 raw_symlink->Assign(dirent.symlink());
331 9 return result;
332 9 }
333
334
335 9 bool Catalog::LookupXattrsMd5Path(const shash::Md5 &md5path,
336 XattrList *xattrs) const {
337
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 9 times.
9 assert(IsInitialized());
338
339 9 const MutexLockGuard m(lock_);
340
1/2
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
9 sql_lookup_xattrs_->BindPathHash(md5path);
341
1/2
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
9 const bool found = sql_lookup_xattrs_->FetchRow();
342
2/4
✓ Branch 0 taken 9 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 9 times.
✗ Branch 3 not taken.
9 if (found && (xattrs != NULL)) {
343
1/2
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
9 *xattrs = sql_lookup_xattrs_->GetXattrs();
344 }
345
1/2
✓ Branch 1 taken 9 times.
✗ Branch 2 not taken.
9 sql_lookup_xattrs_->Reset();
346
347 9 return found;
348 9 }
349
350
351 /**
352 * Perform a listing of the directory with the given MD5 path hash.
353 * @param path_hash the MD5 hash of the path of the directory to list
354 * @param listing will be set to the resulting DirectoryEntryList
355 * @return true on successful listing, false otherwise
356 */
357 127 bool Catalog::ListingMd5PathStat(const shash::Md5 &md5path,
358 StatEntryList *listing) const {
359
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 127 times.
127 assert(IsInitialized());
360
361
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 DirectoryEntry dirent;
362
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 StatEntry entry;
363
364 127 const MutexLockGuard m(lock_);
365
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 sql_listing_->BindPathHash(md5path);
366
3/4
✓ Branch 1 taken 475 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 348 times.
✓ Branch 4 taken 127 times.
475 while (sql_listing_->FetchRow()) {
367
2/4
✓ Branch 1 taken 348 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 348 times.
✗ Branch 5 not taken.
348 dirent = sql_listing_->GetDirent(this);
368
2/2
✓ Branch 1 taken 11 times.
✓ Branch 2 taken 337 times.
348 if (dirent.IsHidden())
369 11 continue;
370
1/2
✓ Branch 1 taken 337 times.
✗ Branch 2 not taken.
337 FixTransitionPoint(md5path, &dirent);
371
2/4
✓ Branch 1 taken 337 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 337 times.
✗ Branch 5 not taken.
337 entry.name = dirent.name();
372
1/2
✓ Branch 1 taken 337 times.
✗ Branch 2 not taken.
337 entry.info = dirent.GetStatStructure();
373
1/2
✓ Branch 1 taken 337 times.
✗ Branch 2 not taken.
337 listing->PushBack(entry);
374 }
375
1/2
✓ Branch 1 taken 127 times.
✗ Branch 2 not taken.
127 sql_listing_->Reset();
376
377 127 return true;
378 127 }
379
380
381 /**
382 * Perform a listing of the directory with the given MD5 path hash.
383 * Returns only struct stat values
384 * @param path_hash the MD5 hash of the path of the directory to list
385 * @param listing will be set to the resulting DirectoryEntryList
386 * @param expand_symlink defines if magic symlinks should be resolved
387 * @return true on successful listing, false otherwise
388 */
389 4001 bool Catalog::ListingMd5Path(const shash::Md5 &md5path,
390 DirectoryEntryList *listing,
391 const bool expand_symlink) const {
392
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 4001 times.
4001 assert(IsInitialized());
393
394 4001 const MutexLockGuard m(lock_);
395
396
1/2
✓ Branch 1 taken 4001 times.
✗ Branch 2 not taken.
4001 sql_listing_->BindPathHash(md5path);
397
3/4
✓ Branch 1 taken 11249 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 7248 times.
✓ Branch 4 taken 4001 times.
11249 while (sql_listing_->FetchRow()) {
398
1/2
✓ Branch 1 taken 7248 times.
✗ Branch 2 not taken.
7248 DirectoryEntry dirent = sql_listing_->GetDirent(this, expand_symlink);
399
1/2
✓ Branch 1 taken 7248 times.
✗ Branch 2 not taken.
7248 FixTransitionPoint(md5path, &dirent);
400
1/2
✓ Branch 1 taken 7248 times.
✗ Branch 2 not taken.
7248 listing->push_back(dirent);
401 7248 }
402
1/2
✓ Branch 1 taken 4001 times.
✗ Branch 2 not taken.
4001 sql_listing_->Reset();
403
404 4001 return true;
405 4001 }
406
407
408 22 bool Catalog::AllChunksBegin() { return sql_all_chunks_->Open(); }
409
410
411 110 bool Catalog::AllChunksNext(shash::Any *hash,
412 zlib::Algorithms *compression_alg) {
413 110 return sql_all_chunks_->Next(hash, compression_alg);
414 }
415
416
417 22 bool Catalog::AllChunksEnd() { return sql_all_chunks_->Close(); }
418
419
420 /**
421 * Hash algorithm is given by the unchunked file.
422 * Could be figured out by a join but it is faster if the user of this
423 * method tells us.
424 */
425 11 bool Catalog::ListMd5PathChunks(const shash::Md5 &md5path,
426 const shash::Algorithms interpret_hashes_as,
427 FileChunkList *chunks) const {
428
2/4
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
11 assert(IsInitialized() && chunks->IsEmpty());
429
430 11 const MutexLockGuard m(lock_);
431
432
1/2
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
11 sql_chunks_listing_->BindPathHash(md5path);
433
3/4
✓ Branch 1 taken 22 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 11 times.
✓ Branch 4 taken 11 times.
22 while (sql_chunks_listing_->FetchRow()) {
434
2/4
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
11 chunks->PushBack(sql_chunks_listing_->GetFileChunk(interpret_hashes_as));
435 }
436
1/2
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
11 sql_chunks_listing_->Reset();
437
438 11 return true;
439 11 }
440
441
442 /**
443 * Only used by the garbage collection
444 */
445 22 const Catalog::HashVector &Catalog::GetReferencedObjects() const {
446
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 22 times.
22 if (!referenced_hashes_.empty()) {
447 return referenced_hashes_;
448 }
449
450 // retrieve all referenced content hashes of both files and file chunks
451
1/2
✓ Branch 2 taken 22 times.
✗ Branch 3 not taken.
22 SqlListContentHashes list_content_hashes(database());
452
3/4
✓ Branch 1 taken 165 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 143 times.
✓ Branch 4 taken 22 times.
165 while (list_content_hashes.FetchRow()) {
453
2/4
✓ Branch 1 taken 143 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 143 times.
✗ Branch 5 not taken.
143 referenced_hashes_.push_back(list_content_hashes.GetHash());
454 }
455
456 22 return referenced_hashes_;
457 22 }
458
459
460 3879 void Catalog::TakeDatabaseFileOwnership() {
461 3879 managed_database_ = true;
462
2/2
✓ Branch 0 taken 3653 times.
✓ Branch 1 taken 226 times.
3879 if (NULL != database_) {
463 3653 database_->TakeFileOwnership();
464 }
465 3879 }
466
467
468 11 void Catalog::DropDatabaseFileOwnership() {
469 11 managed_database_ = false;
470
1/2
✓ Branch 0 taken 11 times.
✗ Branch 1 not taken.
11 if (NULL != database_) {
471 11 database_->DropFileOwnership();
472 }
473 11 }
474
475
476 886 uint64_t Catalog::GetTTL() const {
477 886 const MutexLockGuard m(lock_);
478
2/4
✓ Branch 3 taken 886 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 886 times.
✗ Branch 7 not taken.
1772 return database().GetPropertyDefault<uint64_t>("TTL", kDefaultTTL);
479 886 }
480
481
482 bool Catalog::HasExplicitTTL() const {
483 const MutexLockGuard m(lock_);
484 return database().HasProperty("TTL");
485 }
486
487
488 1835 bool Catalog::GetVOMSAuthz(string *authz) const {
489 bool result;
490 1835 const MutexLockGuard m(lock_);
491
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1835 times.
1835 if (voms_authz_status_ == kVomsPresent) {
492 if (authz) {
493 *authz = voms_authz_;
494 }
495 result = true;
496
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1835 times.
1835 } else if (voms_authz_status_ == kVomsNone) {
497 result = false;
498 } else {
499
3/6
✓ Branch 3 taken 1835 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 1835 times.
✗ Branch 7 not taken.
✗ Branch 10 not taken.
✓ Branch 11 taken 1835 times.
1835 if (database().HasProperty("voms_authz")) {
500 voms_authz_ = database().GetProperty<string>("voms_authz");
501 if (authz) {
502 *authz = voms_authz_;
503 }
504 voms_authz_status_ = kVomsPresent;
505 } else {
506 1835 voms_authz_status_ = kVomsNone;
507 }
508 1835 result = (voms_authz_status_ == kVomsPresent);
509 }
510 1835 return result;
511 1835 }
512
513 4906 uint64_t Catalog::GetRevision() const {
514 4906 const MutexLockGuard m(lock_);
515
2/4
✓ Branch 3 taken 4906 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 4906 times.
✗ Branch 7 not taken.
9812 return database().GetPropertyDefault<uint64_t>("revision", 0);
516 4906 }
517
518 1857 uint64_t Catalog::GetLastModified() const {
519
1/2
✓ Branch 2 taken 1857 times.
✗ Branch 3 not taken.
1857 const std::string prop_name = "last_modified";
520
1/2
✓ Branch 2 taken 1857 times.
✗ Branch 3 not taken.
1857 return (database().HasProperty(prop_name))
521
2/4
✓ Branch 0 taken 1857 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 1857 times.
✗ Branch 5 not taken.
1857 ? database().GetProperty<int>(prop_name)
522 1857 : 0u;
523 1857 }
524
525
526 uint64_t Catalog::GetNumChunks() const {
527 return counters_.Get("self_regular") + counters_.Get("self_chunks");
528 }
529
530
531 22 uint64_t Catalog::GetNumEntries() const {
532
1/2
✓ Branch 2 taken 22 times.
✗ Branch 3 not taken.
22 const string sql = "SELECT count(*) FROM catalog;";
533
534 22 const MutexLockGuard m(lock_);
535
1/2
✓ Branch 2 taken 22 times.
✗ Branch 3 not taken.
22 SqlCatalog stmt(database(), sql);
536
3/6
✓ Branch 1 taken 22 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 22 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 22 times.
✗ Branch 7 not taken.
44 return (stmt.FetchRow()) ? stmt.RetrieveInt64(0) : 0;
537 22 }
538
539
540 22 shash::Any Catalog::GetPreviousRevision() const {
541 22 const MutexLockGuard m(lock_);
542 22 const std::string hash_string = database().GetPropertyDefault<std::string>(
543
3/6
✓ Branch 2 taken 22 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 22 times.
✗ Branch 7 not taken.
✓ Branch 9 taken 22 times.
✗ Branch 10 not taken.
44 "previous_revision", "");
544 22 return (!hash_string.empty())
545
1/2
✓ Branch 2 taken 11 times.
✗ Branch 3 not taken.
11 ? shash::MkFromHexPtr(shash::HexPtr(hash_string),
546 shash::kSuffixCatalog)
547
3/4
✓ Branch 0 taken 11 times.
✓ Branch 1 taken 11 times.
✓ Branch 3 taken 11 times.
✗ Branch 4 not taken.
44 : shash::Any();
548 22 }
549
550
551 11 string Catalog::PrintMemStatistics() const {
552 11 sqlite::MemStatistics stats;
553 {
554 11 const MutexLockGuard m(lock_);
555
1/2
✓ Branch 2 taken 11 times.
✗ Branch 3 not taken.
11 database().GetMemStatistics(&stats);
556 11 }
557
4/8
✓ Branch 2 taken 11 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 11 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 11 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 11 times.
✗ Branch 14 not taken.
22 return string(mountpoint().GetChars(), mountpoint().GetLength()) + ": "
558
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.lookaside_slots_used) + " / "
559
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.lookaside_slots_max) + " slots -- "
560
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.lookaside_hit) + " hits, "
561
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.lookaside_miss_size) + " misses-size, "
562
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.lookaside_miss_full) + " misses-full -- "
563
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.page_cache_used / 1024) + " kB pages -- "
564
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.page_cache_hit) + " hits, "
565
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.page_cache_miss) + " misses -- "
566
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.schema_used / 1024) + " kB schema -- "
567
3/6
✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 11 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 11 times.
✗ Branch 8 not taken.
44 + StringifyInt(stats.stmt_used / 1024) + " kB statements";
568 }
569
570
571 /**
572 * Determine the actual inode of a DirectoryEntry.
573 * The first used entry from a hardlink group deterimines the inode of the
574 * others.
575 * @param row_id the row id of a read row in the sqlite database
576 * @param hardlink_group the id of a possibly present hardlink group
577 * @return the assigned inode number
578 */
579 33257 inode_t Catalog::GetMangledInode(const uint64_t row_id,
580 const uint64_t hardlink_group) const {
581
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 33257 times.
33257 assert(IsInitialized());
582
583
2/2
✓ Branch 1 taken 386 times.
✓ Branch 2 taken 32871 times.
33257 if (inode_range_.IsDummy()) {
584 386 return DirectoryEntry::kInvalidInode;
585 }
586
587 32871 inode_t inode = row_id + inode_range_.offset;
588
589 // Hardlinks are encoded in catalog-wide unique hard link group ids.
590 // These ids must be resolved to actual inode relationships at runtime.
591
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 32871 times.
32871 if (hardlink_group > 0) {
592 const HardlinkGroupMap::const_iterator inode_iter = hardlink_groups_.find(
593 hardlink_group);
594
595 // Use cached entry if possible
596 if (inode_iter == hardlink_groups_.end()) {
597 hardlink_groups_[hardlink_group] = inode;
598 } else {
599 inode = inode_iter->second;
600 }
601 }
602
603
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 32871 times.
32871 if (inode_annotation_) {
604 inode = inode_annotation_->Annotate(inode);
605 }
606
607 32871 return inode;
608 }
609
610
611 /**
612 * Get a list of all registered nested catalogs and bind mountpoints in this
613 * catalog.
614 * @return a list of all nested catalog and bind mountpoints.
615 */
616 18241 const Catalog::NestedCatalogList &Catalog::ListNestedCatalogs() const {
617 18241 const MutexLockGuard m(lock_);
618
619
2/2
✓ Branch 0 taken 4246 times.
✓ Branch 1 taken 13995 times.
18241 if (nested_catalog_cache_dirty_) {
620
1/2
✓ Branch 2 taken 4246 times.
✗ Branch 3 not taken.
4246 LogCvmfs(kLogCatalog, kLogDebug, "refreshing nested catalog cache of '%s'",
621
1/2
✓ Branch 1 taken 4246 times.
✗ Branch 2 not taken.
8492 mountpoint().c_str());
622
3/4
✓ Branch 1 taken 6505 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 2259 times.
✓ Branch 4 taken 4246 times.
6505 while (sql_list_nested_->FetchRow()) {
623
1/2
✓ Branch 1 taken 2259 times.
✗ Branch 2 not taken.
2259 NestedCatalog nested;
624
3/6
✓ Branch 1 taken 2259 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2259 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 2259 times.
✗ Branch 8 not taken.
2259 nested.mountpoint = PlantPath(sql_list_nested_->GetPath());
625
1/2
✓ Branch 1 taken 2259 times.
✗ Branch 2 not taken.
2259 nested.hash = sql_list_nested_->GetContentHash();
626
1/2
✓ Branch 1 taken 2259 times.
✗ Branch 2 not taken.
2259 nested.size = sql_list_nested_->GetSize();
627
1/2
✓ Branch 1 taken 2259 times.
✗ Branch 2 not taken.
2259 nested_catalog_cache_.push_back(nested);
628 2259 }
629
1/2
✓ Branch 1 taken 4246 times.
✗ Branch 2 not taken.
4246 sql_list_nested_->Reset();
630 4246 nested_catalog_cache_dirty_ = false;
631 }
632
633 18241 return nested_catalog_cache_;
634 18241 }
635
636
637 /**
638 * Get a list of all registered nested catalogs without bind mountpoints. Used
639 * for replication and garbage collection.
640 * @return a list of all nested catalogs.
641 */
642 1584 const Catalog::NestedCatalogList Catalog::ListOwnNestedCatalogs() const {
643 1584 NestedCatalogList result;
644
645 1584 const MutexLockGuard m(lock_);
646
647
3/4
✓ Branch 1 taken 1783 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 199 times.
✓ Branch 4 taken 1584 times.
1783 while (sql_own_list_nested_->FetchRow()) {
648
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 NestedCatalog nested;
649
3/6
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 199 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 199 times.
✗ Branch 8 not taken.
199 nested.mountpoint = PlantPath(sql_own_list_nested_->GetPath());
650
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 nested.hash = sql_own_list_nested_->GetContentHash();
651
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 nested.size = sql_own_list_nested_->GetSize();
652
1/2
✓ Branch 1 taken 199 times.
✗ Branch 2 not taken.
199 result.push_back(nested);
653 199 }
654
1/2
✓ Branch 1 taken 1584 times.
✗ Branch 2 not taken.
1584 sql_own_list_nested_->Reset();
655
656 3168 return result;
657 1584 }
658
659
660 /**
661 * Drops the nested catalog cache. Usually this is only useful in subclasses
662 * that implement writable catalogs.
663 *
664 * Note: this action is _not_ secured by the catalog's mutex. If serialisation
665 * is required the subclass needs to ensure that.
666 */
667 3085 void Catalog::ResetNestedCatalogCacheUnprotected() {
668 3085 nested_catalog_cache_.clear();
669 3085 nested_catalog_cache_dirty_ = true;
670 3085 }
671
672
673 /**
674 * Looks for a specific registered nested catalog based on a path.
675 */
676 2317 bool Catalog::FindNested(const PathString &mountpoint, shash::Any *hash,
677 uint64_t *size) const {
678 2317 const MutexLockGuard m(lock_);
679
1/2
✓ Branch 1 taken 2317 times.
✗ Branch 2 not taken.
2317 const PathString normalized_mountpoint = NormalizePath2(mountpoint);
680
1/2
✓ Branch 1 taken 2317 times.
✗ Branch 2 not taken.
2317 sql_lookup_nested_->BindSearchPath(normalized_mountpoint);
681
1/2
✓ Branch 1 taken 2317 times.
✗ Branch 2 not taken.
2317 const bool found = sql_lookup_nested_->FetchRow();
682
3/4
✓ Branch 0 taken 2263 times.
✓ Branch 1 taken 54 times.
✓ Branch 2 taken 2263 times.
✗ Branch 3 not taken.
2317 if (found && (hash != NULL)) {
683
1/2
✓ Branch 1 taken 2263 times.
✗ Branch 2 not taken.
2263 *hash = sql_lookup_nested_->GetContentHash();
684
1/2
✓ Branch 1 taken 2263 times.
✗ Branch 2 not taken.
2263 *size = sql_lookup_nested_->GetSize();
685 }
686
1/2
✓ Branch 1 taken 2317 times.
✗ Branch 2 not taken.
2317 sql_lookup_nested_->Reset();
687
688 2317 return found;
689 2317 }
690
691
692 /**
693 * Sets a new object to do inode annotations (or set to NULL)
694 * The annotation object is not owned by the catalog.
695 */
696 3881 void Catalog::SetInodeAnnotation(InodeAnnotation *new_annotation) {
697 3881 const MutexLockGuard m(lock_);
698 // Since annotated inodes could come back to the catalog in order to
699 // get stripped, exchanging the annotation is not allowed
700
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 3881 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
3881 assert((inode_annotation_ == NULL) || (inode_annotation_ == new_annotation));
701 3881 inode_annotation_ = new_annotation;
702 3881 }
703
704
705 3881 void Catalog::SetOwnerMaps(const OwnerMap *uid_map, const OwnerMap *gid_map) {
706
2/4
✓ Branch 0 taken 3881 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 3881 times.
3881 uid_map_ = (uid_map && uid_map->HasEffect()) ? uid_map : NULL;
707
2/4
✓ Branch 0 taken 3881 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 3881 times.
3881 gid_map_ = (gid_map && gid_map->HasEffect()) ? gid_map : NULL;
708 3881 }
709
710
711 /**
712 * Add a Catalog as child to this Catalog.
713 * @param child the Catalog to define as child
714 */
715 2167 void Catalog::AddChild(Catalog *child) {
716
3/7
✓ Branch 1 taken 2167 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 2167 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2167 times.
2167 assert(NULL == FindChild(child->mountpoint()));
717
718 2167 const MutexLockGuard m(lock_);
719
2/4
✓ Branch 1 taken 2167 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2167 times.
✗ Branch 5 not taken.
2167 children_[child->mountpoint()] = child;
720 2167 child->set_parent(this);
721 2167 }
722
723
724 /**
725 * Removes a Catalog from the children list of this Catalog
726 * @param child the Catalog to delete as child
727 */
728 2044 void Catalog::RemoveChild(Catalog *child) {
729
3/7
✓ Branch 1 taken 2044 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 2044 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 2044 times.
2044 assert(NULL != FindChild(child->mountpoint()));
730
731 2044 const MutexLockGuard m(lock_);
732 2044 child->set_parent(NULL);
733
2/4
✓ Branch 1 taken 2044 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 2044 times.
✗ Branch 5 not taken.
2044 children_.erase(child->mountpoint());
734 2044 }
735
736
737 6087 CatalogList Catalog::GetChildren() const {
738 6087 CatalogList result;
739
740 6087 const MutexLockGuard m(lock_);
741 12174 for (NestedCatalogMap::const_iterator i = children_.begin(),
742 6087 iEnd = children_.end();
743
2/2
✓ Branch 1 taken 3120 times.
✓ Branch 2 taken 6087 times.
9207 i != iEnd;
744 3120 ++i) {
745
1/2
✓ Branch 2 taken 3120 times.
✗ Branch 3 not taken.
3120 result.push_back(i->second);
746 }
747
748 12174 return result;
749 6087 }
750
751
752 /**
753 * Find the nested catalog that serves the given path.
754 * It might be possible that the path is in fact served by a child of the found
755 * nested catalog.
756 * @param path the path to find a best fitting catalog for
757 * @return a pointer to the best fitting child or NULL if it does not fit at all
758 */
759 20778 Catalog *Catalog::FindSubtree(const PathString &path) const {
760 // Check if this catalog fits the beginning of the path.
761
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 20778 times.
20778 if (!path.StartsWith(mountpoint_))
762 return NULL;
763
764
1/2
✓ Branch 2 taken 20778 times.
✗ Branch 3 not taken.
20778 PathString remaining(path.Suffix(mountpoint_.GetLength()));
765
1/2
✓ Branch 1 taken 20778 times.
✗ Branch 2 not taken.
20778 remaining.Append("/", 1);
766
767 // now we recombine the path elements successively
768 // in order to find a child which serves a part of the path
769
1/2
✓ Branch 1 taken 20778 times.
✗ Branch 2 not taken.
20778 PathString path_prefix(mountpoint_);
770 20778 Catalog *result = NULL;
771 // Skip the first '/'
772
1/2
✓ Branch 1 taken 20778 times.
✗ Branch 2 not taken.
20778 path_prefix.Append("/", 1);
773 20778 const char *c = remaining.GetChars() + 1;
774
2/2
✓ Branch 1 taken 192645 times.
✓ Branch 2 taken 16361 times.
209006 for (unsigned i = 1; i < remaining.GetLength(); ++i, ++c) {
775
2/2
✓ Branch 0 taken 43222 times.
✓ Branch 1 taken 149423 times.
192645 if (*c == '/') {
776
1/2
✓ Branch 1 taken 43222 times.
✗ Branch 2 not taken.
43222 result = FindChild(path_prefix);
777
778 // If we found a child serving a part of the path we can stop searching.
779 // Remaining sub path elements are possibly served by a grand child.
780
2/2
✓ Branch 0 taken 4417 times.
✓ Branch 1 taken 38805 times.
43222 if (result != NULL)
781 4417 break;
782 }
783
1/2
✓ Branch 1 taken 188228 times.
✗ Branch 2 not taken.
188228 path_prefix.Append(c, 1);
784 }
785
786 20778 return result;
787 20778 }
788
789
790 /**
791 * Looks for a child catalog, which is a subset of all registered nested
792 * catalogs.
793 */
794 47950 Catalog *Catalog::FindChild(const PathString &mountpoint) const {
795 47950 NestedCatalogMap::const_iterator nested_iter;
796
797 47950 const MutexLockGuard m(lock_);
798
1/2
✓ Branch 1 taken 47950 times.
✗ Branch 2 not taken.
47950 nested_iter = children_.find(mountpoint);
799
2/2
✓ Branch 2 taken 41316 times.
✓ Branch 3 taken 6634 times.
54584 Catalog *result = (nested_iter == children_.end()) ? NULL
800 6634 : nested_iter->second;
801
802 47950 return result;
803 47950 }
804
805
806 /**
807 * For the transition points for nested catalogs and bind mountpoints, the inode
808 * is ambiguous. It has to be set to the parent inode because nested catalogs
809 * are lazily loaded.
810 * @param md5path the MD5 hash of the entry to check
811 * @param dirent the DirectoryEntry to perform coherence fixes on
812 */
813 33246 void Catalog::FixTransitionPoint(const shash::Md5 &md5path,
814 DirectoryEntry *dirent) const {
815
2/2
✓ Branch 1 taken 23891 times.
✓ Branch 2 taken 9355 times.
33246 if (!HasParent())
816 23891 return;
817
818
2/2
✓ Branch 1 taken 1177 times.
✓ Branch 2 taken 8178 times.
9355 if (dirent->IsNestedCatalogRoot()) {
819 // Normal nested catalog
820
1/2
✓ Branch 1 taken 1177 times.
✗ Branch 2 not taken.
1177 DirectoryEntry parent_dirent;
821
1/2
✓ Branch 1 taken 1177 times.
✗ Branch 2 not taken.
1177 const bool retval = parent_->LookupMd5Path(md5path, &parent_dirent);
822
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1177 times.
1177 assert(retval);
823 1177 dirent->set_inode(parent_dirent.inode());
824
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 8178 times.
9355 } else if (md5path == kMd5PathEmpty) {
825 // Bind mountpoint
826 DirectoryEntry parent_dirent;
827 const bool retval = parent_->LookupPath(mountpoint_, &parent_dirent);
828 assert(retval);
829 dirent->set_inode(parent_dirent.inode());
830 }
831 }
832
833 } // namespace catalog
834