Skip to content

Commit

Permalink
ostree: Add command admin nsenter
Browse files Browse the repository at this point in the history
  • Loading branch information
ruihe774 committed Dec 18, 2024
1 parent 1ca7e0e commit f0461d8
Show file tree
Hide file tree
Showing 4 changed files with 130 additions and 5 deletions.
1 change: 1 addition & 0 deletions Makefile-ostree.am
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,7 @@ ostree_SOURCES += \
src/ostree/ot-admin-builtin-upgrade.c \
src/ostree/ot-admin-builtin-unlock.c \
src/ostree/ot-admin-builtin-state-overlay.c \
src/ostree/ot-admin-builtin-nsenter.c \
src/ostree/ot-admin-builtins.h \
src/ostree/ot-admin-instutil-builtin-selinux-ensure-labeled.c \
src/ostree/ot-admin-instutil-builtin-set-kargs.c \
Expand Down
126 changes: 126 additions & 0 deletions src/ostree/ot-admin-builtin-nsenter.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,126 @@
/*
* Copyright (C) 2024 Colin Walters <[email protected]>
*
* SPDX-License-Identifier: LGPL-2.0+
*
* This library 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 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 <https://www.gnu.org/licenses/>.
*
* Author: Misaki Kasumi <[email protected]>
*/

#include "config.h"

#include "libglnx.h"
#include "ostree.h"
#include "ot-admin-builtins.h"
#include "ot-admin-functions.h"

#include <spawn.h>
#include <sys/wait.h>

static gboolean opt_lock;
static gboolean opt_exec;

static GOptionEntry options[] = {
{ "lock", 0, 0, G_OPTION_ARG_NONE, &opt_lock,
"Make /sysroot writable in the mount namespace and acquire an exclusive multi-process write lock", NULL },
{ "exec", 0, 0, G_OPTION_ARG_NONE, &opt_exec,
"Replace the process instead of spawning the program as child", NULL},
{ NULL } };

gboolean
ot_admin_builtin_nsenter (int argc, char **argv, OstreeCommandInvocation *invocation,
GCancellable *cancellable, GError **error)
{
g_autoptr (GOptionContext) context = NULL;
g_autoptr (OstreeSysroot) sysroot = NULL;
g_autofree char **arguments = NULL;

context = g_option_context_new ("[PROGRAM [ARGUMENTS...]]");

int new_argc = argc;
char **new_argv = argv;

for (int i = 1; i < argc; i++)
{
if (g_str_equal (argv[i], "--"))
{
new_argc -= i;
argc = i;
new_argv = argv + i;
argv[i] = NULL;
break;
}
}

if (!ostree_admin_option_context_parse (context, options, &argc, &argv,
OSTREE_ADMIN_BUILTIN_FLAG_UNLOCKED, invocation, &sysroot,
cancellable, error))
return FALSE;

argc = new_argc;
argv = new_argv;
if (argc <= 1)
{
arguments = g_malloc_n (2, sizeof (char *));
if ((arguments[0] = getenv ("SHELL")) == NULL)
arguments[0] = "/bin/sh";
arguments[1] = NULL;
}
else
{
arguments = g_malloc_n (argc, sizeof (char *));
memcpy (arguments, argv + 1, (argc - 1) * sizeof (char *));
arguments[argc - 1] = NULL;
}

if (opt_lock)
{
if (opt_exec)
return glnx_throw (error, "cannot specify both --lock and --exec");
if (!ostree_sysroot_lock (sysroot, error))
return FALSE;
}

pid_t child_pid;
if (opt_exec)
{
if (execvp (arguments[0], arguments) < 0)
return glnx_throw_errno_prefix (error, "execvp");
}
else
{
if (posix_spawnp (&child_pid, arguments[0], NULL, NULL, arguments, environ) != 0)
return glnx_throw_errno_prefix (error, "posix_spawnp");
}

int status;
while (waitpid (child_pid, &status, 0) < 0)
{
if (errno != EINTR)
return glnx_throw_errno_prefix (error, "waitpid");
}

if (opt_lock)
ostree_sysroot_unlock (sysroot);

if (!WIFEXITED (status))
return glnx_throw (error, "child process killed by signal");

int exit_status = WEXITSTATUS (status);
if (exit_status != EXIT_SUCCESS)
exit (exit_status);

return TRUE;
}
1 change: 1 addition & 0 deletions src/ostree/ot-admin-builtins.h
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ BUILTINPROTO (kargs);
BUILTINPROTO (post_copy);
BUILTINPROTO (lock_finalization);
BUILTINPROTO (state_overlay);
BUILTINPROTO (nsenter);

#undef BUILTINPROTO

Expand Down
7 changes: 2 additions & 5 deletions src/ostree/ot-builtin-admin.c
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,8 @@ static OstreeCommand admin_subcommands[] = {
{ "upgrade", OSTREE_BUILTIN_FLAG_NO_REPO, ot_admin_builtin_upgrade,
"Construct new tree from current origin and deploy it, if it changed" },
{ "kargs", OSTREE_BUILTIN_FLAG_NO_REPO, ot_admin_builtin_kargs, "Change kernel arguments" },
{"nsenter", OSTREE_BUILTIN_FLAG_NO_REPO | OSTREE_BUILTIN_FLAG_HIDDEN, ot_admin_builtin_nsenter,
"Run program in the mount namespace where /sysroot is present"},
{ NULL, 0, NULL, NULL }
};

Expand Down Expand Up @@ -121,11 +123,6 @@ ostree_builtin_admin (int argc, char **argv, OstreeCommandInvocation *invocation
}
}

else if (g_str_equal (argv[in], "--"))
{
break;
}

argv[out] = argv[in];
}

Expand Down

0 comments on commit f0461d8

Please sign in to comment.