Re: [PATCH 2/2] policAycoreutils: share setfiles restore function with restorecond

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 8/20/09 1:45 PM, "Daniel J Walsh" <dwalsh@xxxxxxxxxx> wrote:

> On 08/20/2009 10:15 AM, Joshua Brindle wrote:
>> Daniel J Walsh wrote:
>>> On 08/19/2009 05:01 PM, Joshua Brindle wrote:
>>>> Thomas Liu wrote:
>>>>> This patch edits restorecond to use the shared restore
>>>>> 
>>>>> functions split out from the first patch in this series.
>>>>> 
>>>> Is the DBus stuff in here suppose to be here? It looks like you are
>>>> adding libdbus and include dirs but I don't see any dbus usage.
>>>> 
>> <snip>
>>>> 
>>> restorcond -u is a dbus session bus service.
>>> 
>>> The idea is to run restorecond as your UID when you log into a console
>>> session.
>>> 
>> 
>> Maybe I don't get how DBus works. Why are the include dirs added to
>> CFLAGS but no apparent dbus #includes?
>> 
>> Further, will this _only_ work with DBus after this patch? If so there
>> needs to be a way to not build it if dbus isn't present on the system
>> (embedded).
>> 
>> -- 
>> This message was distributed to subscribers of the selinux mailing list.
>> If you no longer wish to subscribe, send mail to majordomo@xxxxxxxxxxxxx
>> with
>> the words "unsubscribe selinux" without quotes as the message.
>> 
>> 
> Ok here is another patch with all of the dbus stuff aggregated under a
> HAVE_DBUS flag.
> 
> You can build it with or without DBUS support and it should work fine.
> 
> DBUS is really used to make sure multiple restorecond do not get started.
> diff --git a/policycoreutils/restorecond/Makefile
> b/policycoreutils/restorecond/Makefile
> index 3f235e6..24aa266 100644
> --- a/policycoreutils/restorecond/Makefile
> +++ b/policycoreutils/restorecond/Makefile
> @@ -1,17 +1,28 @@
>  # Installation directories.
>  PREFIX ?= ${DESTDIR}/usr
>  SBINDIR ?= $(PREFIX)/sbin
> +LIBDIR ?= $(PREFIX)/lib
>  MANDIR = $(PREFIX)/share/man
> +AUTOSTARTDIR = $(DESTDIR)/etc/xdg/autostart
> +DBUSSERVICEDIR = $(DESTDIR)/usr/share/dbus-1/services
> +
> +autostart_DATA = sealertauto.desktop
>  INITDIR = $(DESTDIR)/etc/rc.d/init.d
>  SELINUXDIR = $(DESTDIR)/etc/selinux
>  
> +DBUSFLAGS = -DHAVE_DBUS -I/usr/include/dbus-1.0 -I/usr/lib64/dbus-1.0/include
> -I/usr/lib/dbus-1.0/include
> +DBUSLIB = -ldbus-glib-1
> +
>  CFLAGS ?= -g -Werror -Wall -W
> -override CFLAGS += -I$(PREFIX)/include -D_FILE_OFFSET_BITS=64
> -LDLIBS += -lselinux -L$(PREFIX)/lib
> +override CFLAGS += -I$(PREFIX)/include $(DBUSFLAGS) -I/usr/include/glib-2.0
> -I/usr/lib64/glib-2.0/include -I/usr/lib/glib-2.0/include
> +
> +LDLIBS += -lselinux $(DBUSLIB) -lglib-2.0 -L$(LIBDIR)
>  
>  all: restorecond
>  
> -restorecond:  restorecond.o utmpwatcher.o stringslist.o
> +restorecond.o utmpwatcher.o stringslist.o user.o watch.o: restorecond.h

You added watch.o and user.o here but there is no user.c or watch.c in this
patch. So, it won't compile (and many functions are missing).

> +
> +restorecond:  ../setfiles/restore.o restorecond.o utmpwatcher.o stringslist.o
> user.o watch.o
>      $(CC) $(LDFLAGS) -o $@ $^ $(LDLIBS)
>  
>  install: all
> @@ -22,7 +33,12 @@ install: all
>      -mkdir -p $(INITDIR)
>      install -m 755 restorecond.init $(INITDIR)/restorecond
>      -mkdir -p $(SELINUXDIR)
> -    install -m 600 restorecond.conf $(SELINUXDIR)/restorecond.conf
> +    install -m 644 restorecond.conf $(SELINUXDIR)/restorecond.conf
> +    install -m 644 restorecond_user.conf $(SELINUXDIR)/restorecond_user.conf
> +    -mkdir -p $(AUTOSTARTDIR)
> +    install -m 644 restorecond.desktop $(AUTOSTARTDIR)/restorecond.desktop
> +    -mkdir -p $(DBUSSERVICEDIR)
> +    install -m 600 org.selinux.Restorecond.service
> $(DBUSSERVICEDIR)/org.selinux.Restorecond.service
>  
>  relabel: install
>      /sbin/restorecon $(SBINDIR)/restorecond
> diff --git a/policycoreutils/restorecond/restorecond.c
> b/policycoreutils/restorecond/restorecond.c
> index 58774e6..384f13e 100644
> --- a/policycoreutils/restorecond/restorecond.c
> +++ b/policycoreutils/restorecond/restorecond.c
> @@ -48,294 +48,38 @@
>  #include <signal.h>
>  #include <string.h>
>  #include <unistd.h>
> -#include <ctype.h>
> +#include "../setfiles/restore.h"
>  #include <sys/types.h>
> -#include <sys/stat.h>
>  #include <syslog.h>
>  #include <limits.h>
> +#include <pwd.h>
> +#include <sys/stat.h>
> +#include <string.h>
> +#include <stdio.h>
>  #include <fcntl.h>
> -
>  #include "restorecond.h"
> -#include "stringslist.h"
>  #include "utmpwatcher.h"
>  
> -extern char *dirname(char *path);
> +const char *homedir;
>  static int master_fd = -1;
> -static int master_wd = -1;
> -static int terminate = 0;
> -
> -#include <selinux/selinux.h>
> -#include <utmp.h>
> -
> -/* size of the event structure, not counting name */
> -#define EVENT_SIZE  (sizeof (struct inotify_event))
> -/* reasonable guess as to size of 1024 events */
> -#define BUF_LEN        (1024 * (EVENT_SIZE + 16))
> -
> -static int debug_mode = 0;
> -static int verbose_mode = 0;
> -
> -static void restore(const char *filename, int exact);
> -
> -struct watchList {
> -    struct watchList *next;
> -    int wd;
> -    char *dir;
> -    struct stringsList *files;
> -};
> -struct watchList *firstDir = NULL;
> -
> -/* Compare two contexts to see if their differences are "significant",
> - * or whether the only difference is in the user. */
> -static int only_changed_user(const char *a, const char *b)
> -{
> -    char *rest_a, *rest_b;    /* Rest of the context after the user */
> -    if (!a || !b)
> -        return 0;
> -    rest_a = strchr(a, ':');
> -    rest_b = strchr(b, ':');
> -    if (!rest_a || !rest_b)
> -        return 0;
> -    return (strcmp(rest_a, rest_b) == 0);
> -}
> -
> -/* 
> -   A file was in a direcroty has been created. This function checks to
> -   see if it is one that we are watching.
> -*/
> -
> -static int watch_list_find(int wd, const char *file)
> -{
> -    struct watchList *ptr = NULL;
> -    ptr = firstDir;
> -
> -    if (debug_mode)
> -        printf("%d: File=%s\n", wd, file);
> -    while (ptr != NULL) {
> -        if (ptr->wd == wd) {
> -            int exact=0;
> -            if (strings_list_find(ptr->files, file, &exact) == 0) {
> -                char *path = NULL;
> -                if (asprintf(&path, "%s/%s", ptr->dir, file) <
> -                    0)
> -                    exitApp("Error allocating memory.");
> -                restore(path, exact);
> -                free(path);
> -                return 0;
> -            }
> -            if (debug_mode)
> -                strings_list_print(ptr->files);
> -
> -            /* Not found in this directory */
> -            return -1;
> -        }
> -        ptr = ptr->next;
> -    }
> -    /* Did not find a directory */
> -    return -1;
> -}
> -
> -static void watch_list_free(int fd)
> -{
> -    struct watchList *ptr = NULL;
> -    struct watchList *prev = NULL;
> -    ptr = firstDir;
> -
> -    while (ptr != NULL) {
> -        inotify_rm_watch(fd, ptr->wd);
> -        strings_list_free(ptr->files);
> -        free(ptr->dir);
> -        prev = ptr;
> -        ptr = ptr->next;
> -        free(prev);
> -    }
> -    firstDir = NULL;
> -}
> -
> -/* 
> -   Set the file context to the default file context for this system.
> -   Same as restorecon.
> -*/
> -static void restore(const char *filename, int exact)
> -{
> -    int retcontext = 0;
> -    security_context_t scontext = NULL;
> -    security_context_t prev_context = NULL;
> -    struct stat st;
> -    int fd = -1;
> -    if (debug_mode)
> -        printf("restore %s\n", filename);
> -
> -    fd = open(filename, O_NOFOLLOW | O_RDONLY);
> -    if (fd < 0) {
> -        if (verbose_mode)
> -            syslog(LOG_ERR, "Unable to open file (%s) %s\n",
> -                   filename, strerror(errno));
> -        return;
> -    }
> -
> -    if (fstat(fd, &st) != 0) {
> -        syslog(LOG_ERR, "Unable to stat file (%s) %s\n", filename,
> -               strerror(errno));
> -        close(fd);
> -        return;
> -    }
> -
> -    if (!(st.st_mode & S_IFDIR) && st.st_nlink > 1) {
> -        if (exact) {
> -            syslog(LOG_ERR,
> -                   "Will not restore a file with more than one hard link (%s)
> %s\n",
> -                   filename, strerror(errno));
> -        }
> -        close(fd);
> -        return;
> -    }
> -
> -    if (matchpathcon(filename, st.st_mode, &scontext) < 0) {
> -        if (errno == ENOENT)
> -            return;
> -        syslog(LOG_ERR, "matchpathcon(%s) failed %s\n", filename,
> -               strerror(errno));
> -        return;
> -    }
> -    retcontext = fgetfilecon_raw(fd, &prev_context);
> -
> -    if (retcontext >= 0 || errno == ENODATA) {
> -        if (retcontext < 0)
> -            prev_context = NULL;
> -        if (retcontext < 0 || (strcmp(prev_context, scontext) != 0)) {
> -
> -            if (only_changed_user(scontext, prev_context) != 0) {
> -                free(scontext);
> -                free(prev_context);
> -                close(fd);
> -                return;
> -            }
> -
> -            if (fsetfilecon(fd, scontext) < 0) {
> -                if (errno != EOPNOTSUPP)
> -                    syslog(LOG_ERR,
> -                           "set context %s->%s failed:'%s'\n",
> -                           filename, scontext, strerror(errno));
> -                if (retcontext >= 0)
> -                    free(prev_context);
> -                free(scontext);
> -                close(fd);
> -                return;
> -            }
> -            syslog(LOG_WARNING, "Reset file context %s: %s->%s\n",
> -                   filename, prev_context, scontext);
> -        }
> -        if (retcontext >= 0)
> -            free(prev_context);
> -    } else {
> -        if (errno != EOPNOTSUPP)
> -            syslog(LOG_ERR, "get context on %s failed: '%s'\n",
> -                   filename, strerror(errno));
> -    }
> -    free(scontext);
> -    close(fd);
> -}
> -
> -static void process_config(int fd, FILE * cfg)
> -{
> -    char *line_buf = NULL;
> -    size_t len = 0;
> -
> -    while (getline(&line_buf, &len, cfg) > 0) {
> -        char *buffer = line_buf;
> -        while (isspace(*buffer))
> -            buffer++;
> -        if (buffer[0] == '#')
> -            continue;
> -        int l = strlen(buffer) - 1;
> -        if (l <= 0)
> -            continue;
> -        buffer[l] = 0;
> -        if (buffer[0] == '~')
> -            utmpwatcher_add(fd, &buffer[1]);
> -        else {
> -            watch_list_add(fd, buffer);
> -        }
> -    }
> -    free(line_buf);
> -}
> -
> -/* 
> -   Read config file ignoring Comment lines
> -   Files specified one per line.  Files with "~" will be expanded to the
> logged in users
> -   homedirs.
> -*/
> -
> -static void read_config(int fd)
> -{
> -    char *watch_file_path = "/etc/selinux/restorecond.conf";
>  
> -    FILE *cfg = NULL;
> -    if (debug_mode)
> -        printf("Read Config\n");
> -
> -    watch_list_free(fd);
> -
> -    cfg = fopen(watch_file_path, "r");
> -    if (!cfg)
> -        exitApp("Error reading config file.");
> -    process_config(fd, cfg);
> -    fclose(cfg);
> -
> -    inotify_rm_watch(fd, master_wd);
> -    master_wd =
> -        inotify_add_watch(fd, watch_file_path, IN_MOVED_FROM | IN_MODIFY);
> -    if (master_wd == -1)
> -        exitApp("Error watching config file.");
> -}
> +static char *server_watch_file  = "/etc/selinux/restorecond.conf";
> +static char *user_watch_file  = "/etc/selinux/restorecond_user.conf";
> +static char *watch_file;
> +static struct restore_opts r_opts;
>  
> -/* 
> -   Inotify watch loop
> -*/
> -static int watch(int fd)
> -{
> -    char buf[BUF_LEN];
> -    int len, i = 0;
> -    len = read(fd, buf, BUF_LEN);
> -    if (len < 0) {
> -        if (terminate == 0) {
> -            syslog(LOG_ERR, "Read error (%s)", strerror(errno));
> -            return 0;
> -        }
> -        syslog(LOG_ERR, "terminated");
> -        return -1;
> -    } else if (!len)
> -        /* BUF_LEN too small? */
> -        return -1;
> -    while (i < len) {
> -        struct inotify_event *event;
> -        event = (struct inotify_event *)&buf[i];
> -        if (debug_mode)
> -            printf("wd=%d mask=%u cookie=%u len=%u\n",
> -                   event->wd, event->mask,
> -                   event->cookie, event->len);
> -        if (event->wd == master_wd)
> -            read_config(fd);
> -        else {
> -            switch (utmpwatcher_handle(fd, event->wd)) {
> -            case -1:    /* Message was not for utmpwatcher */
> -                if (event->len)
> -                    watch_list_find(event->wd, event->name);
> -                break;
> +#include <selinux/selinux.h>
>  
> -            case 1:    /* utmp has changed need to reload */
> -                read_config(fd);
> -                break;
> +int debug_mode = 0;
> +int terminate = 0;
> +int master_wd = -1;
> +int run_as_user = 0;
>  
> -            default:    /* No users logged in or out */
> -                break;
> -            }
> -        }
> -
> -        i += EVENT_SIZE + event->len;
> -    }
> -    return 0;
> +static void done(void) {
> +    watch_list_free(master_fd);
> +    close(master_fd);
> +    utmpwatcher_free();
> +    matchpathcon_fini();
>  }
>  
>  static const char *pidfile = "/var/run/restorecond.pid";
> @@ -374,7 +118,7 @@ static void term_handler()
>  
>  static void usage(char *program)
>  {
> -    printf("%s [-d] [-v] \n", program);
> +    printf("%s [-d] [-s] [-f restorecond_file ] [-v] \n", program);
>      exit(0);
>  }
>  
Could you limit the refactoring patch to just refactor? Adding functionality
(like the new -s and -f options) should be part of a separate patch.

Also, you list a -s option here, but don't use it below, while you use a -u
option below that's not listed here.

> @@ -390,74 +134,35 @@ void exitApp(const char *msg)
>     to see if it is one that we are watching.
>  */
>  
> -void watch_list_add(int fd, const char *path)
> -{
> -    struct watchList *ptr = NULL;
> -    struct watchList *prev = NULL;
> -    char *x = strdup(path);
> -    if (!x)
> -        exitApp("Out of Memory");
> -    char *dir = dirname(x);
> -    char *file = basename(path);
> -    ptr = firstDir;
> -
> -    restore(path, 1);
> -
> -    while (ptr != NULL) {
> -        if (strcmp(dir, ptr->dir) == 0) {
> -            strings_list_add(&ptr->files, file);
> -            free(x);
> -            return;
> -        }
> -        prev = ptr;
> -        ptr = ptr->next;
> -    }
> -    ptr = calloc(1, sizeof(struct watchList));
> -
> -    if (!ptr)
> -        exitApp("Out of Memory");
> -
> -    ptr->wd = inotify_add_watch(fd, dir, IN_CREATE | IN_MOVED_TO);
> -    if (ptr->wd == -1) {
> -        free(ptr);
> -        syslog(LOG_ERR, "Unable to watch (%s) %s\n",
> -               path, strerror(errno));
> -        return;
> -    }
> -
> -    ptr->dir = strdup(dir);
> -    if (!ptr->dir)
> -        exitApp("Out of Memory");
> -
> -    strings_list_add(&ptr->files, file);
> -    if (prev)
> -        prev->next = ptr;
> -    else
> -        firstDir = ptr;
> -
> -    if (debug_mode)
> -        printf("%d: Dir=%s, File=%s\n", ptr->wd, ptr->dir, file);
> -
> -    free(x);
> -}
> -
>  int main(int argc, char **argv)
>  {
>      int opt;
>      struct sigaction sa;
>  
> -#ifndef DEBUG
> -    /* Make sure we are root */
> -    if (getuid() != 0) {
> -        fprintf(stderr, "You must be root to run this program.\n");
> -        return 1;
> -    }
> -#endif
> -    /* Make sure we are root */
> -    if (is_selinux_enabled() != 1) {
> -        fprintf(stderr, "Daemon requires SELinux be enabled to run.\n");
> -        return 1;
> -    }
> +    memset(&r_opts, 0, sizeof(r_opts));
> +
> +    r_opts.progress = 0;
> +    r_opts.count = 0;
> +    r_opts.debug = 0;
> +    r_opts.change = 1;
> +    r_opts.verbose = 0;
> +    r_opts.logging = 0;
> +    r_opts.rootpath = NULL;
> +    r_opts.expand_realpath = 0;
> +    r_opts.rootpathlen = 0;
> +    r_opts.outfile = NULL;
> +    r_opts.force = 0;
> +    r_opts.hard_links = 0;
> +    r_opts.expand_realpath = 1;
> +    r_opts.abort_on_error = 0;
> +    r_opts.add_assoc = 0;
> +    r_opts.fts_flags = FTS_PHYSICAL;
> +    r_opts.selabel_opt_validate = NULL;
> +    r_opts.selabel_opt_path = NULL;
> +    
> +    restore_init(&r_opts);
> +    /* If we are not running SELinux then just exit */
> +    if (is_selinux_enabled() != 1) return 0;
>  
>      /* Register sighandlers */
>      sa.sa_flags = 0;
> @@ -467,38 +172,59 @@ int main(int argc, char **argv)
>  
>      set_matchpathcon_flags(MATCHPATHCON_NOTRANS);
>  
> -    master_fd = inotify_init();
> -    if (master_fd < 0)
> -        exitApp("inotify_init");
> -
> -    while ((opt = getopt(argc, argv, "dv")) > 0) {
> +    atexit( done );
> +    while ((opt = getopt(argc, argv, "uf:dv")) > 0) {
>          switch (opt) {
>          case 'd':
>              debug_mode = 1;
>              break;
> +        case 'f':
> +            watch_file = optarg;
> +            break;
> +        case 'u':
> +            run_as_user = 1;
> +            break;
>          case 'v':
> -            verbose_mode = 1;
> +            r_opts.verbose++;
>              break;
>          case '?':
>              usage(argv[0]);
>          }
>      }
> -    read_config(master_fd);
> +
> +    master_fd = inotify_init();
> +    if (master_fd < 0)
> +        exitApp("inotify_init");
> +
> +    uid_t uid = getuid();
> +    struct passwd *pwd = getpwuid(uid);
> +    homedir = pwd->pw_dir;
> +    if (uid != 0) {
> +        if (run_as_user)
> +            return server(master_fd, user_watch_file);
> +        if (start() != 0)
> +            return server(master_fd, user_watch_file);
> +        return 0;
> +    }
> +
> +    watch_file = server_watch_file;
> +    read_config(master_fd, watch_file);
>  
>      if (!debug_mode)
>          daemon(0, 0);
>  
>      write_pid_file();
>  
> -    while (watch(master_fd) == 0) {
> +    while (watch(master_fd, watch_file) == 0) {
>      };
>  
>      watch_list_free(master_fd);
>      close(master_fd);
>      matchpathcon_fini();
> -    utmpwatcher_free();
>      if (pidfile)
>          unlink(pidfile);
>  
>      return 0;
>  }
> +
> +
> diff --git a/policycoreutils/restorecond/restorecond.conf
> b/policycoreutils/restorecond/restorecond.conf
> index 3fc9376..58b723a 100644
> --- a/policycoreutils/restorecond/restorecond.conf
> +++ b/policycoreutils/restorecond/restorecond.conf
> @@ -4,8 +4,5 @@
>  /etc/mtab
>  /var/run/utmp
>  /var/log/wtmp
> -~/*
> -/root/.ssh
> +/root/*
>  /root/.ssh/*
> -
> -
> diff --git a/policycoreutils/restorecond/restorecond.h
> b/policycoreutils/restorecond/restorecond.h
> index e1666bf..f6452ea 100644
> --- a/policycoreutils/restorecond/restorecond.h
> +++ b/policycoreutils/restorecond/restorecond.h
> @@ -24,7 +24,21 @@
>  #ifndef RESTORED_CONFIG_H
>  #define RESTORED_CONFIG_H
>  
> -void exitApp(const char *msg);
> -void watch_list_add(int inotify_fd, const char *path);
> +extern int debug_mode;
> +extern const char *homedir;
> +extern int terminate;
> +extern int master_wd;
> +extern int run_as_user;
> +
> +extern int start(void);
> +extern int server(int, const char *watch_file);
> +
> +extern void exitApp(const char *msg);
> +extern void read_config(int fd,    const char *watch_file);
> +
> +extern int watch(int fd, const char *watch_file);
> +extern void watch_list_add(int inotify_fd, const char *path);
> +extern int watch_list_find(int wd, const char *file);
> +extern void watch_list_free(int fd);
>  
>  #endif


--
This message was distributed to subscribers of the selinux mailing list.
If you no longer wish to subscribe, send mail to majordomo@xxxxxxxxxxxxx with
the words "unsubscribe selinux" without quotes as the message.

[Index of Archives]     [Selinux Refpolicy]     [Linux SGX]     [Fedora Users]     [Fedora Desktop]     [Yosemite Photos]     [Yosemite Camping]     [Yosemite Campsites]     [KDE Users]     [Gnome Users]

  Powered by Linux