script.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. /*
  2. * dpkg - main program for package management
  3. * script.c - maintainer script routines
  4. *
  5. * Copyright © 1995 Ian Jackson <ijackson@chiark.greenend.org.uk>
  6. * Copyright © 2007-2014 Guillem Jover <guillem@debian.org>
  7. *
  8. * This is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  20. */
  21. #include <config.h>
  22. #include <compat.h>
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25. #include <assert.h>
  26. #include <errno.h>
  27. #include <string.h>
  28. #include <unistd.h>
  29. #include <stdlib.h>
  30. #ifdef WITH_SELINUX
  31. #include <selinux/selinux.h>
  32. #endif
  33. #include <dpkg/i18n.h>
  34. #include <dpkg/debug.h>
  35. #include <dpkg/dpkg.h>
  36. #include <dpkg/dpkg-db.h>
  37. #include <dpkg/pkg.h>
  38. #include <dpkg/subproc.h>
  39. #include <dpkg/command.h>
  40. #include <dpkg/triglib.h>
  41. #include "filesdb.h"
  42. #include "infodb.h"
  43. #include "main.h"
  44. void
  45. post_postinst_tasks(struct pkginfo *pkg, enum pkgstatus new_status)
  46. {
  47. if (new_status < PKG_STAT_TRIGGERSAWAITED)
  48. pkg_set_status(pkg, new_status);
  49. else if (pkg->trigaw.head)
  50. pkg_set_status(pkg, PKG_STAT_TRIGGERSAWAITED);
  51. else if (pkg->trigpend_head)
  52. pkg_set_status(pkg, PKG_STAT_TRIGGERSPENDING);
  53. else
  54. pkg_set_status(pkg, PKG_STAT_INSTALLED);
  55. modstatdb_note(pkg);
  56. debug(dbg_triggersdetail, "post_postinst_tasks - trig_incorporate");
  57. trig_incorporate(modstatdb_get_status());
  58. }
  59. static void
  60. post_script_tasks(void)
  61. {
  62. debug(dbg_triggersdetail, "post_script_tasks - ensure_diversions");
  63. ensure_diversions();
  64. debug(dbg_triggersdetail, "post_script_tasks - trig_incorporate");
  65. trig_incorporate(modstatdb_get_status());
  66. }
  67. static void
  68. cu_post_script_tasks(int argc, void **argv)
  69. {
  70. post_script_tasks();
  71. }
  72. static void
  73. setexecute(const char *path, struct stat *stab)
  74. {
  75. if ((stab->st_mode & 0555) == 0555)
  76. return;
  77. if (!chmod(path, 0755))
  78. return;
  79. ohshite(_("unable to set execute permissions on '%.250s'"), path);
  80. }
  81. /**
  82. * Returns the path to the script inside the chroot.
  83. */
  84. static const char *
  85. maintscript_pre_exec(struct command *cmd)
  86. {
  87. const char *admindir = dpkg_db_get_dir();
  88. size_t instdirl = strlen(instdir);
  89. if (*instdir) {
  90. if (strncmp(admindir, instdir, instdirl) != 0)
  91. ohshit(_("admindir must be inside instdir for dpkg to work properly"));
  92. if (setenv("DPKG_ADMINDIR", admindir + instdirl, 1) < 0)
  93. ohshite(_("unable to setenv for subprocesses"));
  94. if (chroot(instdir))
  95. ohshite(_("failed to chroot to '%.250s'"), instdir);
  96. }
  97. /* Switch to a known good directory to give the maintainer script
  98. * a saner environment, also needed after the chroot(). */
  99. if (chdir("/"))
  100. ohshite(_("failed to chdir to '%.255s'"), "/");
  101. if (debug_has_flag(dbg_scripts)) {
  102. struct varbuf args = VARBUF_INIT;
  103. const char **argv = cmd->argv;
  104. while (*++argv) {
  105. varbuf_add_char(&args, ' ');
  106. varbuf_add_str(&args, *argv);
  107. }
  108. varbuf_end_str(&args);
  109. debug(dbg_scripts, "fork/exec %s (%s )", cmd->filename,
  110. args.buf);
  111. varbuf_destroy(&args);
  112. }
  113. if (!instdirl)
  114. return cmd->filename;
  115. assert(strlen(cmd->filename) >= instdirl);
  116. return cmd->filename + instdirl;
  117. }
  118. /**
  119. * Set a new security execution context for the maintainer script.
  120. *
  121. * Try to create a new execution context based on the current one and the
  122. * specific maintainer script filename. If it's the same as the current
  123. * one, use the given fallback.
  124. */
  125. static int
  126. maintscript_set_exec_context(struct command *cmd, const char *fallback)
  127. {
  128. int rc = 0;
  129. #ifdef WITH_SELINUX
  130. rc = setexecfilecon(cmd->filename, fallback);
  131. #endif
  132. return rc < 0 ? rc : 0;
  133. }
  134. static int
  135. maintscript_exec(struct pkginfo *pkg, struct pkgbin *pkgbin,
  136. struct command *cmd, struct stat *stab, int warn)
  137. {
  138. pid_t pid;
  139. int rc;
  140. setexecute(cmd->filename, stab);
  141. push_cleanup(cu_post_script_tasks, ehflag_bombout, NULL, 0, 0);
  142. pid = subproc_fork();
  143. if (pid == 0) {
  144. char *pkg_count;
  145. const char *maintscript_debug;
  146. pkg_count = str_fmt("%d", pkgset_installed_instances(pkg->set));
  147. maintscript_debug = debug_has_flag(dbg_scripts) ? "1" : "0";
  148. if (setenv("DPKG_MAINTSCRIPT_PACKAGE", pkg->set->name, 1) ||
  149. setenv("DPKG_MAINTSCRIPT_PACKAGE_REFCOUNT", pkg_count, 1) ||
  150. setenv("DPKG_MAINTSCRIPT_ARCH", pkgbin->arch->name, 1) ||
  151. setenv("DPKG_MAINTSCRIPT_NAME", cmd->argv[0], 1) ||
  152. setenv("DPKG_MAINTSCRIPT_DEBUG", maintscript_debug, 1) ||
  153. setenv("DPKG_RUNNING_VERSION", PACKAGE_VERSION, 1))
  154. ohshite(_("unable to setenv for maintainer script"));
  155. cmd->filename = cmd->argv[0] = maintscript_pre_exec(cmd);
  156. if (maintscript_set_exec_context(cmd, "dpkg_script_t") < 0)
  157. ohshite(_("cannot set security execution context for "
  158. "maintainer script"));
  159. command_exec(cmd);
  160. }
  161. subproc_signals_ignore(cmd->name);
  162. rc = subproc_reap(pid, cmd->name, warn);
  163. subproc_signals_restore();
  164. pop_cleanup(ehflag_normaltidy);
  165. return rc;
  166. }
  167. static int
  168. vmaintscript_installed(struct pkginfo *pkg, const char *scriptname,
  169. const char *desc, va_list args)
  170. {
  171. struct command cmd;
  172. const char *scriptpath;
  173. struct stat stab;
  174. char buf[100];
  175. scriptpath = pkg_infodb_get_file(pkg, &pkg->installed, scriptname);
  176. sprintf(buf, _("installed %s script"), desc);
  177. command_init(&cmd, scriptpath, buf);
  178. command_add_arg(&cmd, scriptname);
  179. command_add_argv(&cmd, args);
  180. if (stat(scriptpath, &stab)) {
  181. command_destroy(&cmd);
  182. if (errno == ENOENT) {
  183. debug(dbg_scripts,
  184. "vmaintscript_installed nonexistent %s",
  185. scriptname);
  186. return 0;
  187. }
  188. ohshite(_("unable to stat %s '%.250s'"), buf, scriptpath);
  189. }
  190. maintscript_exec(pkg, &pkg->installed, &cmd, &stab, 0);
  191. command_destroy(&cmd);
  192. return 1;
  193. }
  194. /*
  195. * All ...'s in maintscript_* are const char *'s.
  196. */
  197. int
  198. maintscript_installed(struct pkginfo *pkg, const char *scriptname,
  199. const char *desc, ...)
  200. {
  201. va_list args;
  202. int rc;
  203. va_start(args, desc);
  204. rc = vmaintscript_installed(pkg, scriptname, desc, args);
  205. va_end(args);
  206. if (rc)
  207. post_script_tasks();
  208. return rc;
  209. }
  210. int
  211. maintscript_postinst(struct pkginfo *pkg, ...)
  212. {
  213. va_list args;
  214. int rc;
  215. va_start(args, pkg);
  216. rc = vmaintscript_installed(pkg, POSTINSTFILE, "post-installation", args);
  217. va_end(args);
  218. if (rc)
  219. ensure_diversions();
  220. return rc;
  221. }
  222. int
  223. maintscript_new(struct pkginfo *pkg, const char *scriptname,
  224. const char *desc, const char *cidir, char *cidirrest, ...)
  225. {
  226. struct command cmd;
  227. struct stat stab;
  228. va_list args;
  229. char buf[100];
  230. strcpy(cidirrest, scriptname);
  231. sprintf(buf, _("new %s script"), desc);
  232. va_start(args, cidirrest);
  233. command_init(&cmd, cidir, buf);
  234. command_add_arg(&cmd, scriptname);
  235. command_add_argv(&cmd, args);
  236. va_end(args);
  237. if (stat(cidir, &stab)) {
  238. command_destroy(&cmd);
  239. if (errno == ENOENT) {
  240. debug(dbg_scripts,
  241. "maintscript_new nonexistent %s '%s'",
  242. scriptname, cidir);
  243. return 0;
  244. }
  245. ohshite(_("unable to stat %s '%.250s'"), buf, cidir);
  246. }
  247. maintscript_exec(pkg, &pkg->available, &cmd, &stab, 0);
  248. command_destroy(&cmd);
  249. post_script_tasks();
  250. return 1;
  251. }
  252. int
  253. maintscript_fallback(struct pkginfo *pkg,
  254. const char *scriptname, const char *desc,
  255. const char *cidir, char *cidirrest,
  256. const char *ifok, const char *iffallback)
  257. {
  258. struct command cmd;
  259. const char *oldscriptpath;
  260. struct stat stab;
  261. char buf[100];
  262. oldscriptpath = pkg_infodb_get_file(pkg, &pkg->installed, scriptname);
  263. sprintf(buf, _("old %s script"), desc);
  264. command_init(&cmd, oldscriptpath, buf);
  265. command_add_args(&cmd, scriptname, ifok,
  266. versiondescribe(&pkg->available.version, vdew_nonambig),
  267. NULL);
  268. if (stat(oldscriptpath, &stab)) {
  269. if (errno == ENOENT) {
  270. debug(dbg_scripts,
  271. "maintscript_fallback nonexistent %s '%s'",
  272. scriptname, oldscriptpath);
  273. command_destroy(&cmd);
  274. return 0;
  275. }
  276. warning(_("unable to stat %s '%.250s': %s"),
  277. cmd.name, oldscriptpath, strerror(errno));
  278. } else {
  279. if (!maintscript_exec(pkg, &pkg->installed, &cmd, &stab, SUBPROC_WARN)) {
  280. command_destroy(&cmd);
  281. post_script_tasks();
  282. return 1;
  283. }
  284. }
  285. notice(_("trying script from the new package instead ..."));
  286. strcpy(cidirrest, scriptname);
  287. sprintf(buf, _("new %s script"), desc);
  288. command_destroy(&cmd);
  289. command_init(&cmd, cidir, buf);
  290. command_add_args(&cmd, scriptname, iffallback,
  291. versiondescribe(&pkg->installed.version, vdew_nonambig),
  292. versiondescribe(&pkg->available.version, vdew_nonambig),
  293. NULL);
  294. if (stat(cidir, &stab)) {
  295. command_destroy(&cmd);
  296. if (errno == ENOENT)
  297. ohshit(_("there is no script in the new version of the package - giving up"));
  298. else
  299. ohshite(_("unable to stat %s '%.250s'"), buf, cidir);
  300. }
  301. maintscript_exec(pkg, &pkg->available, &cmd, &stab, 0);
  302. notice(_("... it looks like that went OK"));
  303. command_destroy(&cmd);
  304. post_script_tasks();
  305. return 1;
  306. }