/* vi:set et sw=2 sts=2 cin cino=t0,f0,(0,{s,>2s,n-s,^-s,e-s: * Copyright © 2015 Red Hat, Inc * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . * * Authors: * Alexander Larsson */ #include "config.h" #include "flatpak-utils-private.h" #include "flatpak-ref.h" #include "flatpak-enum-types.h" /** * SECTION:flatpak-ref * @Title: FlatpakRef * @Short_description: Application reference * * Currently Flatpak manages two types of binary artifacts: applications, and * runtimes. Applications contain a program that desktop users can run, while * runtimes contain only libraries and data. An FlatpakRef object (or short: ref) * can refer to either of these. * * Both applications and runtimes are identified by a 4-tuple of strings: kind, * name, arch and branch, e.g. app/org.gnome.evince/x86_64/master. The functions * flatpak_ref_parse() and flatpak_ref_format_ref() can be used to convert * FlatpakRef objects into this string representation and back. * * Note that the identifiers must be unique within a repo (e.g. Flathub) based * only on the name, arch, and branch 3-tuple, without regard to the kind. In * other words if app/org.gnome.evince/x86_64/master exists, * runtime/org.gnome.evince/x86_64/master must not exist. This requirement is * not enforced by libflatpak but is enforced by GNOME Software's use of * libappstream, since Appstream IDs are assumed to be unique. * * FlatpakRef objects are immutable and can be passed freely between threads. * * To uniquely identify a particular version of an application or runtime, you * need a commit. * * The subclasses #FlatpakInstalledRef and #FlatpakRemoteRef provide more information * for artifacts that are locally installed or available from a remote repository. */ typedef struct _FlatpakRefPrivate FlatpakRefPrivate; struct _FlatpakRefPrivate { char *name; char *arch; char *branch; char *commit; FlatpakRefKind kind; char *collection_id; char *cached_full_ref; }; G_DEFINE_TYPE_WITH_PRIVATE (FlatpakRef, flatpak_ref, G_TYPE_OBJECT) enum { PROP_0, PROP_NAME, PROP_ARCH, PROP_BRANCH, PROP_COMMIT, PROP_KIND, PROP_COLLECTION_ID, }; static void flatpak_ref_finalize (GObject *object) { FlatpakRef *self = FLATPAK_REF (object); FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); g_free (priv->name); g_free (priv->arch); g_free (priv->branch); g_free (priv->commit); g_free (priv->collection_id); g_free ((char *)g_atomic_pointer_get (&priv->cached_full_ref)); G_OBJECT_CLASS (flatpak_ref_parent_class)->finalize (object); } /* These support setting e.g. the arch from referencing a ref. * i.e. it would get "x86_64/master" as an argument. */ static char * value_dup_ref_part (const GValue *value) { const char *part = value->data[0].v_pointer; const char *slash; slash = strchr (part, '/'); if (slash) return g_strndup (part, slash - part); return g_strdup (part); } static void flatpak_ref_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { FlatpakRef *self = FLATPAK_REF (object); FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); switch (prop_id) { case PROP_NAME: g_assert (priv->name == NULL); /* Construct-only */ priv->name = value_dup_ref_part (value); break; case PROP_ARCH: g_assert (priv->arch == NULL); /* Construct-only */ priv->arch = value_dup_ref_part (value); break; case PROP_BRANCH: g_assert (priv->branch == NULL); /* Construct-only */ priv->branch = value_dup_ref_part (value); break; case PROP_COMMIT: g_assert (priv->commit == NULL); /* Construct-only */ priv->commit = g_value_dup_string (value); break; case PROP_KIND: priv->kind = g_value_get_enum (value); break; case PROP_COLLECTION_ID: g_assert (priv->collection_id == NULL); /* Construct-only */ priv->collection_id = g_value_dup_string (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void flatpak_ref_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { FlatpakRef *self = FLATPAK_REF (object); FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); switch (prop_id) { case PROP_NAME: g_value_set_string (value, priv->name); break; case PROP_ARCH: g_value_set_string (value, priv->arch); break; case PROP_BRANCH: g_value_set_string (value, priv->branch); break; case PROP_COMMIT: g_value_set_string (value, priv->commit); break; case PROP_KIND: g_value_set_enum (value, priv->kind); break; case PROP_COLLECTION_ID: g_value_set_string (value, priv->collection_id); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void flatpak_ref_class_init (FlatpakRefClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->get_property = flatpak_ref_get_property; object_class->set_property = flatpak_ref_set_property; object_class->finalize = flatpak_ref_finalize; g_object_class_install_property (object_class, PROP_NAME, g_param_spec_string ("name", "Name", "The name of the application or runtime", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (object_class, PROP_ARCH, g_param_spec_string ("arch", "Architecture", "The architecture of the application or runtime", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (object_class, PROP_BRANCH, g_param_spec_string ("branch", "Branch", "The branch of the application or runtime", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (object_class, PROP_COMMIT, g_param_spec_string ("commit", "Commit", "The commit", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (object_class, PROP_KIND, g_param_spec_enum ("kind", "Kind", "The kind of artifact", FLATPAK_TYPE_REF_KIND, FLATPAK_REF_KIND_APP, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (object_class, PROP_COLLECTION_ID, g_param_spec_string ("collection-id", "Collection ID", "The collection ID", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS)); } static void flatpak_ref_init (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); priv->kind = FLATPAK_REF_KIND_APP; } /** * flatpak_ref_get_name: * @self: a #FlatpakRef * * Gets the name of the ref. * * Returns: (transfer none): the name */ const char * flatpak_ref_get_name (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->name; } /** * flatpak_ref_get_arch: * @self: a #FlatpakRef * * Gets the arch or the ref. * * Returns: (transfer none): the arch */ const char * flatpak_ref_get_arch (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->arch; } /** * flatpak_ref_get_branch: * @self: a #FlatpakRef * * Gets the branch of the ref. * * Returns: (transfer none): the branch */ const char * flatpak_ref_get_branch (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->branch; } /** * flatpak_ref_get_commit: * @self: a #FlatpakRef * * Gets the commit of the ref. * * Returns: (transfer none): the commit */ const char * flatpak_ref_get_commit (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->commit; } /** * flatpak_ref_get_kind: * @self: a #FlatpakRef * * Gets the kind of artifact that this ref refers to. * * Returns: the kind of artifact */ FlatpakRefKind flatpak_ref_get_kind (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->kind; } /** * flatpak_ref_format_ref: * @self: a #FlatpakRef * * Convert an FlatpakRef object into a string representation that * can be parsed by flatpak_ref_parse(). * * Returns: (transfer full): string representation */ char * flatpak_ref_format_ref (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); if (priv->kind == FLATPAK_REF_KIND_APP) return flatpak_build_app_ref (priv->name, priv->branch, priv->arch); else return flatpak_build_runtime_ref (priv->name, priv->branch, priv->arch); } /** * flatpak_ref_format_ref_cached: * @self: a #FlatpakRef * * Like flatpak_ref_format_ref() but this returns the same string each time * it's called rather than allocating a new one. * * Returns: (transfer none): string representation * * Since: 1.9.1 */ const char * flatpak_ref_format_ref_cached (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); const char *full_ref; char *full_ref_new; full_ref = (const char *)g_atomic_pointer_get (&priv->cached_full_ref); if (full_ref == NULL) { full_ref_new = flatpak_ref_format_ref (self); if (!g_atomic_pointer_compare_and_exchange ((void**) &priv->cached_full_ref, NULL, full_ref_new)) g_free (full_ref_new); /* Raced with someone, free our version */ full_ref = (const char *)g_atomic_pointer_get (&priv->cached_full_ref); /* Now guaranteed to be non-NULL */ } return full_ref; } /** * flatpak_ref_parse: * @ref: A string ref name, such as "app/org.test.App/x86_64/master" * @error: return location for a #GError * * Tries to parse a full ref name and return a #FlatpakRef (without a * commit set) or fail if the ref is invalid somehow. * * Returns: (transfer full): an #FlatpakRef, or %NULL */ FlatpakRef * flatpak_ref_parse (const char *ref, GError **error) { g_autoptr(FlatpakDecomposed) decomposed = NULL; decomposed = flatpak_decomposed_new_from_ref (ref, error); if (decomposed == NULL) return NULL; return FLATPAK_REF (g_object_new (FLATPAK_TYPE_REF, "kind", flatpak_decomposed_get_kind (decomposed), "name", flatpak_decomposed_peek_id (decomposed, NULL), "arch", flatpak_decomposed_peek_arch (decomposed, NULL), "branch", flatpak_decomposed_peek_branch (decomposed, NULL), NULL)); } /** * flatpak_ref_get_collection_id: * @self: a #FlatpakRef * * Gets the collection ID of the ref. * * Returns: (transfer none): the collection ID */ const char * flatpak_ref_get_collection_id (FlatpakRef *self) { FlatpakRefPrivate *priv = flatpak_ref_get_instance_private (self); return priv->collection_id; }