summaryrefslogtreecommitdiffstats
path: root/src/tools/sssctl/sssctl_data.c
diff options
context:
space:
mode:
authorPavel Březina <pbrezina@redhat.com>2016-06-01 13:38:22 +0200
committerJakub Hrozek <jhrozek@redhat.com>2016-06-27 16:34:58 +0200
commite157b9f6cb370e1b94bcac2044d26ad66d640fba (patch)
tree4ef6a872b33be17c40f327d7c85dcf13450d7ba9 /src/tools/sssctl/sssctl_data.c
parent9e9ad4cb181c6c0ec70caacfb31319753f889e98 (diff)
downloadsssd-e157b9f6cb370e1b94bcac2044d26ad66d640fba.tar.gz
sssd-e157b9f6cb370e1b94bcac2044d26ad66d640fba.tar.xz
sssd-e157b9f6cb370e1b94bcac2044d26ad66d640fba.zip
sssctl: new tool
Reviewed-by: Jakub Hrozek <jhrozek@redhat.com>
Diffstat (limited to 'src/tools/sssctl/sssctl_data.c')
-rw-r--r--src/tools/sssctl/sssctl_data.c239
1 files changed, 239 insertions, 0 deletions
diff --git a/src/tools/sssctl/sssctl_data.c b/src/tools/sssctl/sssctl_data.c
new file mode 100644
index 000000000..31b7984a5
--- /dev/null
+++ b/src/tools/sssctl/sssctl_data.c
@@ -0,0 +1,239 @@
+/*
+ Authors:
+ Pavel Březina <pbrezina@redhat.com>
+
+ Copyright (C) 2016 Red Hat
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 3 of the License, or
+ (at your option) any later version.
+
+ This program 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 General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include <popt.h>
+#include <stdio.h>
+
+#include "util/util.h"
+#include "tools/common/sss_tools.h"
+#include "tools/sssctl/sssctl.h"
+
+#define CACHE_FILE(db) " " DB_PATH "/" db
+#define CACHE_FILES CACHE_FILE("*.ldb")
+
+#define SSS_BACKUP_DIR SSS_STATEDIR "/backup"
+#define SSS_BACKUP_USER_OVERRIDES SSS_BACKUP_DIR "/sssd_user_overrides.bak"
+#define SSS_BACKUP_GROUP_OVERRIDES SSS_BACKUP_DIR "/sssd_group_overrides.bak"
+
+struct sssctl_data_opts {
+ int override;
+ int start;
+ int stop;
+ int restart;
+};
+
+static errno_t sssctl_create_backup_dir(const char *path)
+{
+ mode_t old_umask;
+ errno_t ret;
+
+ old_umask = umask(SSS_DFL_X_UMASK);
+ ret = mkdir(path, 0700);
+ umask(old_umask);
+ if (ret != EOK && errno != EEXIST) {
+ ret = errno;
+ DEBUG(SSSDBG_MINOR_FAILURE, "Unable to create backup directory "
+ "[%d]: %s\n", ret, sss_strerror(ret));
+ return ret;
+ }
+
+ return EOK;
+}
+
+static bool sssctl_backup_file_exists(const char *file)
+{
+ return access(file, F_OK) == 0;
+}
+
+static bool sssctl_backup_exist(const char **files)
+{
+ int i;
+
+ for (i = 0; files[i] != NULL; i++) {
+ if (sssctl_backup_file_exists(files[i])) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+static errno_t sssctl_backup(bool force)
+{
+ const char *files[] = {SSS_BACKUP_USER_OVERRIDES,
+ SSS_BACKUP_GROUP_OVERRIDES,
+ NULL};
+ enum sssctl_prompt_result prompt;
+ errno_t ret;
+
+ ret = sssctl_create_backup_dir(SSS_BACKUP_DIR);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to create backup directory [%d]: %s"),
+ ret, sss_strerror(ret));
+ return ret;
+ }
+
+ if (sssctl_backup_exist(files) && !force) {
+ prompt = sssctl_prompt(_("SSSD backup of local data already exist, "
+ "override?"), SSSCTL_PROMPT_NO);
+ switch (prompt) {
+ case SSSCTL_PROMPT_YES:
+ /* continue */
+ break;
+ case SSSCTL_PROMPT_NO:
+ return EEXIST;
+ case SSSCTL_PROMPT_ERROR:
+ return EIO;
+ }
+ }
+
+ ret = sssctl_run_command("sss_override user-export "
+ SSS_BACKUP_USER_OVERRIDES);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to export user overrides\n"));
+ return ret;
+ }
+
+ ret = sssctl_run_command("sss_override group-export "
+ SSS_BACKUP_GROUP_OVERRIDES);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to export group overrides\n"));
+ return ret;
+ }
+
+ return ret;
+}
+
+errno_t sssctl_backup_local_data(struct sss_cmdline *cmdline,
+ struct sss_tool_ctx *tool_ctx,
+ void *pvt)
+{
+ struct sssctl_data_opts opts = {0};
+ errno_t ret;
+
+ /* Parse command line. */
+ struct poptOption options[] = {
+ {"override", 'o', POPT_ARG_NONE, &opts.override, 0, _("Override existing backup"), NULL },
+ POPT_TABLEEND
+ };
+
+ ret = sss_tool_popt(cmdline, options, SSS_TOOL_OPT_OPTIONAL, NULL, NULL);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE, "Unable to parse command arguments\n");
+ return ret;
+ }
+
+ ret = sssctl_backup(opts.override);
+ if (ret == EEXIST) {
+ return EOK;
+ }
+
+ return ret;
+}
+
+errno_t sssctl_restore_local_data(struct sss_cmdline *cmdline,
+ struct sss_tool_ctx *tool_ctx,
+ void *pvt)
+{
+ struct sssctl_data_opts opts = {0};
+ errno_t ret;
+
+ /* Parse command line. */
+ struct poptOption options[] = {
+ {"start", 's', POPT_ARG_NONE, &opts.start, 0, _("Start SSSD if it is not running"), NULL },
+ {"restart", 'r', POPT_ARG_NONE, &opts.restart, 0, _("Restart SSSD after data import"), NULL },
+ POPT_TABLEEND
+ };
+
+ ret = sss_tool_popt(cmdline, options, SSS_TOOL_OPT_OPTIONAL, NULL, NULL);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE, "Unable to parse command arguments\n");
+ return ret;
+ }
+
+ if (!sssctl_start_sssd(opts.start)) {
+ return ERR_SSSD_NOT_RUNNING;
+ }
+
+
+ if (sssctl_backup_file_exists(SSS_BACKUP_USER_OVERRIDES)) {
+ ret = sssctl_run_command("sss_override user-import "
+ SSS_BACKUP_USER_OVERRIDES);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to import user overrides\n"));
+ return ret;
+ }
+ }
+
+ if (sssctl_backup_file_exists(SSS_BACKUP_USER_OVERRIDES)) {
+ ret = sssctl_run_command("sss_override group-import "
+ SSS_BACKUP_GROUP_OVERRIDES);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to import group overrides\n"));
+ return ret;
+ }
+ }
+
+ sssctl_restart_sssd(opts.restart);
+
+ return ret;
+}
+
+errno_t sssctl_remove_cache(struct sss_cmdline *cmdline,
+ struct sss_tool_ctx *tool_ctx,
+ void *pvt)
+{
+ struct sssctl_data_opts opts = {0};
+ errno_t ret;
+
+ /* Parse command line. */
+ struct poptOption options[] = {
+ {"override", 'o', POPT_ARG_NONE, &opts.override, 0, _("Override existing backup"), NULL },
+ {"stop", 's', POPT_ARG_NONE, &opts.stop, 0, _("Stop SSSD if it is running"), NULL },
+ POPT_TABLEEND
+ };
+
+ ret = sss_tool_popt(cmdline, options, SSS_TOOL_OPT_OPTIONAL, NULL, NULL);
+ if (ret != EOK) {
+ DEBUG(SSSDBG_CRIT_FAILURE, "Unable to parse command arguments\n");
+ return ret;
+ }
+
+ if (!sssctl_stop_sssd(opts.stop)) {
+ return ERR_SSSD_RUNNING;
+ }
+
+ printf(_("Creating backup of local data...\n"));
+ ret = sssctl_backup(opts.override);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to create backup of local data,"
+ " can not remove the cache.\n"));
+ return ret;
+ }
+
+ printf(_("Removing cache files...\n"));
+ ret = sssctl_run_command("rm -f " CACHE_FILES);
+ if (ret != EOK) {
+ fprintf(stderr, _("Unable to remove cache files\n"));
+ return ret;
+ }
+
+ return EOK;
+}