statcmd.c 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. /*
  2. * dpkg-statoverride - override ownership and mode of files
  3. *
  4. * Copyright © 2000, 2001 Wichert Akkerman <wakkerma@debian.org>
  5. * Copyright © 2006-2014 Guillem Jover <guillem@debian.org>
  6. *
  7. * This is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  19. *
  20. */
  21. #include <config.h>
  22. #include <compat.h>
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25. #include <errno.h>
  26. #if HAVE_LOCALE_H
  27. #include <locale.h>
  28. #endif
  29. #include <string.h>
  30. #include <grp.h>
  31. #include <pwd.h>
  32. #include <fnmatch.h>
  33. #include <unistd.h>
  34. #include <stdlib.h>
  35. #include <stdio.h>
  36. #include <dpkg/i18n.h>
  37. #include <dpkg/dpkg.h>
  38. #include <dpkg/dpkg-db.h>
  39. #include <dpkg/path.h>
  40. #include <dpkg/dir.h>
  41. #include <dpkg/glob.h>
  42. #include <dpkg/options.h>
  43. #include "main.h"
  44. #include "filesdb.h"
  45. static const char printforhelp[] = N_(
  46. "Use --help for help about overriding file stat information.");
  47. static void DPKG_ATTR_NORET
  48. printversion(const struct cmdinfo *cip, const char *value)
  49. {
  50. printf(_("Debian %s version %s.\n"), dpkg_get_progname(),
  51. DPKG_VERSION_ARCH);
  52. printf(_(
  53. "This is free software; see the GNU General Public License version 2 or\n"
  54. "later for copying conditions. There is NO warranty.\n"));
  55. m_output(stdout, _("<standard output>"));
  56. exit(0);
  57. }
  58. static void DPKG_ATTR_NORET
  59. usage(const struct cmdinfo *cip, const char *value)
  60. {
  61. printf(_(
  62. "Usage: %s [<option> ...] <command>\n"
  63. "\n"), dpkg_get_progname());
  64. printf(_(
  65. "Commands:\n"
  66. " --add <owner> <group> <mode> <path>\n"
  67. " add a new <path> entry into the database.\n"
  68. " --remove <path> remove <path> from the database.\n"
  69. " --list [<glob-pattern>] list current overrides in the database.\n"
  70. "\n"));
  71. printf(_(
  72. "Options:\n"
  73. " --admindir <directory> set the directory with the statoverride file.\n"
  74. " --update immediately update <path> permissions.\n"
  75. " --force force an action even if a sanity check fails.\n"
  76. " --quiet quiet operation, minimal output.\n"
  77. " --help show this help message.\n"
  78. " --version show the version.\n"
  79. "\n"));
  80. m_output(stdout, _("<standard output>"));
  81. exit(0);
  82. }
  83. static const char *admindir;
  84. static int opt_verbose = 1;
  85. static int opt_force = 0;
  86. static int opt_update = 0;
  87. static char *
  88. path_cleanup(const char *path)
  89. {
  90. char *new_path = m_strdup(path);
  91. path_trim_slash_slashdot(new_path);
  92. if (opt_verbose && strcmp(path, new_path) != 0)
  93. warning(_("stripping trailing /"));
  94. return new_path;
  95. }
  96. static struct file_stat *
  97. statdb_node_new(const char *user, const char *group, const char *mode)
  98. {
  99. struct file_stat *filestat;
  100. filestat = nfmalloc(sizeof(*filestat));
  101. filestat->uid = statdb_parse_uid(user);
  102. if (filestat->uid == (uid_t)-1)
  103. filestat->uname = nfstrsave(user);
  104. else
  105. filestat->uname = NULL;
  106. filestat->gid = statdb_parse_gid(group);
  107. if (filestat->gid == (gid_t)-1)
  108. filestat->gname = nfstrsave(group);
  109. else
  110. filestat->gname = NULL;
  111. filestat->mode = statdb_parse_mode(mode);
  112. return filestat;
  113. }
  114. static struct file_stat **
  115. statdb_node_find(const char *filename)
  116. {
  117. struct filenamenode *file;
  118. file = findnamenode(filename, 0);
  119. return &file->statoverride;
  120. }
  121. static int
  122. statdb_node_remove(const char *filename)
  123. {
  124. struct filenamenode *file;
  125. file = findnamenode(filename, fnn_nonew);
  126. if (!file || (file && !file->statoverride))
  127. return 0;
  128. file->statoverride = NULL;
  129. return 1;
  130. }
  131. static void
  132. statdb_node_apply(const char *filename, struct file_stat *filestat)
  133. {
  134. if (chown(filename, filestat->uid, filestat->gid) < 0)
  135. ohshite(_("error setting ownership of `%.255s'"), filename);
  136. if (chmod(filename, filestat->mode))
  137. ohshite(_("error setting permissions of `%.255s'"), filename);
  138. dpkg_selabel_load();
  139. dpkg_selabel_set_context(filename, filename, filestat->mode);
  140. dpkg_selabel_close();
  141. }
  142. static void
  143. statdb_node_print(FILE *out, struct filenamenode *file)
  144. {
  145. struct file_stat *filestat = file->statoverride;
  146. struct passwd *pw;
  147. struct group *gr;
  148. if (!filestat)
  149. return;
  150. pw = getpwuid(filestat->uid);
  151. if (pw)
  152. fprintf(out, "%s ", pw->pw_name);
  153. else if (filestat->uname)
  154. fprintf(out, "%s ", filestat->uname);
  155. else
  156. fprintf(out, "#%d ", filestat->uid);
  157. gr = getgrgid(filestat->gid);
  158. if (gr)
  159. fprintf(out, "%s ", gr->gr_name);
  160. else if (filestat->gname)
  161. fprintf(out, "%s ", filestat->gname);
  162. else
  163. fprintf(out, "#%d ", filestat->gid);
  164. fprintf(out, "%o %s\n", filestat->mode, file->name);
  165. }
  166. static void
  167. statdb_write(void)
  168. {
  169. char *dbname;
  170. struct atomic_file *dbfile;
  171. struct fileiterator *iter;
  172. struct filenamenode *file;
  173. dbname = dpkg_db_get_path(STATOVERRIDEFILE);
  174. dbfile = atomic_file_new(dbname, ATOMIC_FILE_BACKUP);
  175. atomic_file_open(dbfile);
  176. iter = files_db_iter_new();
  177. while ((file = files_db_iter_next(iter)))
  178. statdb_node_print(dbfile->fp, file);
  179. files_db_iter_free(iter);
  180. atomic_file_sync(dbfile);
  181. atomic_file_close(dbfile);
  182. atomic_file_commit(dbfile);
  183. atomic_file_free(dbfile);
  184. dir_sync_path(dpkg_db_get_dir());
  185. free(dbname);
  186. }
  187. static int
  188. statoverride_add(const char *const *argv)
  189. {
  190. const char *user = argv[0];
  191. const char *group = argv[1];
  192. const char *mode = argv[2];
  193. const char *path = argv[3];
  194. char *filename;
  195. struct file_stat **filestat;
  196. if (!user || !group || !mode || !path || argv[4])
  197. badusage(_("--%s needs four arguments"), cipaction->olong);
  198. if (strchr(path, '\n'))
  199. badusage(_("path may not contain newlines"));
  200. filename = path_cleanup(path);
  201. filestat = statdb_node_find(filename);
  202. if (*filestat != NULL) {
  203. if (opt_force)
  204. warning(_("an override for '%s' already exists, "
  205. "but --force specified so will be ignored"),
  206. filename);
  207. else
  208. ohshit(_("an override for '%s' already exists; "
  209. "aborting"), filename);
  210. }
  211. *filestat = statdb_node_new(user, group, mode);
  212. if (opt_update) {
  213. struct stat st;
  214. if (stat(filename, &st) == 0)
  215. statdb_node_apply(filename, *filestat);
  216. else if (opt_verbose)
  217. warning(_("--update given but %s does not exist"),
  218. filename);
  219. }
  220. statdb_write();
  221. free(filename);
  222. return 0;
  223. }
  224. static int
  225. statoverride_remove(const char *const *argv)
  226. {
  227. const char *path = argv[0];
  228. char *filename;
  229. if (!path || argv[1])
  230. badusage(_("--%s needs a single argument"), "remove");
  231. filename = path_cleanup(path);
  232. if (!statdb_node_remove(filename)) {
  233. if (opt_verbose)
  234. warning(_("no override present"));
  235. if (opt_force)
  236. return 0;
  237. else
  238. return 2;
  239. }
  240. if (opt_update && opt_verbose)
  241. warning(_("--update is useless for --remove"));
  242. statdb_write();
  243. free(filename);
  244. return 0;
  245. }
  246. static int
  247. statoverride_list(const char *const *argv)
  248. {
  249. struct fileiterator *iter;
  250. struct filenamenode *file;
  251. const char *thisarg;
  252. struct glob_node *glob_list = NULL;
  253. int ret = 1;
  254. while ((thisarg = *argv++)) {
  255. char *pattern = path_cleanup(thisarg);
  256. glob_list_prepend(&glob_list, pattern);
  257. }
  258. if (glob_list == NULL)
  259. glob_list_prepend(&glob_list, m_strdup("*"));
  260. iter = files_db_iter_new();
  261. while ((file = files_db_iter_next(iter))) {
  262. struct glob_node *g;
  263. for (g = glob_list; g; g = g->next) {
  264. if (fnmatch(g->pattern, file->name, 0) == 0) {
  265. statdb_node_print(stdout, file);
  266. ret = 0;
  267. break;
  268. }
  269. }
  270. }
  271. files_db_iter_free(iter);
  272. glob_list_free(glob_list);
  273. return ret;
  274. }
  275. static const struct cmdinfo cmdinfos[] = {
  276. ACTION("add", 0, act_install, statoverride_add),
  277. ACTION("remove", 0, act_remove, statoverride_remove),
  278. ACTION("list", 0, act_listfiles, statoverride_list),
  279. { "admindir", 0, 1, NULL, &admindir, NULL },
  280. { "quiet", 0, 0, &opt_verbose, NULL, NULL, 0 },
  281. { "force", 0, 0, &opt_force, NULL, NULL, 1 },
  282. { "update", 0, 0, &opt_update, NULL, NULL, 1 },
  283. { "help", '?', 0, NULL, NULL, usage },
  284. { "version", 0, 0, NULL, NULL, printversion },
  285. { NULL, 0 }
  286. };
  287. int
  288. main(int argc, const char *const *argv)
  289. {
  290. int ret;
  291. dpkg_locales_init(PACKAGE);
  292. dpkg_program_init("dpkg-statoverride");
  293. dpkg_options_parse(&argv, cmdinfos, printforhelp);
  294. admindir = dpkg_db_set_dir(admindir);
  295. if (!cipaction)
  296. badusage(_("need an action option"));
  297. filesdbinit();
  298. ensure_statoverrides(STATDB_PARSE_LAX);
  299. ret = cipaction->action(argv);
  300. dpkg_program_done();
  301. return ret;
  302. }