From 5f95cb224fbf1e6b779363e311ab4a3a8ff404c0 Mon Sep 17 00:00:00 2001 From: PengZheng Date: Tue, 1 Aug 2023 19:18:06 +0800 Subject: [PATCH 1/7] Avoid unnecessary cache directory traversal when start with a clean cache. --- libs/framework/src/celix_bundle_cache.c | 17 +++++++---------- 1 file changed, 7 insertions(+), 10 deletions(-) diff --git a/libs/framework/src/celix_bundle_cache.c b/libs/framework/src/celix_bundle_cache.c index 95c5d0349..64fb332fb 100644 --- a/libs/framework/src/celix_bundle_cache.c +++ b/libs/framework/src/celix_bundle_cache.c @@ -53,6 +53,7 @@ struct celix_bundle_cache { celix_thread_mutex_t mutex; //protects below and access to the cache dir celix_string_hash_map_t* locationToBundleIdLookupMap; //key = location, value = bundle id. + bool locationToBundleIdLookupMapLoaded; //true if the locationToBundleIdLookupMap is loaded from disk }; static const char* bundleCache_progamName() { @@ -163,6 +164,7 @@ celix_status_t celix_bundleCache_create(celix_framework_t* fw, celix_bundle_cach cache->cacheDir, errorStr); return status; } + cache->locationToBundleIdLookupMapLoaded = false; celix_steal_ptr(cacheDir); celix_steal_ptr(mutex); celix_steal_ptr(locationToBundleIdLookupMap); @@ -233,8 +235,8 @@ celix_status_t celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bun const char* loc = NULL; celixThreadMutex_lock(&cache->mutex); (void) bundleArchive_getLocation(archive, &loc); - (void) celix_stringHashMap_remove(cache->locationToBundleIdLookupMap, loc); if (permanent) { + (void) celix_stringHashMap_remove(cache->locationToBundleIdLookupMap, loc); status = bundleArchive_closeAndDelete(archive); } celixThreadMutex_unlock(&cache->mutex); @@ -246,12 +248,10 @@ celix_status_t celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bun * Update location->bundle id lookup map. */ static void celix_bundleCache_updateIdForLocationLookupMap(celix_bundle_cache_t* cache) { - celixThreadMutex_lock(&cache->mutex); DIR* dir = opendir(cache->cacheDir); if (dir == NULL) { fw_logCode(cache->fw->logger, CELIX_LOG_LEVEL_ERROR, CELIX_BUNDLE_EXCEPTION, "Cannot open bundle cache directory %s", cache->cacheDir); - celixThreadMutex_unlock(&cache->mutex); return; } char archiveRootBuffer[CELIX_DEFAULT_STRING_CREATE_BUFFER_SIZE]; @@ -276,22 +276,19 @@ static void celix_bundleCache_updateIdForLocationLookupMap(celix_bundle_cache_t* } } closedir(dir); - celixThreadMutex_unlock(&cache->mutex); } long celix_bundleCache_findBundleIdForLocation(celix_bundle_cache_t* cache, const char* location) { long bndId = -1; celixThreadMutex_lock(&cache->mutex); + if (!cache->locationToBundleIdLookupMapLoaded) { + celix_bundleCache_updateIdForLocationLookupMap(cache); + cache->locationToBundleIdLookupMapLoaded = true; + } if (celix_stringHashMap_hasKey(cache->locationToBundleIdLookupMap, location)) { bndId = celix_stringHashMap_getLong(cache->locationToBundleIdLookupMap, location, -1); } celixThreadMutex_unlock(&cache->mutex); - if (bndId == -1) { - celix_bundleCache_updateIdForLocationLookupMap(cache); - celixThreadMutex_lock(&cache->mutex); - bndId = celix_stringHashMap_getLong(cache->locationToBundleIdLookupMap, location, -1); - celixThreadMutex_unlock(&cache->mutex); - } return bndId; } From 8f08933285bff807b927d2872da9913a701c4878 Mon Sep 17 00:00:00 2001 From: PengZheng Date: Wed, 2 Aug 2023 17:25:02 +0800 Subject: [PATCH 2/7] Code cleanup for bundle_archive.c --- ...undleArchiveWithErrorInjectionTestSuite.cc | 6 +- libs/framework/src/bundle_archive.c | 66 +++++++++---------- libs/framework/src/bundle_archive_private.h | 2 +- libs/framework/src/celix_bundle_cache.c | 2 +- 4 files changed, 37 insertions(+), 39 deletions(-) diff --git a/libs/framework/gtest/src/BundleArchiveWithErrorInjectionTestSuite.cc b/libs/framework/gtest/src/BundleArchiveWithErrorInjectionTestSuite.cc index e4d5d422b..03100ddff 100644 --- a/libs/framework/gtest/src/BundleArchiveWithErrorInjectionTestSuite.cc +++ b/libs/framework/gtest/src/BundleArchiveWithErrorInjectionTestSuite.cc @@ -198,7 +198,7 @@ TEST_F(CelixBundleArchiveErrorInjectionTestSuite, ArchiveCreateErrorTest) { EXPECT_FALSE(celix_utils_directoryExists(TEST_ARCHIVE_ROOT)); teardownErrorInjectors(); - celix_ei_expect_lstat((void*) celix_bundleArchive_create, 2, -1); + celix_ei_expect_lstat((void*) celix_bundleArchive_create, 3, -1); EXPECT_EQ(CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,EACCES), celix_bundleArchive_create(&fw, TEST_ARCHIVE_ROOT, 1, SIMPLE_TEST_BUNDLE1_LOCATION, &archive)); EXPECT_EQ(nullptr, archive); @@ -213,7 +213,7 @@ TEST_F(CelixBundleArchiveErrorInjectionTestSuite, ArchiveCreateErrorTest) { archive = nullptr; std::this_thread::sleep_for(std::chrono::milliseconds(10)); celix_utils_touch(SIMPLE_TEST_BUNDLE1_LOCATION); - celix_ei_expect_unlink((void*)celix_bundleArchive_create, 2, -1); + celix_ei_expect_unlink((void*)celix_bundleArchive_create, 3, -1); EXPECT_EQ(CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,EACCES), celix_bundleArchive_create(&fw, TEST_ARCHIVE_ROOT, 2, SIMPLE_TEST_BUNDLE1_LOCATION, &archive)); EXPECT_EQ(nullptr, archive); @@ -226,7 +226,7 @@ TEST_F(CelixBundleArchiveErrorInjectionTestSuite, ArchiveCreateErrorTest) { archive = nullptr; std::this_thread::sleep_for(std::chrono::milliseconds(10)); celix_utils_touch(SIMPLE_TEST_BUNDLE1_LOCATION); - celix_ei_expect_celix_utils_deleteDirectory((void*)celix_bundleArchive_create, 2, CELIX_FILE_IO_EXCEPTION); + celix_ei_expect_celix_utils_deleteDirectory((void*)celix_bundleArchive_create, 3, CELIX_FILE_IO_EXCEPTION); EXPECT_EQ(CELIX_FILE_IO_EXCEPTION, celix_bundleArchive_create(&fw, TEST_ARCHIVE_ROOT, 1, SIMPLE_TEST_BUNDLE1_LOCATION, &archive)); EXPECT_EQ(nullptr, archive); diff --git a/libs/framework/src/bundle_archive.c b/libs/framework/src/bundle_archive.c index 8d55e77e0..37f5ed2ca 100644 --- a/libs/framework/src/bundle_archive.c +++ b/libs/framework/src/bundle_archive.c @@ -97,6 +97,35 @@ static celix_status_t celix_bundleArchive_storeBundleStateProperties(bundle_arch return CELIX_SUCCESS; } +static celix_status_t celix_bundleArchive_removeResourceCache(bundle_archive_t* archive) { + celix_status_t status = CELIX_SUCCESS; + const char* error = NULL; + struct stat st; + status = lstat(archive->resourceCacheRoot, &st); + if (status == -1 && errno == ENOENT) { + return CELIX_SUCCESS; + } else if(status == -1 && errno != ENOENT) { + status = CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,errno); + fw_logCode(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, status, "Failed to stat bundle archive directory '%s'", archive->resourceCacheRoot); + return status; + } + // assert(status == 0); + // celix_utils_deleteDirectory does not work for dangling symlinks, so handle this case separately + if (S_ISLNK(st.st_mode)) { + status = unlink(archive->resourceCacheRoot); + if (status == -1) { + status = CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,errno); + error = "Failed to remove existing bundle symlink"; + } + } else { + status = celix_utils_deleteDirectory(archive->resourceCacheRoot, &error); + } + if (status != CELIX_SUCCESS) { + fw_logCode(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, status, "Failed to remove existing bundle archive revision directory '%s': %s", archive->resourceCacheRoot, error); + } + return status; +} + static celix_status_t celix_bundleArchive_extractBundle(bundle_archive_t* archive, const char* bundleUrl) { celix_status_t status = CELIX_SUCCESS; @@ -126,30 +155,10 @@ celix_bundleArchive_extractBundle(bundle_archive_t* archive, const char* bundleU * If dlopen/dlsym is used with newer files, but with the same inode already used in dlopen/dlsym this leads to * segfaults. */ - const char* error = NULL; - struct stat st; - status = lstat(archive->resourceCacheRoot, &st); - if(status == -1 && errno != ENOENT) { - status = CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,errno); - fw_logCode(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, status, "Failed to stat bundle archive directory '%s'", archive->resourceCacheRoot); + status = celix_bundleArchive_removeResourceCache(archive); + if (status != CELIX_SUCCESS) { return status; } - if (status == 0) { - // celix_utils_deleteDirectory does not work for dangling symlinks, so handle this case separately - if (S_ISLNK(st.st_mode)) { - status = unlink(archive->resourceCacheRoot); - if (status == -1) { - status = CELIX_ERROR_MAKE(CELIX_FACILITY_CERRNO,errno); - error = "Failed to remove existing bundle symlink"; - } - } else { - status = celix_utils_deleteDirectory(archive->resourceCacheRoot, &error); - } - if (status != CELIX_SUCCESS) { - fw_logCode(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, status, "Failed to remove existing bundle archive revision directory '%s': %s", archive->resourceCacheRoot, error); - return status; - } - } status = celix_framework_utils_extractBundle(archive->fw, bundleUrl, archive->resourceCacheRoot); if (status != CELIX_SUCCESS) { fw_log(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, "Failed to initialize archive. Failed to extract bundle zip to revision directory."); @@ -235,7 +244,6 @@ celix_status_t celix_bundleArchive_create(celix_framework_t* fw, const char *arc archive->fw = fw; archive->id = id; - celixThreadMutex_create(&archive->lock, NULL); if (isSystemBundle) { archive->resourceCacheRoot = getcwd(NULL, 0); @@ -312,7 +320,7 @@ celix_status_t celix_bundleArchive_create(celix_framework_t* fw, const char *arc return status; } -celix_status_t bundleArchive_destroy(bundle_archive_pt archive) { +void bundleArchive_destroy(bundle_archive_pt archive) { if (archive != NULL) { free(archive->location); free(archive->savedBundleStatePropertiesPath); @@ -322,10 +330,8 @@ celix_status_t bundleArchive_destroy(bundle_archive_pt archive) { free(archive->bundleSymbolicName); free(archive->bundleVersion); bundleRevision_destroy(archive->revision); - celixThreadMutex_destroy(&archive->lock); free(archive); } - return CELIX_SUCCESS; } celix_status_t bundleArchive_getId(bundle_archive_pt archive, long* id) { @@ -342,19 +348,15 @@ const char* celix_bundleArchive_getSymbolicName(bundle_archive_pt archive) { } celix_status_t bundleArchive_getLocation(bundle_archive_pt archive, const char **location) { - celixThreadMutex_lock(&archive->lock); *location = archive->location; - celixThreadMutex_unlock(&archive->lock); return CELIX_SUCCESS; } char* celix_bundleArchive_getLocation(bundle_archive_pt archive) { char* result = NULL; - celixThreadMutex_lock(&archive->lock); if (archive->location) { result = celix_utils_strdup(archive->location); } - celixThreadMutex_unlock(&archive->lock); return result; } @@ -371,9 +373,7 @@ celix_status_t bundleArchive_getCurrentRevisionNumber(bundle_archive_pt archive, //LCOV_EXCL_STOP celix_status_t bundleArchive_getCurrentRevision(bundle_archive_pt archive, bundle_revision_pt* revision) { - celixThreadMutex_lock(&archive->lock); *revision = archive->revision; - celixThreadMutex_unlock(&archive->lock); return CELIX_SUCCESS; } @@ -416,9 +416,7 @@ celix_status_t bundleArchive_getLastModified(bundle_archive_pt archive, time_t* celix_status_t celix_bundleArchive_getLastModified(bundle_archive_pt archive, struct timespec* lastModified) { celix_status_t status; - celixThreadMutex_lock(&archive->lock); status = celix_utils_getLastModified(archive->resourceCacheRoot, lastModified); - celixThreadMutex_unlock(&archive->lock); return status; } diff --git a/libs/framework/src/bundle_archive_private.h b/libs/framework/src/bundle_archive_private.h index b6a837d9a..d777ea50c 100644 --- a/libs/framework/src/bundle_archive_private.h +++ b/libs/framework/src/bundle_archive_private.h @@ -49,7 +49,7 @@ extern "C" { */ celix_status_t celix_bundleArchive_create(celix_framework_t* fw, const char *archiveRoot, long id, const char *location, bundle_archive_pt *bundle_archive); -celix_status_t bundleArchive_destroy(bundle_archive_pt archive); +void bundleArchive_destroy(bundle_archive_pt archive); /** * @brief Returns the bundle id of the bundle archive. diff --git a/libs/framework/src/celix_bundle_cache.c b/libs/framework/src/celix_bundle_cache.c index 64fb332fb..059c26a49 100644 --- a/libs/framework/src/celix_bundle_cache.c +++ b/libs/framework/src/celix_bundle_cache.c @@ -234,8 +234,8 @@ celix_status_t celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bun celix_status_t status = CELIX_SUCCESS; const char* loc = NULL; celixThreadMutex_lock(&cache->mutex); - (void) bundleArchive_getLocation(archive, &loc); if (permanent) { + (void) bundleArchive_getLocation(archive, &loc); (void) celix_stringHashMap_remove(cache->locationToBundleIdLookupMap, loc); status = bundleArchive_closeAndDelete(archive); } From 523a31a47bf63fd8f3baa9044a8971e1d7b194e9 Mon Sep 17 00:00:00 2001 From: PengZheng Date: Wed, 2 Aug 2023 18:17:13 +0800 Subject: [PATCH 3/7] Force bundle cache update when using a different bundle location. --- ...CelixBundleCacheErrorInjectionTestSuite.cc | 12 ------- .../gtest/src/CelixBundleCacheTestSuite.cc | 8 +++-- libs/framework/src/bundle_archive.c | 33 +++++++++++++++++-- libs/framework/src/bundle_archive_private.h | 5 +++ libs/framework/src/celix_bundle_cache.c | 12 +++---- libs/framework/src/celix_bundle_cache.h | 9 ++--- libs/framework/src/framework.c | 6 +++- 7 files changed, 52 insertions(+), 33 deletions(-) diff --git a/libs/framework/gtest/src/CelixBundleCacheErrorInjectionTestSuite.cc b/libs/framework/gtest/src/CelixBundleCacheErrorInjectionTestSuite.cc index 7f0896c54..3e4d66d4c 100644 --- a/libs/framework/gtest/src/CelixBundleCacheErrorInjectionTestSuite.cc +++ b/libs/framework/gtest/src/CelixBundleCacheErrorInjectionTestSuite.cc @@ -147,18 +147,6 @@ TEST_F(CelixBundleCacheErrorInjectionTestSuite, SystemArchiveCreateErrorTest) { EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_destroy(cache)); } -TEST_F(CelixBundleCacheErrorInjectionTestSuite, ArchiveDestroyErrorTest) { - celix_bundle_cache_t* cache = nullptr; - createCache(&cache); - bundle_archive_t* archive = nullptr; - EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_createArchive(cache, 1, SIMPLE_TEST_BUNDLE1_LOCATION, &archive)); - celix_ei_expect_celix_utils_deleteDirectory((void*)celix_bundleCache_destroyArchive, 1, CELIX_FILE_IO_EXCEPTION); - std::string storeRoot = celix_bundleArchive_getPersistentStoreRoot(archive); - EXPECT_EQ(CELIX_FILE_IO_EXCEPTION, celix_bundleCache_destroyArchive(cache, archive, true)); - EXPECT_TRUE(celix_utils_directoryExists(storeRoot.c_str())); - EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_destroy(cache)); -} - TEST_F(CelixBundleCacheErrorInjectionTestSuite, CreateBundleArchivesCacheErrorTest) { celix_bundle_cache_t* cache = nullptr; celix_properties_set(fw.configurationMap, CELIX_AUTO_START_1, SIMPLE_TEST_BUNDLE1_LOCATION); diff --git a/libs/framework/gtest/src/CelixBundleCacheTestSuite.cc b/libs/framework/gtest/src/CelixBundleCacheTestSuite.cc index c37ef142d..42e397c53 100644 --- a/libs/framework/gtest/src/CelixBundleCacheTestSuite.cc +++ b/libs/framework/gtest/src/CelixBundleCacheTestSuite.cc @@ -58,7 +58,8 @@ TEST_F(CelixBundleCacheTestSuite, ArchiveCreateDestroyTest) { EXPECT_TRUE(celix_bundleCache_isBundleIdAlreadyUsed(fw.cache, 1)); std::string loc = celix_bundleArchive_getPersistentStoreRoot(archive); EXPECT_TRUE(celix_utils_directoryExists(loc.c_str())); - EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_destroyArchive(fw.cache, archive, true)); + celix_bundleArchive_invalidate(archive); + celix_bundleCache_destroyArchive(fw.cache, archive); EXPECT_EQ(-1, celix_bundleCache_findBundleIdForLocation(fw.cache, SIMPLE_TEST_BUNDLE1_LOCATION)); EXPECT_FALSE(celix_bundleCache_isBundleIdAlreadyUsed(fw.cache, 1)); EXPECT_FALSE(celix_utils_directoryExists(loc.c_str())); @@ -70,7 +71,7 @@ TEST_F(CelixBundleCacheTestSuite, NonPermanentDestroyTest) { EXPECT_NE(nullptr, archive); std::string loc = celix_bundleArchive_getPersistentStoreRoot(archive); EXPECT_TRUE(celix_utils_directoryExists(loc.c_str())); - EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_destroyArchive(fw.cache, archive, false)); + celix_bundleCache_destroyArchive(fw.cache, archive); EXPECT_EQ(1, celix_bundleCache_findBundleIdForLocation(fw.cache, SIMPLE_TEST_BUNDLE1_LOCATION)); EXPECT_TRUE(celix_bundleCache_isBundleIdAlreadyUsed(fw.cache, 1)); EXPECT_TRUE(celix_utils_directoryExists(loc.c_str())); @@ -85,7 +86,8 @@ TEST_F(CelixBundleCacheTestSuite, SystemArchiveCreateDestroyTest) { EXPECT_EQ(CELIX_SUCCESS, bundleArchive_getArchiveRoot(archive, &archiveRoot)); EXPECT_EQ(nullptr, archiveRoot); EXPECT_EQ(nullptr, celix_bundleArchive_getLocation(archive)); - EXPECT_EQ(CELIX_SUCCESS, celix_bundleCache_destroyArchive(fw.cache, archive, true)); + celix_bundleArchive_invalidate(archive); + celix_bundleCache_destroyArchive(fw.cache, archive); } TEST_F(CelixBundleCacheTestSuite, CreateBundleArchivesCacheTest) { diff --git a/libs/framework/src/bundle_archive.c b/libs/framework/src/bundle_archive.c index 37f5ed2ca..ee60ffe00 100644 --- a/libs/framework/src/bundle_archive.c +++ b/libs/framework/src/bundle_archive.c @@ -54,10 +54,10 @@ struct bundleArchive { char* resourceCacheRoot; char* bundleSymbolicName; // read from the manifest char* bundleVersion; // read from the manifest - - celix_thread_mutex_t lock; // protects below and saving of bundle state properties bundle_revision_t* revision; // the current revision char* location; + bool cacheValid; // is the cache valid (e.g. not deleted) + bool valid; // is the archive valid (e.g. not deleted) }; static celix_status_t celix_bundleArchive_storeBundleStateProperties(bundle_archive_pt archive) { @@ -304,7 +304,8 @@ celix_status_t celix_bundleArchive_create(celix_framework_t* fw, const char *arc goto store_prop_failed; } } - + archive->cacheValid = true; + archive->valid = true; *bundle_archive = archive; return CELIX_SUCCESS; store_prop_failed: @@ -466,3 +467,29 @@ const char* celix_bundleArchive_getCurrentRevisionRoot(bundle_archive_t* archive } +void celix_bundleArchive_invalidate(bundle_archive_pt archive) { + archive->valid = false; + archive->cacheValid = false; +} + +void celix_bundleArchive_invalidateCache(bundle_archive_pt archive) { + archive->cacheValid = false; +} + +bool celix_bundleArchive_isCacheValid(bundle_archive_pt archive) { + return archive->cacheValid; +} + +void celix_bundleArchive_removeInvalidDirs(bundle_archive_pt archive) { + if (archive->id == CELIX_FRAMEWORK_BUNDLE_ID) { + return; + } + if (!archive->valid) { + celix_status_t status = CELIX_SUCCESS; + const char* err = NULL; + status = celix_utils_deleteDirectory(archive->archiveRoot, &err); + framework_logIfError(archive->fw->logger, status, NULL, "Failed to remove invalid archive root '%s': %s", archive->archiveRoot, err); + } else if (!archive->cacheValid){ + (void)celix_bundleArchive_removeResourceCache(archive); + } +} diff --git a/libs/framework/src/bundle_archive_private.h b/libs/framework/src/bundle_archive_private.h index d777ea50c..1fe326fc3 100644 --- a/libs/framework/src/bundle_archive_private.h +++ b/libs/framework/src/bundle_archive_private.h @@ -75,6 +75,11 @@ const char* celix_bundleArchive_getPersistentStoreRoot(bundle_archive_t *archive */ const char* celix_bundleArchive_getCurrentRevisionRoot(bundle_archive_pt archive); +void celix_bundleArchive_invalidate(bundle_archive_pt archive); +void celix_bundleArchive_invalidateCache(bundle_archive_pt archive); +bool celix_bundleArchive_isCacheValid(bundle_archive_pt archive); +void celix_bundleArchive_removeInvalidDirs(bundle_archive_pt archive); + #ifdef __cplusplus } #endif diff --git a/libs/framework/src/celix_bundle_cache.c b/libs/framework/src/celix_bundle_cache.c index 059c26a49..f69d03da3 100644 --- a/libs/framework/src/celix_bundle_cache.c +++ b/libs/framework/src/celix_bundle_cache.c @@ -230,18 +230,16 @@ celix_status_t celix_bundleCache_createSystemArchive(celix_framework_t* fw, bund return celix_bundleCache_createArchive(fw->cache, CELIX_FRAMEWORK_BUNDLE_ID, NULL, archive); } -celix_status_t celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bundle_archive_pt archive, bool permanent) { - celix_status_t status = CELIX_SUCCESS; - const char* loc = NULL; +void celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bundle_archive_pt archive) { celixThreadMutex_lock(&cache->mutex); - if (permanent) { + if (!celix_bundleArchive_isCacheValid(archive)) { + const char* loc = NULL; (void) bundleArchive_getLocation(archive, &loc); (void) celix_stringHashMap_remove(cache->locationToBundleIdLookupMap, loc); - status = bundleArchive_closeAndDelete(archive); } + (void)celix_bundleArchive_removeInvalidDirs(archive); celixThreadMutex_unlock(&cache->mutex); - (void) bundleArchive_destroy(archive); - return status; + bundleArchive_destroy(archive); } /** diff --git a/libs/framework/src/celix_bundle_cache.h b/libs/framework/src/celix_bundle_cache.h index fb0937056..309138b8e 100644 --- a/libs/framework/src/celix_bundle_cache.h +++ b/libs/framework/src/celix_bundle_cache.h @@ -91,16 +91,11 @@ celix_status_t celix_bundleCache_createSystemArchive(celix_framework_t* fw, bund /** * @brief Destroy the archive from the cache. - * It releases all resources allocated in celix_bundleCache_createArchive and deletes the archive directory if requested. + * It releases all resources allocated in celix_bundleCache_createArchive and deletes the invalid directories if needed. * @param [in] cache The bundle cache to destroy archive from. * @param [in] archive The archive to destroy. - * @param [in] permanent Whether the archive directory should be deleted or not. - * @return Status code indication failure or success: - * - CELIX_SUCCESS when no errors are encountered. - * - CELIX_FILE_IO_EXCEPTION when root of the archive is not a directory. - * - errno when the directory cannot be deleted for other reasons, check error codes of fts_open/fts_read/remove. */ -celix_status_t celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bundle_archive_pt archive, bool permanent); +void celix_bundleCache_destroyArchive(celix_bundle_cache_t* cache, bundle_archive_pt archive); /** * @brief Deletes the entire bundle cache. diff --git a/libs/framework/src/framework.c b/libs/framework/src/framework.c index 3f4c6bf78..c5fed0723 100644 --- a/libs/framework/src/framework.c +++ b/libs/framework/src/framework.c @@ -2010,7 +2010,10 @@ static celix_status_t celix_framework_uninstallBundleEntryImpl(celix_framework_t celix_framework_waitForEmptyEventQueue(framework); //to ensure that the uninstall event is triggered and handled (void)bundle_closeModules(bnd); (void)bundle_destroy(bnd); - (void)celix_bundleCache_destroyArchive(framework->cache, archive, permanent); + if(permanent) { + celix_bundleArchive_invalidate(archive); + } + (void)celix_bundleCache_destroyArchive(framework->cache, archive); } framework_logIfError(framework->logger, status, "", "Cannot uninstall bundle"); return status; @@ -2359,6 +2362,7 @@ celix_status_t celix_framework_updateBundleEntry(celix_framework_t* framework, status = CELIX_ILLEGAL_STATE; break; } + celix_bundleArchive_invalidateCache(celix_bundle_getArchive(bndEntry->bnd)); } status = celix_framework_uninstallBundleEntryImpl(framework, bndEntry, false); if (status != CELIX_SUCCESS) { From aa79fcbaf1c550d6338086340f7e0405fbfb3edf Mon Sep 17 00:00:00 2001 From: PengZheng Date: Wed, 2 Aug 2023 18:31:56 +0800 Subject: [PATCH 4/7] Minor documentation improvement. --- libs/framework/src/bundle_archive_private.h | 15 +++++++++++++++ libs/framework/src/celix_bundle_cache.h | 2 +- 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/libs/framework/src/bundle_archive_private.h b/libs/framework/src/bundle_archive_private.h index 1fe326fc3..7a2c774a8 100644 --- a/libs/framework/src/bundle_archive_private.h +++ b/libs/framework/src/bundle_archive_private.h @@ -75,9 +75,24 @@ const char* celix_bundleArchive_getPersistentStoreRoot(bundle_archive_t *archive */ const char* celix_bundleArchive_getCurrentRevisionRoot(bundle_archive_pt archive); +/** + * @brief Invalidate the whole bundle archive. + */ void celix_bundleArchive_invalidate(bundle_archive_pt archive); + +/** + * @brief Invalidate the bundle archive's bundle cache. + */ void celix_bundleArchive_invalidateCache(bundle_archive_pt archive); + +/** + * @brief Return if the bundle cache is valid. + */ bool celix_bundleArchive_isCacheValid(bundle_archive_pt archive); + +/** + * @brief Remove all valid directories of the bundle archive. + */ void celix_bundleArchive_removeInvalidDirs(bundle_archive_pt archive); #ifdef __cplusplus diff --git a/libs/framework/src/celix_bundle_cache.h b/libs/framework/src/celix_bundle_cache.h index 309138b8e..fef8a7422 100644 --- a/libs/framework/src/celix_bundle_cache.h +++ b/libs/framework/src/celix_bundle_cache.h @@ -91,7 +91,7 @@ celix_status_t celix_bundleCache_createSystemArchive(celix_framework_t* fw, bund /** * @brief Destroy the archive from the cache. - * It releases all resources allocated in celix_bundleCache_createArchive and deletes the invalid directories if needed. + * It releases all resources allocated in celix_bundleCache_createArchive and deletes invalid directories if any. * @param [in] cache The bundle cache to destroy archive from. * @param [in] archive The archive to destroy. */ From 096db1ee4914e3cca786a7323c7cb51968586093 Mon Sep 17 00:00:00 2001 From: PengZheng Date: Wed, 2 Aug 2023 20:03:56 +0800 Subject: [PATCH 5/7] Add test case for updating bundle from a different location. --- .../src/CelixBundleContextBundlesTestSuite.cc | 21 +++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/libs/framework/gtest/src/CelixBundleContextBundlesTestSuite.cc b/libs/framework/gtest/src/CelixBundleContextBundlesTestSuite.cc index f0c1b0a44..6b673a56f 100644 --- a/libs/framework/gtest/src/CelixBundleContextBundlesTestSuite.cc +++ b/libs/framework/gtest/src/CelixBundleContextBundlesTestSuite.cc @@ -29,6 +29,9 @@ #include "bundle_context_private.h" #include "celix_api.h" +#include "celix_bundle.h" +#include "celix_bundle_context.h" +#include "celix_stdlib_cleanup.h" #include "celix_file_utils.h" class CelixBundleContextBundlesTestSuite : public ::testing::Test { @@ -313,6 +316,24 @@ TEST_F(CelixBundleContextBundlesTestSuite, UpdateBundlesTest) { ASSERT_FALSE(celix_bundleContext_updateBundle(ctx, bndId1, NULL)); } +TEST_F(CelixBundleContextBundlesTestSuite, ForceUpdateUsingBundleFromDifferentLocation) { + long bndId1 = celix_bundleContext_installBundle(ctx, TEST_BND1_LOC, true); + ASSERT_TRUE(bndId1 >= 0L); + ASSERT_TRUE(celix_bundleContext_isBundleInstalled(ctx, bndId1)); + ASSERT_TRUE(celix_bundleContext_isBundleActive(ctx, bndId1)); + celix_autofree char* name1 = celix_bundleContext_getBundleSymbolicName(ctx, bndId1); + EXPECT_TRUE(celix_bundleContext_useBundle(ctx, bndId1, nullptr, [] (void *, const bundle_t *bundle) { + // make bundle cache root newer than the bundle at location TEST_BND2_LOC + celix_autofree char* root = celix_bundle_getEntry(bundle, "/"); + celix_utils_touch(root); + })); + ASSERT_TRUE(celix_bundleContext_updateBundle(ctx, bndId1, TEST_BND2_LOC)); + ASSERT_TRUE(celix_bundleContext_isBundleInstalled(ctx, bndId1)); + celix_autofree char* name2 = celix_bundleContext_getBundleSymbolicName(ctx, bndId1); + // bundle cache contains the bundle at location TEST_BND2_LOC + ASSERT_STRNE(name1, name2); +} + TEST_F(CelixBundleContextBundlesTestSuite, UpdateCorruptUncompressedBundlesTest) { const char* testExtractDir1 = "extractBundleTestDir1"; const char* testExtractDir2 = "extractBundleTestDir2"; From 6c86749810582266f141bf8ee14f85f5bad4561c Mon Sep 17 00:00:00 2001 From: PengZheng Date: Wed, 2 Aug 2023 21:03:20 +0800 Subject: [PATCH 6/7] Remove unused deprecated APIs. --- libs/framework/src/bundle.c | 16 ++-------------- libs/framework/src/bundle_archive.c | 14 ++++++-------- 2 files changed, 8 insertions(+), 22 deletions(-) diff --git a/libs/framework/src/bundle.c b/libs/framework/src/bundle.c index ad6f73d2e..20a14e7ed 100644 --- a/libs/framework/src/bundle.c +++ b/libs/framework/src/bundle.c @@ -334,20 +334,8 @@ celix_status_t bundle_close(const_bundle_pt bundle) { } celix_status_t bundle_closeAndDelete(const_bundle_pt bundle) { - celix_status_t status; - - bundle_archive_pt archive = NULL; - - bundle_closeModules(bundle); - bundle_closeRevisions(bundle); - status = bundle_getArchive(bundle, &archive); - if (status == CELIX_SUCCESS) { - bundleArchive_closeAndDelete(archive); - } - - framework_logIfError(bundle->framework->logger, status, NULL, "Failed to close and delete bundle"); - - return status; + fw_log(bundle->framework->logger, CELIX_LOG_LEVEL_DEBUG, "Usage of bundle_closeAndDelete is deprecated and no longer needed. Called for bundle %s", bundle->symbolicName); + return CELIX_SUCCESS; } celix_status_t bundle_closeRevisions(const_bundle_pt bundle) { diff --git a/libs/framework/src/bundle_archive.c b/libs/framework/src/bundle_archive.c index ee60ffe00..c8880dae0 100644 --- a/libs/framework/src/bundle_archive.c +++ b/libs/framework/src/bundle_archive.c @@ -446,17 +446,15 @@ celix_status_t bundleArchive_close(bundle_archive_pt archive) { // not yet needed/possible return CELIX_SUCCESS; } -//LCOV_EXCL_STOP celix_status_t bundleArchive_closeAndDelete(bundle_archive_pt archive) { - celix_status_t status = CELIX_SUCCESS; - if (archive->id != CELIX_FRAMEWORK_BUNDLE_ID) { - const char* err = NULL; - status = celix_utils_deleteDirectory(archive->archiveRoot, &err); - framework_logIfError(archive->fw->logger, status, NULL, "Failed to delete archive root '%s': %s", archive->archiveRoot, err); - } - return status; + fw_log(archive->fw->logger, + CELIX_LOG_LEVEL_DEBUG, + "Usage of bundleArchive_closeAndDelete is deprecated and no longer needed. Called for bundle %s", + archive->bundleSymbolicName); + return CELIX_SUCCESS; } +//LCOV_EXCL_STOP const char* celix_bundleArchive_getPersistentStoreRoot(bundle_archive_t* archive) { return archive->storeRoot; From 9a0709b03e6e75529b08ab82d4b87e05206d19ac Mon Sep 17 00:00:00 2001 From: PengZheng Date: Thu, 3 Aug 2023 09:16:32 +0800 Subject: [PATCH 7/7] Fix a potential memory leak. --- libs/framework/src/bundle_archive.c | 2 +- libs/framework/src/celix_bundle_cache.c | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/libs/framework/src/bundle_archive.c b/libs/framework/src/bundle_archive.c index c8880dae0..572072961 100644 --- a/libs/framework/src/bundle_archive.c +++ b/libs/framework/src/bundle_archive.c @@ -109,7 +109,7 @@ static celix_status_t celix_bundleArchive_removeResourceCache(bundle_archive_t* fw_logCode(archive->fw->logger, CELIX_LOG_LEVEL_ERROR, status, "Failed to stat bundle archive directory '%s'", archive->resourceCacheRoot); return status; } - // assert(status == 0); + assert(status == 0); // celix_utils_deleteDirectory does not work for dangling symlinks, so handle this case separately if (S_ISLNK(st.st_mode)) { status = unlink(archive->resourceCacheRoot); diff --git a/libs/framework/src/celix_bundle_cache.c b/libs/framework/src/celix_bundle_cache.c index f69d03da3..b6191ab06 100644 --- a/libs/framework/src/celix_bundle_cache.c +++ b/libs/framework/src/celix_bundle_cache.c @@ -272,6 +272,7 @@ static void celix_bundleCache_updateIdForLocationLookupMap(celix_bundle_cache_t* } celix_properties_destroy(props); } + celix_utils_freeStringIfNotEqual(archiveRootBuffer, bundleStateProperties); } closedir(dir); }