summaryrefslogtreecommitdiff
path: root/src/libostree
diff options
context:
space:
mode:
authorJonathan Lebon <jlebon@redhat.com>2017-08-23 12:15:46 -0400
committerAtomic Bot <atomic-devel@projectatomic.io>2017-08-25 01:02:15 +0000
commit9342be6e34b26891c38b671ec285bdcbc9107acd (patch)
tree0f91f5606ecb11213356aa9424f468d672a6f8d2 /src/libostree
parentd0f40a6af8fb391d5ee34ca6d4bc2abfc5b21e16 (diff)
downloadostree-9342be6e34b26891c38b671ec285bdcbc9107acd.tar.gz
ostree-sysroot: make simple_write_deployment smarter
This is a follow-up to https://github.com/ostreedev/ostree/pull/1097. We make simple_write_deployment smart enough so that it can be used for rpm-ostree's purposes. This is mostly an upstreaming of logic that already existed there. Notably we correctly append NOT_DEFAULT deployments *after* the booted deployment and we now support RETAIN_PENDING and RETAIN_ROLLBACK flags to have more granularity on deployment pruning. Expose these new flags on the CLI using new options (as well as expose the previously existing NOT_DEFAULT flag as --not-as-default). I couldn't add tests for --retain-pending because the merge deployment is always the topmost one. Though I did check that it worked in a VM. Closes: #1110 Approved by: cgwalters
Diffstat (limited to 'src/libostree')
-rw-r--r--src/libostree/ostree-sysroot.c70
-rw-r--r--src/libostree/ostree-sysroot.h2
2 files changed, 52 insertions, 20 deletions
diff --git a/src/libostree/ostree-sysroot.c b/src/libostree/ostree-sysroot.c
index 65cfb7c3..8e6c475c 100644
--- a/src/libostree/ostree-sysroot.c
+++ b/src/libostree/ostree-sysroot.c
@@ -1487,6 +1487,12 @@ ostree_sysroot_init_osname (OstreeSysroot *self,
* If %OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN is
* specified, then all current deployments will be kept.
*
+ * If %OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_PENDING is
+ * specified, then pending deployments will be kept.
+ *
+ * If %OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_ROLLBACK is
+ * specified, then rollback deployments will be kept.
+ *
* If %OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_NOT_DEFAULT is
* specified, then instead of prepending, the new deployment will be
* added right after the booted or merge deployment, instead of first.
@@ -1507,10 +1513,14 @@ ostree_sysroot_simple_write_deployment (OstreeSysroot *sysroot,
{
const gboolean postclean =
(flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_NO_CLEAN) == 0;
- const gboolean retain =
- (flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN) > 0;
const gboolean make_default =
!((flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_NOT_DEFAULT) > 0);
+ const gboolean retain_pending =
+ (flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_PENDING) > 0;
+ const gboolean retain_rollback =
+ (flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_ROLLBACK) > 0;
+ gboolean retain =
+ (flags & OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN) > 0;
g_autoptr(GPtrArray) deployments = ostree_sysroot_get_deployments (sysroot);
OstreeDeployment *booted_deployment = ostree_sysroot_get_booted_deployment (sysroot);
@@ -1526,34 +1536,54 @@ ostree_sysroot_simple_write_deployment (OstreeSysroot *sysroot,
added_new = TRUE;
}
+ /* without a booted and a merge deployment, retain_pending/rollback become meaningless;
+ * let's just retain all deployments in that case */
+ if (!booted_deployment && !merge_deployment && (retain_pending || retain_rollback))
+ retain = TRUE;
+
+ /* tracks when we come across the booted deployment */
+ gboolean before_booted = TRUE;
+ gboolean before_merge = TRUE;
for (guint i = 0; i < deployments->len; i++)
{
OstreeDeployment *deployment = deployments->pdata[i];
- const gboolean is_merge_or_booted =
- ostree_deployment_equal (deployment, booted_deployment) ||
- ostree_deployment_equal (deployment, merge_deployment);
-
- /* Keep deployments with different osnames, as well as the
- * booted and merge deployments
+ const gboolean osname_matches =
+ (osname == NULL || g_str_equal (ostree_deployment_get_osname (deployment), osname));
+ const gboolean is_booted = ostree_deployment_equal (deployment, booted_deployment);
+ const gboolean is_merge = ostree_deployment_equal (deployment, merge_deployment);
+
+ if (is_booted)
+ before_booted = FALSE;
+ if (is_merge)
+ before_merge = FALSE;
+
+ /* use the booted deployment as the "crossover" point between pending and rollback
+ * deployments, fall back on merge deployment */
+ const gboolean passed_crossover = booted_deployment ? !before_booted : !before_merge;
+
+ /* Retain deployment if:
+ * - we're explicitly asked to, or
+ * - the deployment is for another osname, or
+ * - we're keeping pending deployments and this is a pending deployment, or
+ * - this is the merge or boot deployment, or
+ * - we're keeping rollback deployments and this is a rollback deployment
*/
- if (retain ||
- (osname != NULL && strcmp (ostree_deployment_get_osname (deployment), osname) != 0) ||
- is_merge_or_booted)
- {
- g_ptr_array_add (new_deployments, g_object_ref (deployment));
- }
-
- if ((!added_new) && is_merge_or_booted)
+ if (retain
+ || !osname_matches
+ || (retain_pending && !passed_crossover)
+ || (is_booted || is_merge)
+ || (retain_rollback && passed_crossover))
+ g_ptr_array_add (new_deployments, g_object_ref (deployment));
+
+ /* add right after booted/merge deployment */
+ if (!added_new && passed_crossover)
{
g_ptr_array_add (new_deployments, g_object_ref (new_deployment));
added_new = TRUE;
}
}
- /* In this non-default case , an improvement in the future would be
- * to put the new deployment right after the current default in the
- * order.
- */
+ /* add it last if no crossover defined (or it's the first deployment in the sysroot) */
if (!added_new)
{
g_ptr_array_add (new_deployments, g_object_ref (new_deployment));
diff --git a/src/libostree/ostree-sysroot.h b/src/libostree/ostree-sysroot.h
index 3d2446f9..f2573d6b 100644
--- a/src/libostree/ostree-sysroot.h
+++ b/src/libostree/ostree-sysroot.h
@@ -207,6 +207,8 @@ typedef enum {
OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN = (1 << 0),
OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_NOT_DEFAULT = (1 << 1),
OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_NO_CLEAN = (1 << 2),
+ OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_PENDING = (1 << 3),
+ OSTREE_SYSROOT_SIMPLE_WRITE_DEPLOYMENT_FLAGS_RETAIN_ROLLBACK = (1 << 4),
} OstreeSysrootSimpleWriteDeploymentFlags;
_OSTREE_PUBLIC