GCC Code Coverage Report


Directory: cvmfs/
File: cvmfs/catalog.cc
Date: 2026-05-19 11:45:12
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 666 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 666 PathString(imaginary_mountpoint.data(), imaginary_mountpoint.length()),
32 catalog_hash,
33 parent,
34
2/4
✓ Branch 1 taken 666 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 666 times.
✗ Branch 5 not taken.
666 is_nested);
35 666 const bool successful_init = catalog->InitStandalone(file);
36
2/2
✓ Branch 0 taken 31 times.
✓ Branch 1 taken 635 times.
666 if (!successful_init) {
37
1/2
✓ Branch 0 taken 31 times.
✗ Branch 1 not taken.
31 delete catalog;
38 31 return NULL;
39 }
40 635 return catalog;
41 }
42
43
44 4724 Catalog::Catalog(const PathString &mountpoint,
45 const shash::Any &catalog_hash,
46 Catalog *parent,
47 4724 const bool is_nested)
48 4724 : catalog_hash_(catalog_hash)
49
1/2
✓ Branch 1 taken 4724 times.
✗ Branch 2 not taken.
4724 , mountpoint_(mountpoint)
50
1/2
✓ Branch 1 taken 4724 times.
✗ Branch 2 not taken.
4724 , is_regular_mountpoint_(mountpoint_ == root_prefix_)
51 4724 , volatile_flag_(false)
52
3/4
✓ Branch 0 taken 2876 times.
✓ Branch 1 taken 1848 times.
✓ Branch 2 taken 2876 times.
✗ Branch 3 not taken.
4724 , is_root_(parent == NULL && !is_nested)
53 4724 , managed_database_(false)
54 4724 , parent_(parent)
55 4724 , nested_catalog_cache_dirty_(true)
56 4724 , voms_authz_status_(kVomsUnknown)
57
1/2
✓ Branch 8 taken 4724 times.
✗ Branch 9 not taken.
9448 , initialized_(false) {
58 4724 max_row_id_ = 0;
59 4724 inode_annotation_ = NULL;
60 4724 lock_ = reinterpret_cast<pthread_mutex_t *>(smalloc(sizeof(pthread_mutex_t)));
61 4724 const int retval = pthread_mutex_init(lock_, NULL);
62
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 4724 times.
4724 assert(retval == 0);
63
64 4724 database_ = NULL;
65 4724 uid_map_ = NULL;
66 4724 gid_map_ = NULL;
67 4724 sql_listing_ = NULL;
68 4724 sql_lookup_md5path_ = NULL;
69 4724 sql_lookup_nested_ = NULL;
70 4724 sql_list_nested_ = NULL;
71 4724 sql_own_list_nested_ = NULL;
72 4724 sql_all_chunks_ = NULL;
73 4724 sql_chunks_listing_ = NULL;
74 4724 sql_lookup_xattrs_ = NULL;
75 4724 }
76
77
78 12278 Catalog::~Catalog() {
79 9372 pthread_mutex_destroy(lock_);
80 9372 free(lock_);
81 9372 FinalizePreparedStatements();
82
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
9372 delete database_;
83 12278 }
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 4462 void Catalog::InitPreparedStatements() {
93
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_listing_ = new SqlListing(database());
94
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_lookup_md5path_ = new SqlLookupPathHash(database());
95
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_lookup_nested_ = new SqlNestedCatalogLookup(database());
96
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_list_nested_ = new SqlNestedCatalogListing(database());
97
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_own_list_nested_ = new SqlOwnNestedCatalogListing(database());
98
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_all_chunks_ = new SqlAllChunks(database());
99
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_chunks_listing_ = new SqlChunksListing(database());
100
1/2
✓ Branch 3 taken 4462 times.
✗ Branch 4 not taken.
4462 sql_lookup_xattrs_ = new SqlLookupXattrs(database());
101 4462 }
102
103
104 4686 void Catalog::FinalizePreparedStatements() {
105
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_lookup_xattrs_;
106
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_chunks_listing_;
107
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_all_chunks_;
108
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_listing_;
109
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_lookup_md5path_;
110
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_lookup_nested_;
111
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_list_nested_;
112
2/2
✓ Branch 0 taken 4426 times.
✓ Branch 1 taken 260 times.
4686 delete sql_own_list_nested_;
113 4686 }
114
115
116 1378 bool Catalog::InitStandalone(const std::string &database_file) {
117
1/2
✓ Branch 1 taken 1378 times.
✗ Branch 2 not taken.
1378 const bool retval = OpenDatabase(database_file);
118
2/2
✓ Branch 0 taken 31 times.
✓ Branch 1 taken 1347 times.
1378 if (!retval) {
119 31 return false;
120 }
121
122 1347 InodeRange inode_range;
123 1347 inode_range.MakeDummy();
124 1347 set_inode_range(inode_range);
125 1347 return true;
126 }
127
128
129 7183 bool Catalog::ReadCatalogCounters() {
130
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 7183 times.
7183 assert(database_ != NULL);
131 bool statistics_loaded;
132 7183 if (database().schema_version() < CatalogDatabase::kLatestSupportedSchema
133
2/2
✓ Branch 0 taken 64 times.
✓ Branch 1 taken 7119 times.
7183 - CatalogDatabase::kSchemaEpsilon) {
134 64 statistics_loaded = counters_.ReadFromDatabase(database(),
135 LegacyMode::kLegacy);
136
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 7119 times.
7119 } 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 7119 times.
7119 } 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 7119 times.
7119 } else if (database().schema_revision() < 5) {
143 statistics_loaded = counters_.ReadFromDatabase(database(),
144 LegacyMode::kNoSpecials);
145 } else {
146 7119 statistics_loaded = counters_.ReadFromDatabase(database());
147 }
148 7183 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 4526 bool Catalog::OpenDatabase(const string &db_path) {
158
2/4
✓ Branch 1 taken 4526 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4526 times.
✗ Branch 5 not taken.
4526 database_ = CatalogDatabase::Open(db_path, DatabaseOpenMode());
159
2/2
✓ Branch 0 taken 64 times.
✓ Branch 1 taken 4462 times.
4526 if (NULL == database_) {
160 64 return false;
161 }
162
163
2/2
✓ Branch 2 taken 64 times.
✓ Branch 3 taken 4398 times.
4462 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 64 database(),
168 "SELECT count(*) FROM sqlite_master "
169
2/4
✓ Branch 2 taken 64 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 64 times.
✗ Branch 6 not taken.
192 "WHERE type='table' AND name='nested_catalogs' AND sql LIKE '%sha1%';");
170
1/2
✓ Branch 1 taken 64 times.
✗ Branch 2 not taken.
64 const bool retval = sql_has_nested_sha1.FetchRow();
171
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 64 times.
64 assert(retval == true);
172
1/2
✓ Branch 1 taken 64 times.
✗ Branch 2 not taken.
64 const bool has_nested_sha1 = sql_has_nested_sha1.RetrieveInt64(0);
173
2/2
✓ Branch 0 taken 31 times.
✓ Branch 1 taken 33 times.
64 if (!has_nested_sha1) {
174 31 database_->EnforceSchema(0.9, 0);
175 }
176 64 }
177
178
1/2
✓ Branch 1 taken 4462 times.
✗ Branch 2 not taken.
4462 InitPreparedStatements();
179
180 // Set the database file ownership if requested
181
2/2
✓ Branch 0 taken 263 times.
✓ Branch 1 taken 4199 times.
4462 if (managed_database_) {
182
1/2
✓ Branch 1 taken 263 times.
✗ Branch 2 not taken.
263 database_->TakeFileOwnership();
183 }
184
185 // Find out the maximum row id of this database file
186
2/4
✓ Branch 2 taken 4462 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 4462 times.
✗ Branch 7 not taken.
8924 SqlCatalog sql_max_row_id(database(), "SELECT MAX(rowid) FROM catalog;");
187
2/4
✓ Branch 1 taken 4462 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 4462 times.
4462 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 4462 times.
✗ Branch 2 not taken.
4462 max_row_id_ = sql_max_row_id.RetrieveInt64(0);
195
196 // Get root prefix
197
4/6
✓ Branch 2 taken 4462 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 4462 times.
✗ Branch 6 not taken.
✓ Branch 9 taken 2116 times.
✓ Branch 10 taken 2346 times.
4462 if (database_->HasProperty("root_prefix")) {
198 2116 const std::string root_prefix = database_->GetProperty<std::string>(
199
2/4
✓ Branch 2 taken 2116 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 2116 times.
✗ Branch 6 not taken.
4232 "root_prefix");
200
1/2
✓ Branch 3 taken 2116 times.
✗ Branch 4 not taken.
2116 root_prefix_.Assign(root_prefix.data(), root_prefix.size());
201
1/2
✓ Branch 3 taken 2116 times.
✗ Branch 4 not taken.
2116 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 2116 times.
✗ Branch 2 not taken.
2116 is_regular_mountpoint_ = (root_prefix_ == mountpoint_);
205 2116 } else {
206
1/2
✓ Branch 2 taken 2346 times.
✗ Branch 3 not taken.
2346 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 4462 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4462 times.
✗ Branch 5 not taken.
4462 volatile_flag_ = database_->GetPropertyDefault<bool>("volatile",
212 4462 volatile_flag_);
213
214 // Read Catalog Counter Statistics
215
2/4
✓ Branch 1 taken 4462 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 4462 times.
4462 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 1848 times.
✓ Branch 2 taken 2614 times.
4462 if (HasParent()) {
223
1/2
✓ Branch 1 taken 1848 times.
✗ Branch 2 not taken.
1848 parent_->AddChild(this);
224 }
225
226 4462 initialized_ = true;
227 4462 return true;
228 4462 }
229
230
231 /**
232 * Removes the mountpoint and prepends the root prefix to path
233 */
234 22523 shash::Md5 Catalog::NormalizePath(const PathString &path) const {
235
2/2
✓ Branch 0 taken 22283 times.
✓ Branch 1 taken 240 times.
22523 if (is_regular_mountpoint_)
236
1/2
✓ Branch 3 taken 22283 times.
✗ Branch 4 not taken.
22283 return shash::Md5(path.GetChars(), path.GetLength());
237
238
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 240 times.
240 assert(path.GetLength() >= mountpoint_.GetLength());
239 // Piecewise hash calculation: root_prefix plus tail of path
240
1/2
✓ Branch 1 taken 240 times.
✗ Branch 2 not taken.
240 shash::Any result(shash::kMd5);
241
1/2
✓ Branch 1 taken 240 times.
✗ Branch 2 not taken.
240 shash::ContextPtr ctx(shash::kMd5);
242 240 ctx.buffer = alloca(ctx.size);
243
1/2
✓ Branch 1 taken 240 times.
✗ Branch 2 not taken.
240 shash::Init(ctx);
244
1/2
✓ Branch 2 taken 240 times.
✗ Branch 3 not taken.
480 shash::Update(
245 240 reinterpret_cast<const unsigned char *>(root_prefix_.GetChars()),
246 root_prefix_.GetLength(),
247 ctx);
248 240 shash::Update(reinterpret_cast<const unsigned char *>(path.GetChars())
249
1/2
✓ Branch 2 taken 240 times.
✗ Branch 3 not taken.
240 + mountpoint_.GetLength(),
250 240 path.GetLength() - mountpoint_.GetLength(),
251 ctx);
252
1/2
✓ Branch 1 taken 240 times.
✗ Branch 2 not taken.
240 shash::Final(ctx, &result);
253
1/2
✓ Branch 1 taken 240 times.
✗ Branch 2 not taken.
240 return result.CastToMd5();
254 }
255
256
257 /**
258 * Same as NormalizePath but returns a PathString instead of an Md5 hash.
259 */
260 2021 PathString Catalog::NormalizePath2(const PathString &path) const {
261
2/2
✓ Branch 0 taken 1889 times.
✓ Branch 1 taken 132 times.
2021 if (is_regular_mountpoint_)
262
1/2
✓ Branch 1 taken 1889 times.
✗ Branch 2 not taken.
1889 return path;
263
264
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 132 times.
132 assert(path.GetLength() >= mountpoint_.GetLength());
265
1/2
✓ Branch 1 taken 132 times.
✗ Branch 2 not taken.
132 PathString result = root_prefix_;
266
1/2
✓ Branch 2 taken 132 times.
✗ Branch 3 not taken.
132 const PathString suffix = path.Suffix(mountpoint_.GetLength());
267
1/2
✓ Branch 3 taken 132 times.
✗ Branch 4 not taken.
132 result.Append(suffix.GetChars(), suffix.GetLength());
268
1/2
✓ Branch 1 taken 132 times.
✗ Branch 2 not taken.
132 return result;
269 132 }
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 1894 PathString Catalog::PlantPath(const PathString &path) const {
278
2/2
✓ Branch 0 taken 1762 times.
✓ Branch 1 taken 132 times.
1894 if (is_regular_mountpoint_)
279
1/2
✓ Branch 1 taken 1762 times.
✗ Branch 2 not taken.
1762 return path;
280
281
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 132 times.
132 assert(path.GetLength() >= root_prefix_.GetLength());
282
1/2
✓ Branch 1 taken 132 times.
✗ Branch 2 not taken.
132 PathString result = mountpoint_;
283
1/2
✓ Branch 2 taken 132 times.
✗ Branch 3 not taken.
132 const PathString suffix = path.Suffix(root_prefix_.GetLength());
284
1/2
✓ Branch 3 taken 132 times.
✗ Branch 4 not taken.
132 result.Append(suffix.GetChars(), suffix.GetLength());
285
1/2
✓ Branch 1 taken 132 times.
✗ Branch 2 not taken.
132 return result;
286 132 }
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 19808 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 19808 times.
19808 assert(IsInitialized());
299
300 19808 const MutexLockGuard m(lock_);
301
1/2
✓ Branch 1 taken 19808 times.
✗ Branch 2 not taken.
19808 sql_lookup_md5path_->BindPathHash(md5path);
302
1/2
✓ Branch 1 taken 19808 times.
✗ Branch 2 not taken.
19808 const bool found = sql_lookup_md5path_->FetchRow();
303
4/4
✓ Branch 0 taken 19242 times.
✓ Branch 1 taken 566 times.
✓ Branch 2 taken 19215 times.
✓ Branch 3 taken 27 times.
19808 if (found && (dirent != NULL)) {
304
2/4
✓ Branch 1 taken 19215 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 19215 times.
✗ Branch 5 not taken.
19215 *dirent = sql_lookup_md5path_->GetDirent(this, expand_symlink);
305
1/2
✓ Branch 1 taken 19215 times.
✗ Branch 2 not taken.
19215 FixTransitionPoint(md5path, dirent);
306 }
307
1/2
✓ Branch 1 taken 19808 times.
✗ Branch 2 not taken.
19808 sql_lookup_md5path_->Reset();
308
309 19808 return found;
310 19808 }
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 19777 bool Catalog::LookupMd5Path(const shash::Md5 &md5path,
320 DirectoryEntry *dirent) const {
321 19777 return LookupEntry(md5path, true, dirent);
322 }
323
324
325 31 bool Catalog::LookupRawSymlink(const PathString &path,
326 LinkString *raw_symlink) const {
327
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 DirectoryEntry dirent;
328
2/4
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 31 times.
✗ Branch 5 not taken.
31 const bool result = (LookupEntry(NormalizePath(path), false, &dirent));
329
1/2
✓ Branch 0 taken 31 times.
✗ Branch 1 not taken.
31 if (result)
330
2/4
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 31 times.
✗ Branch 5 not taken.
31 raw_symlink->Assign(dirent.symlink());
331 31 return result;
332 31 }
333
334
335 31 bool Catalog::LookupXattrsMd5Path(const shash::Md5 &md5path,
336 XattrList *xattrs) const {
337
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 31 times.
31 assert(IsInitialized());
338
339 31 const MutexLockGuard m(lock_);
340
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 sql_lookup_xattrs_->BindPathHash(md5path);
341
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 const bool found = sql_lookup_xattrs_->FetchRow();
342
2/4
✓ Branch 0 taken 31 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 31 times.
✗ Branch 3 not taken.
31 if (found && (xattrs != NULL)) {
343
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 *xattrs = sql_lookup_xattrs_->GetXattrs();
344 }
345
1/2
✓ Branch 1 taken 31 times.
✗ Branch 2 not taken.
31 sql_lookup_xattrs_->Reset();
346
347 31 return found;
348 31 }
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 154 bool Catalog::ListingMd5PathStat(const shash::Md5 &md5path,
358 StatEntryList *listing) const {
359
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 154 times.
154 assert(IsInitialized());
360
361
1/2
✓ Branch 1 taken 154 times.
✗ Branch 2 not taken.
154 DirectoryEntry dirent;
362
1/2
✓ Branch 1 taken 154 times.
✗ Branch 2 not taken.
154 StatEntry entry;
363
364 154 const MutexLockGuard m(lock_);
365
1/2
✓ Branch 1 taken 154 times.
✗ Branch 2 not taken.
154 sql_listing_->BindPathHash(md5path);
366
3/4
✓ Branch 1 taken 518 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 364 times.
✓ Branch 4 taken 154 times.
518 while (sql_listing_->FetchRow()) {
367
2/4
✓ Branch 1 taken 364 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 364 times.
✗ Branch 5 not taken.
364 dirent = sql_listing_->GetDirent(this);
368
2/2
✓ Branch 1 taken 33 times.
✓ Branch 2 taken 331 times.
364 if (dirent.IsHidden())
369 33 continue;
370
1/2
✓ Branch 1 taken 331 times.
✗ Branch 2 not taken.
331 FixTransitionPoint(md5path, &dirent);
371
2/4
✓ Branch 1 taken 331 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 331 times.
✗ Branch 5 not taken.
331 entry.name = dirent.name();
372
1/2
✓ Branch 1 taken 331 times.
✗ Branch 2 not taken.
331 entry.info = dirent.GetStatStructure();
373
1/2
✓ Branch 1 taken 331 times.
✗ Branch 2 not taken.
331 listing->PushBack(entry);
374 }
375
1/2
✓ Branch 1 taken 154 times.
✗ Branch 2 not taken.
154 sql_listing_->Reset();
376
377 154 return true;
378 154 }
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 3201 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 3201 times.
3201 assert(IsInitialized());
393
394 3201 const MutexLockGuard m(lock_);
395
396
1/2
✓ Branch 1 taken 3201 times.
✗ Branch 2 not taken.
3201 sql_listing_->BindPathHash(md5path);
397
3/4
✓ Branch 1 taken 8894 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 5693 times.
✓ Branch 4 taken 3201 times.
8894 while (sql_listing_->FetchRow()) {
398
1/2
✓ Branch 1 taken 5693 times.
✗ Branch 2 not taken.
5693 DirectoryEntry dirent = sql_listing_->GetDirent(this, expand_symlink);
399
1/2
✓ Branch 1 taken 5693 times.
✗ Branch 2 not taken.
5693 FixTransitionPoint(md5path, &dirent);
400
1/2
✓ Branch 1 taken 5693 times.
✗ Branch 2 not taken.
5693 listing->push_back(dirent);
401 5693 }
402
1/2
✓ Branch 1 taken 3201 times.
✗ Branch 2 not taken.
3201 sql_listing_->Reset();
403
404 3201 return true;
405 3201 }
406
407
408 62 bool Catalog::AllChunksBegin() { return sql_all_chunks_->Open(); }
409
410
411 310 bool Catalog::AllChunksNext(shash::Any *hash,
412 zlib::Algorithms *compression_alg) {
413 310 return sql_all_chunks_->Next(hash, compression_alg);
414 }
415
416
417 62 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 33 bool Catalog::ListMd5PathChunks(const shash::Md5 &md5path,
426 const shash::Algorithms interpret_hashes_as,
427 FileChunkList *chunks) const {
428
2/4
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
33 assert(IsInitialized() && chunks->IsEmpty());
429
430 33 const MutexLockGuard m(lock_);
431
432
1/2
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
33 sql_chunks_listing_->BindPathHash(md5path);
433
3/4
✓ Branch 1 taken 66 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 33 times.
✓ Branch 4 taken 33 times.
66 while (sql_chunks_listing_->FetchRow()) {
434
2/4
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
33 chunks->PushBack(sql_chunks_listing_->GetFileChunk(interpret_hashes_as));
435 }
436
1/2
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
33 sql_chunks_listing_->Reset();
437
438 33 return true;
439 33 }
440
441
442 /**
443 * Only used by the garbage collection
444 */
445 62 const Catalog::HashVector &Catalog::GetReferencedObjects() const {
446
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 62 times.
62 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 62 times.
✗ Branch 3 not taken.
62 SqlListContentHashes list_content_hashes(database());
452
3/4
✓ Branch 1 taken 465 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 403 times.
✓ Branch 4 taken 62 times.
465 while (list_content_hashes.FetchRow()) {
453
2/4
✓ Branch 1 taken 403 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 403 times.
✗ Branch 5 not taken.
403 referenced_hashes_.push_back(list_content_hashes.GetHash());
454 }
455
456 62 return referenced_hashes_;
457 62 }
458
459
460 3098 void Catalog::TakeDatabaseFileOwnership() {
461 3098 managed_database_ = true;
462
2/2
✓ Branch 0 taken 2835 times.
✓ Branch 1 taken 263 times.
3098 if (NULL != database_) {
463 2835 database_->TakeFileOwnership();
464 }
465 3098 }
466
467
468 31 void Catalog::DropDatabaseFileOwnership() {
469 31 managed_database_ = false;
470
1/2
✓ Branch 0 taken 31 times.
✗ Branch 1 not taken.
31 if (NULL != database_) {
471 31 database_->DropFileOwnership();
472 }
473 31 }
474
475
476 776 uint64_t Catalog::GetTTL() const {
477 776 const MutexLockGuard m(lock_);
478
2/4
✓ Branch 3 taken 776 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 776 times.
✗ Branch 7 not taken.
1552 return database().GetPropertyDefault<uint64_t>("TTL", kDefaultTTL);
479 776 }
480
481
482 bool Catalog::HasExplicitTTL() const {
483 const MutexLockGuard m(lock_);
484 return database().HasProperty("TTL");
485 }
486
487
488 1589 bool Catalog::GetVOMSAuthz(string *authz) const {
489 bool result;
490 1589 const MutexLockGuard m(lock_);
491
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1589 times.
1589 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 1589 times.
1589 } else if (voms_authz_status_ == kVomsNone) {
497 result = false;
498 } else {
499
3/6
✓ Branch 3 taken 1589 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 1589 times.
✗ Branch 7 not taken.
✗ Branch 10 not taken.
✓ Branch 11 taken 1589 times.
1589 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 1589 voms_authz_status_ = kVomsNone;
507 }
508 1589 result = (voms_authz_status_ == kVomsPresent);
509 }
510 1589 return result;
511 1589 }
512
513 4177 uint64_t Catalog::GetRevision() const {
514 4177 const MutexLockGuard m(lock_);
515
2/4
✓ Branch 3 taken 4177 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 4177 times.
✗ Branch 7 not taken.
8354 return database().GetPropertyDefault<uint64_t>("revision", 0);
516 4177 }
517
518 1653 uint64_t Catalog::GetLastModified() const {
519
1/2
✓ Branch 2 taken 1653 times.
✗ Branch 3 not taken.
1653 const std::string prop_name = "last_modified";
520
1/2
✓ Branch 2 taken 1653 times.
✗ Branch 3 not taken.
1653 return (database().HasProperty(prop_name))
521
2/4
✓ Branch 0 taken 1653 times.
✗ Branch 1 not taken.
✓ Branch 4 taken 1653 times.
✗ Branch 5 not taken.
1653 ? database().GetProperty<int>(prop_name)
522 1653 : 0u;
523 1653 }
524
525
526 uint64_t Catalog::GetNumChunks() const {
527 return counters_.Get("self_regular") + counters_.Get("self_chunks");
528 }
529
530
531 62 uint64_t Catalog::GetNumEntries() const {
532
1/2
✓ Branch 2 taken 62 times.
✗ Branch 3 not taken.
62 const string sql = "SELECT count(*) FROM catalog;";
533
534 62 const MutexLockGuard m(lock_);
535
1/2
✓ Branch 2 taken 62 times.
✗ Branch 3 not taken.
62 SqlCatalog stmt(database(), sql);
536
3/6
✓ Branch 1 taken 62 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 62 times.
✗ Branch 4 not taken.
✓ Branch 6 taken 62 times.
✗ Branch 7 not taken.
124 return (stmt.FetchRow()) ? stmt.RetrieveInt64(0) : 0;
537 62 }
538
539
540 62 shash::Any Catalog::GetPreviousRevision() const {
541 62 const MutexLockGuard m(lock_);
542 62 const std::string hash_string = database().GetPropertyDefault<std::string>(
543
3/6
✓ Branch 2 taken 62 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 62 times.
✗ Branch 7 not taken.
✓ Branch 9 taken 62 times.
✗ Branch 10 not taken.
124 "previous_revision", "");
544 62 return (!hash_string.empty())
545
1/2
✓ Branch 2 taken 31 times.
✗ Branch 3 not taken.
31 ? shash::MkFromHexPtr(shash::HexPtr(hash_string),
546 shash::kSuffixCatalog)
547
3/4
✓ Branch 0 taken 31 times.
✓ Branch 1 taken 31 times.
✓ Branch 3 taken 31 times.
✗ Branch 4 not taken.
124 : shash::Any();
548 62 }
549
550
551 33 string Catalog::PrintMemStatistics() const {
552 33 sqlite::MemStatistics stats;
553 {
554 33 const MutexLockGuard m(lock_);
555
1/2
✓ Branch 2 taken 33 times.
✗ Branch 3 not taken.
33 database().GetMemStatistics(&stats);
556 33 }
557
4/8
✓ Branch 2 taken 33 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 33 times.
✗ Branch 7 not taken.
✓ Branch 10 taken 33 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 33 times.
✗ Branch 14 not taken.
66 return string(mountpoint().GetChars(), mountpoint().GetLength()) + ": "
558
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.lookaside_slots_used) + " / "
559
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.lookaside_slots_max) + " slots -- "
560
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.lookaside_hit) + " hits, "
561
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.lookaside_miss_size) + " misses-size, "
562
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.lookaside_miss_full) + " misses-full -- "
563
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.page_cache_used / 1024) + " kB pages -- "
564
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.page_cache_hit) + " hits, "
565
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.page_cache_miss) + " misses -- "
566
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + StringifyInt(stats.schema_used / 1024) + " kB schema -- "
567
3/6
✓ Branch 1 taken 33 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 33 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 33 times.
✗ Branch 8 not taken.
132 + 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 25272 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 25272 times.
25272 assert(IsInitialized());
582
583
2/2
✓ Branch 1 taken 656 times.
✓ Branch 2 taken 24616 times.
25272 if (inode_range_.IsDummy()) {
584 656 return DirectoryEntry::kInvalidInode;
585 }
586
587 24616 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 24616 times.
24616 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 24616 times.
24616 if (inode_annotation_) {
604 inode = inode_annotation_->Annotate(inode);
605 }
606
607 24616 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 13990 const Catalog::NestedCatalogList &Catalog::ListNestedCatalogs() const {
617 13990 const MutexLockGuard m(lock_);
618
619
2/2
✓ Branch 0 taken 3151 times.
✓ Branch 1 taken 10839 times.
13990 if (nested_catalog_cache_dirty_) {
620
1/2
✓ Branch 2 taken 3151 times.
✗ Branch 3 not taken.
3151 LogCvmfs(kLogCatalog, kLogDebug, "refreshing nested catalog cache of '%s'",
621
1/2
✓ Branch 1 taken 3151 times.
✗ Branch 2 not taken.
6302 mountpoint().c_str());
622
3/4
✓ Branch 1 taken 4706 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1555 times.
✓ Branch 4 taken 3151 times.
4706 while (sql_list_nested_->FetchRow()) {
623
1/2
✓ Branch 1 taken 1555 times.
✗ Branch 2 not taken.
1555 NestedCatalog nested;
624
3/6
✓ Branch 1 taken 1555 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1555 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 1555 times.
✗ Branch 8 not taken.
1555 nested.mountpoint = PlantPath(sql_list_nested_->GetPath());
625
1/2
✓ Branch 1 taken 1555 times.
✗ Branch 2 not taken.
1555 nested.hash = sql_list_nested_->GetContentHash();
626
1/2
✓ Branch 1 taken 1555 times.
✗ Branch 2 not taken.
1555 nested.size = sql_list_nested_->GetSize();
627
1/2
✓ Branch 1 taken 1555 times.
✗ Branch 2 not taken.
1555 nested_catalog_cache_.push_back(nested);
628 1555 }
629
1/2
✓ Branch 1 taken 3151 times.
✗ Branch 2 not taken.
3151 sql_list_nested_->Reset();
630 3151 nested_catalog_cache_dirty_ = false;
631 }
632
633 13990 return nested_catalog_cache_;
634 13990 }
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 1244 const Catalog::NestedCatalogList Catalog::ListOwnNestedCatalogs() const {
643 1244 NestedCatalogList result;
644
645 1244 const MutexLockGuard m(lock_);
646
647
3/4
✓ Branch 1 taken 1385 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 141 times.
✓ Branch 4 taken 1244 times.
1385 while (sql_own_list_nested_->FetchRow()) {
648
1/2
✓ Branch 1 taken 141 times.
✗ Branch 2 not taken.
141 NestedCatalog nested;
649
3/6
✓ Branch 1 taken 141 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 141 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 141 times.
✗ Branch 8 not taken.
141 nested.mountpoint = PlantPath(sql_own_list_nested_->GetPath());
650
1/2
✓ Branch 1 taken 141 times.
✗ Branch 2 not taken.
141 nested.hash = sql_own_list_nested_->GetContentHash();
651
1/2
✓ Branch 1 taken 141 times.
✗ Branch 2 not taken.
141 nested.size = sql_own_list_nested_->GetSize();
652
1/2
✓ Branch 1 taken 141 times.
✗ Branch 2 not taken.
141 result.push_back(nested);
653 141 }
654
1/2
✓ Branch 1 taken 1244 times.
✗ Branch 2 not taken.
1244 sql_own_list_nested_->Reset();
655
656 2488 return result;
657 1244 }
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 2587 void Catalog::ResetNestedCatalogCacheUnprotected() {
668 2587 nested_catalog_cache_.clear();
669 2587 nested_catalog_cache_dirty_ = true;
670 2587 }
671
672
673 /**
674 * Looks for a specific registered nested catalog based on a path.
675 */
676 1823 bool Catalog::FindNested(const PathString &mountpoint, shash::Any *hash,
677 uint64_t *size) const {
678 1823 const MutexLockGuard m(lock_);
679
1/2
✓ Branch 1 taken 1823 times.
✗ Branch 2 not taken.
1823 const PathString normalized_mountpoint = NormalizePath2(mountpoint);
680
1/2
✓ Branch 1 taken 1823 times.
✗ Branch 2 not taken.
1823 sql_lookup_nested_->BindSearchPath(normalized_mountpoint);
681
1/2
✓ Branch 1 taken 1823 times.
✗ Branch 2 not taken.
1823 const bool found = sql_lookup_nested_->FetchRow();
682
3/4
✓ Branch 0 taken 1766 times.
✓ Branch 1 taken 57 times.
✓ Branch 2 taken 1766 times.
✗ Branch 3 not taken.
1823 if (found && (hash != NULL)) {
683
1/2
✓ Branch 1 taken 1766 times.
✗ Branch 2 not taken.
1766 *hash = sql_lookup_nested_->GetContentHash();
684
1/2
✓ Branch 1 taken 1766 times.
✗ Branch 2 not taken.
1766 *size = sql_lookup_nested_->GetSize();
685 }
686
1/2
✓ Branch 1 taken 1823 times.
✗ Branch 2 not taken.
1823 sql_lookup_nested_->Reset();
687
688 1823 return found;
689 1823 }
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 3082 void Catalog::SetInodeAnnotation(InodeAnnotation *new_annotation) {
697 3082 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 3082 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
3082 assert((inode_annotation_ == NULL) || (inode_annotation_ == new_annotation));
701 3082 inode_annotation_ = new_annotation;
702 3082 }
703
704
705 3082 void Catalog::SetOwnerMaps(const OwnerMap *uid_map, const OwnerMap *gid_map) {
706
2/4
✓ Branch 0 taken 3082 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 3082 times.
3082 uid_map_ = (uid_map && uid_map->HasEffect()) ? uid_map : NULL;
707
2/4
✓ Branch 0 taken 3082 times.
✗ Branch 1 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 3082 times.
3082 gid_map_ = (gid_map && gid_map->HasEffect()) ? gid_map : NULL;
708 3082 }
709
710
711 /**
712 * Add a Catalog as child to this Catalog.
713 * @param child the Catalog to define as child
714 */
715 1848 void Catalog::AddChild(Catalog *child) {
716
3/7
✓ Branch 1 taken 1848 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 1848 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 1848 times.
1848 assert(NULL == FindChild(child->mountpoint()));
717
718 1848 const MutexLockGuard m(lock_);
719
2/4
✓ Branch 1 taken 1848 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1848 times.
✗ Branch 5 not taken.
1848 children_[child->mountpoint()] = child;
720 1848 child->set_parent(this);
721 1848 }
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 1491 void Catalog::RemoveChild(Catalog *child) {
729
3/7
✓ Branch 1 taken 1491 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 1491 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✓ Branch 7 taken 1491 times.
1491 assert(NULL != FindChild(child->mountpoint()));
730
731 1491 const MutexLockGuard m(lock_);
732 1491 child->set_parent(NULL);
733
2/4
✓ Branch 1 taken 1491 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 1491 times.
✗ Branch 5 not taken.
1491 children_.erase(child->mountpoint());
734 1491 }
735
736
737 4832 CatalogList Catalog::GetChildren() const {
738 4832 CatalogList result;
739
740 4832 const MutexLockGuard m(lock_);
741 9664 for (NestedCatalogMap::const_iterator i = children_.begin(),
742 4832 iEnd = children_.end();
743
2/2
✓ Branch 1 taken 2341 times.
✓ Branch 2 taken 4832 times.
7173 i != iEnd;
744 2341 ++i) {
745
1/2
✓ Branch 2 taken 2341 times.
✗ Branch 3 not taken.
2341 result.push_back(i->second);
746 }
747
748 9664 return result;
749 4832 }
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 14951 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 14951 times.
14951 if (!path.StartsWith(mountpoint_))
762 return NULL;
763
764
1/2
✓ Branch 2 taken 14951 times.
✗ Branch 3 not taken.
14951 PathString remaining(path.Suffix(mountpoint_.GetLength()));
765
1/2
✓ Branch 1 taken 14951 times.
✗ Branch 2 not taken.
14951 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 14951 times.
✗ Branch 2 not taken.
14951 PathString path_prefix(mountpoint_);
770 14951 Catalog *result = NULL;
771 // Skip the first '/'
772
1/2
✓ Branch 1 taken 14951 times.
✗ Branch 2 not taken.
14951 path_prefix.Append("/", 1);
773 14951 const char *c = remaining.GetChars() + 1;
774
2/2
✓ Branch 1 taken 134058 times.
✓ Branch 2 taken 12306 times.
146364 for (unsigned i = 1; i < remaining.GetLength(); ++i, ++c) {
775
2/2
✓ Branch 0 taken 30338 times.
✓ Branch 1 taken 103720 times.
134058 if (*c == '/') {
776
1/2
✓ Branch 1 taken 30338 times.
✗ Branch 2 not taken.
30338 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 2645 times.
✓ Branch 1 taken 27693 times.
30338 if (result != NULL)
781 2645 break;
782 }
783
1/2
✓ Branch 1 taken 131413 times.
✗ Branch 2 not taken.
131413 path_prefix.Append(c, 1);
784 }
785
786 14951 return result;
787 14951 }
788
789
790 /**
791 * Looks for a child catalog, which is a subset of all registered nested
792 * catalogs.
793 */
794 34016 Catalog *Catalog::FindChild(const PathString &mountpoint) const {
795 34016 NestedCatalogMap::const_iterator nested_iter;
796
797 34016 const MutexLockGuard m(lock_);
798
1/2
✓ Branch 1 taken 34016 times.
✗ Branch 2 not taken.
34016 nested_iter = children_.find(mountpoint);
799
2/2
✓ Branch 2 taken 29755 times.
✓ Branch 3 taken 4261 times.
38277 Catalog *result = (nested_iter == children_.end()) ? NULL
800 4261 : nested_iter->second;
801
802 34016 return result;
803 34016 }
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 25239 void Catalog::FixTransitionPoint(const shash::Md5 &md5path,
814 DirectoryEntry *dirent) const {
815
2/2
✓ Branch 1 taken 19337 times.
✓ Branch 2 taken 5902 times.
25239 if (!HasParent())
816 19337 return;
817
818
2/2
✓ Branch 1 taken 775 times.
✓ Branch 2 taken 5127 times.
5902 if (dirent->IsNestedCatalogRoot()) {
819 // Normal nested catalog
820
1/2
✓ Branch 1 taken 775 times.
✗ Branch 2 not taken.
775 DirectoryEntry parent_dirent;
821
1/2
✓ Branch 1 taken 775 times.
✗ Branch 2 not taken.
775 const bool retval = parent_->LookupMd5Path(md5path, &parent_dirent);
822
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 775 times.
775 assert(retval);
823 775 dirent->set_inode(parent_dirent.inode());
824
1/2
✗ Branch 2 not taken.
✓ Branch 3 taken 5127 times.
5902 } 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