/* Low level event script handling Copyright (C) Amitay Isaacs 2017 Copyright (C) Martin Schwenke 2018 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 . */ #include "replace.h" #include "system/filesys.h" #include "system/dir.h" #include "system/glob.h" #include #include "common/event_script.h" static int script_filter(const struct dirent *de) { int ret; /* Match a script pattern */ ret = fnmatch("[0-9][0-9].*.script", de->d_name, 0); if (ret == 0) { return 1; } return 0; } int event_script_get_list(TALLOC_CTX *mem_ctx, const char *script_dir, struct event_script_list **out) { struct dirent **namelist = NULL; struct event_script_list *script_list = NULL; size_t ds_len; int count, ret; int i; count = scandir(script_dir, &namelist, script_filter, alphasort); if (count == -1) { ret = errno; goto done; } script_list = talloc_zero(mem_ctx, struct event_script_list); if (script_list == NULL) { goto nomem; } if (count == 0) { ret = 0; *out = script_list; goto done; } script_list->num_scripts = count; script_list->script = talloc_zero_array(script_list, struct event_script *, count); if (script_list->script == NULL) { goto nomem; } ds_len = strlen(".script"); for (i = 0; i < count; i++) { struct event_script *s; struct stat statbuf; s = talloc_zero(script_list->script, struct event_script); if (s == NULL) { goto nomem; } script_list->script[i] = s; s->name = talloc_strndup(script_list->script, namelist[i]->d_name, strlen(namelist[i]->d_name) - ds_len); if (s->name == NULL) { goto nomem; } s->path = talloc_asprintf(script_list->script, "%s/%s", script_dir, namelist[i]->d_name); if (s->path == NULL) { goto nomem; } ret = stat(s->path, &statbuf); if (ret == 0) { /* * If ret != 0 this is either a dangling * symlink or it has just disappeared. Either * way, it isn't executable. See the note * below about things that have disappeared. */ if (statbuf.st_mode & S_IXUSR) { s->enabled = true; } } } *out = script_list; return 0; nomem: ret = ENOMEM; talloc_free(script_list); done: if (namelist != NULL && count != -1) { for (i=0; i ds_len && strcmp(&script_name[sn_len - ds_len], dot_script) == 0) { script_file = script_name; } else { ret = snprintf(buf, sizeof(buf), "%s.script", script_name); if (ret >= sizeof(buf)) { return ENAMETOOLONG; } script_file = buf; } dirp = opendir(script_dir); if (dirp == NULL) { return errno; } found = false; while ((de = readdir(dirp)) != NULL) { if (strcmp(de->d_name, script_file) == 0) { /* check for valid script names */ ret = script_filter(de); if (ret == 0) { closedir(dirp); return EINVAL; } found = true; found_inode = de->d_ino; break; } } closedir(dirp); if (! found) { return ENOENT; } ret = snprintf(filename, sizeof(filename), "%s/%s", script_dir, script_file); if (ret >= sizeof(filename)) { return ENAMETOOLONG; } fd = open(filename, O_RDWR); if (fd == -1) { ret = errno; goto done; } ret = fstat(fd, &st); if (ret != 0) { ret = errno; goto done; } /* * If the directory entry inode number doesn't match the one * returned by fstat() then this is probably a symlink, so the * caller should not be calling this function. Note that this * is a cheap sanity check to catch most programming errors. * This doesn't cost any extra system calls but can still miss * the unlikely case where the symlink is to a file on a * different filesystem with the same inode number as the * symlink. */ if (found && found_inode != st.st_ino) { ret = EINVAL; goto done; } if (enable) { new_mode = st.st_mode | (S_IXUSR | S_IXGRP | S_IXOTH); } else { new_mode = st.st_mode & ~(S_IXUSR | S_IXGRP | S_IXOTH); } ret = fchmod(fd, new_mode); if (ret != 0) { ret = errno; goto done; } done: if (fd != -1) { close(fd); } return ret; }