diff options
author | Timo Teräs <timo.teras@iki.fi> | 2018-01-04 09:46:03 +0200 |
---|---|---|
committer | Timo Teräs <timo.teras@iki.fi> | 2018-01-04 10:56:09 +0200 |
commit | 8a28c6d0d4f81406c67e76b2bcdbe491aa734717 (patch) | |
tree | 148555889d66a69cdcd979ad77f1ba728b747f53 /src/database.c | |
parent | 2da67940d50865d206f6a79165ce7b3de5a90de3 (diff) | |
download | apk-tools-8a28c6d0d4f81406c67e76b2bcdbe491aa734717.tar.gz apk-tools-8a28c6d0d4f81406c67e76b2bcdbe491aa734717.tar.bz2 apk-tools-8a28c6d0d4f81406c67e76b2bcdbe491aa734717.tar.xz apk-tools-8a28c6d0d4f81406c67e76b2bcdbe491aa734717.zip |
enable automatic update of indexes controlled by --cache-max-age
This modifies apk cache for indexes to be automatically refreshed
periodically without explicit 'update' or '--update-cache' usage.
The default is to do if-modified-since request if the local copy
is older than 4 hours. This age can be changed with --cache-max-age.
Using --update-cache will change this age to 60 seconds to make
sure the cached copy is relatively new. The small age is in order
to try to avoid downloading indexes second time when apk-tools is
upgraded and apk re-execs after self-upgrade.
Accordingly using explicitly 'apk update' will now enforce
--force-refresh and request the very latest index by requesting
any potential http proxy to do refresh too.
Diffstat (limited to 'src/database.c')
-rw-r--r-- | src/database.c | 43 |
1 files changed, 25 insertions, 18 deletions
diff --git a/src/database.c b/src/database.c index 4ea4fb4..df06ade 100644 --- a/src/database.c +++ b/src/database.c @@ -611,10 +611,10 @@ int apk_repo_format_item(struct apk_database *db, struct apk_repository *repo, s } int apk_cache_download(struct apk_database *db, struct apk_repository *repo, - struct apk_package *pkg, int verify, + struct apk_package *pkg, int verify, int autoupdate, apk_progress_cb cb, void *cb_ctx) { - struct stat st; + struct stat st = {0}; struct apk_istream *is; struct apk_bstream *bs; struct apk_sign_ctx sctx; @@ -622,6 +622,7 @@ int apk_cache_download(struct apk_database *db, struct apk_repository *repo, char tmpcacheitem[128], *cacheitem = &tmpcacheitem[tmpprefix.len]; apk_blob_t b = APK_BLOB_BUF(tmpcacheitem); int r, fd; + time_t now = time(NULL); apk_blob_push_blob(&b, tmpprefix); if (pkg != NULL) @@ -633,9 +634,9 @@ int apk_cache_download(struct apk_database *db, struct apk_repository *repo, r = apk_repo_format_real_url(db, repo, pkg, url, sizeof(url)); if (r < 0) return r; - if ((apk_force & APK_FORCE_REFRESH) || - fstatat(db->cache_fd, cacheitem, &st, 0) != 0) - st.st_mtime = 0; + if (!(apk_force & APK_FORCE_REFRESH)) + (void) fstatat(db->cache_fd, cacheitem, &st, 0); + if (autoupdate && now - st.st_mtime <= db->cache_max_age) return -EALREADY; apk_message("fetch %s", url); @@ -645,7 +646,7 @@ int apk_cache_download(struct apk_database *db, struct apk_repository *repo, if (verify != APK_SIGN_NONE) { apk_sign_ctx_init(&sctx, APK_SIGN_VERIFY, NULL, db->keys_fd); bs = apk_bstream_from_url_if_modified(url, st.st_mtime); - bs = apk_bstream_tee(bs, db->cache_fd, tmpcacheitem, cb, cb_ctx); + bs = apk_bstream_tee(bs, db->cache_fd, tmpcacheitem, !autoupdate, cb, cb_ctx); is = apk_bstream_gunzip_mpart(bs, apk_sign_ctx_mpart_cb, &sctx); if (!IS_ERR_OR_NULL(is)) r = apk_tar_parse(is, apk_sign_ctx_verify_tar, &sctx, FALSE, &db->id_cache); @@ -662,13 +663,18 @@ int apk_cache_download(struct apk_database *db, struct apk_repository *repo, if (fd >= 0) { struct apk_file_meta meta; r = apk_istream_splice(is, fd, APK_SPLICE_ALL, cb, cb_ctx); - apk_istream_get_meta(is, &meta); - apk_file_meta_to_fd(fd, &meta); + if (!autoupdate) { + apk_istream_get_meta(is, &meta); + apk_file_meta_to_fd(fd, &meta); + } close(fd); } } if (!IS_ERR_OR_NULL(is)) apk_istream_close(is); - if (r == -EALREADY) return 0; + if (r == -EALREADY) { + if (autoupdate) utimensat(db->cache_fd, cacheitem, NULL, 0); + return r; + } if (r < 0) { unlinkat(db->cache_fd, tmpcacheitem, 0); return r; @@ -1515,6 +1521,7 @@ int apk_db_open(struct apk_database *db, struct apk_db_options *dbopts) apk_db_setup_repositories(db, dbopts->cache_dir); + db->cache_max_age = dbopts->cache_max_age ?: 4*60*60; /* 4 hours default */ db->root = strdup(dbopts->root ?: "/"); db->root_fd = openat(AT_FDCWD, db->root, O_RDONLY | O_CLOEXEC); if (db->root_fd < 0 && (dbopts->open_flags & APK_OPENF_CREATE)) { @@ -1677,7 +1684,7 @@ int apk_db_open(struct apk_database *db, struct apk_db_options *dbopts) add_repos_from_file(db, db->root_fd, dbopts->repositories_file); } - if (apk_flags & APK_UPDATE_CACHE) + if (db->repo_update_counter) apk_db_index_write_nr_cache(db); apk_hash_foreach(&db->available.names, apk_db_name_rdepends, db); @@ -2118,10 +2125,13 @@ static int apk_repository_update(struct apk_database *db, struct apk_repository { int r, verify = (apk_flags & APK_ALLOW_UNTRUSTED) ? APK_SIGN_NONE : APK_SIGN_VERIFY; - r = apk_cache_download(db, repo, NULL, verify, NULL, NULL); + r = apk_cache_download(db, repo, NULL, verify, 1, NULL, NULL); + if (r == -EALREADY) return 0; if (r != 0) { apk_error("%s: %s", repo->url, apk_error_str(r)); db->repo_update_errors++; + } else { + db->repo_update_counter++; } return r; @@ -2247,16 +2257,13 @@ int apk_db_add_repository(apk_database_t _db, apk_blob_t _repository) apk_blob_checksum(brepo, apk_checksum_default(), &repo->csum); if (apk_url_local_file(repo->url) == NULL) { - if (!(apk_flags & APK_NO_NETWORK)) { + if (!(apk_flags & APK_NO_NETWORK)) db->available_repos |= BIT(repo_num); - if (apk_flags & APK_UPDATE_CACHE) - apk_repository_update(db, repo); - } if (apk_flags & APK_NO_CACHE) { r = apk_repo_format_real_url(db, repo, NULL, buf, sizeof(buf)); - if (r == 0) - apk_message("fetch %s", buf); + if (r == 0) apk_message("fetch %s", buf); } else { + apk_repository_update(db, repo); r = apk_repo_format_cache_index(APK_BLOB_BUF(buf), repo); } } else { @@ -2723,7 +2730,7 @@ static int apk_db_unpack_pkg(struct apk_database *db, apk_blob_t b = APK_BLOB_BUF(tmpcacheitem); apk_blob_push_blob(&b, tmpprefix); apk_pkg_format_cache_pkg(b, pkg); - cache_bs = apk_bstream_tee(bs, db->cache_fd, tmpcacheitem, NULL, NULL); + cache_bs = apk_bstream_tee(bs, db->cache_fd, tmpcacheitem, 1, NULL, NULL); if (!IS_ERR_OR_NULL(cache_bs)) bs = cache_bs; else |