2 * dpkg - main program for package management
3 * script.c - maintainer script routines
5 * Copyright © 1995 Ian Jackson <ijackson@chiark.greenend.org.uk>
6 * Copyright © 2007-2014 Guillem Jover <guillem@debian.org>
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.
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.
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/>.
25 #include <sys/types.h>
34 #ifdef WITH_LIBSELINUX
35 #include <selinux/selinux.h>
38 #include <dpkg/i18n.h>
39 #include <dpkg/debug.h>
40 #include <dpkg/dpkg.h>
41 #include <dpkg/dpkg-db.h>
43 #include <dpkg/subproc.h>
44 #include <dpkg/command.h>
45 #include <dpkg/triglib.h>
52 post_postinst_tasks(struct pkginfo *pkg, enum pkgstatus new_status)
54 if (new_status < PKG_STAT_TRIGGERSAWAITED)
55 pkg_set_status(pkg, new_status);
56 else if (pkg->trigaw.head)
57 pkg_set_status(pkg, PKG_STAT_TRIGGERSAWAITED);
58 else if (pkg->trigpend_head)
59 pkg_set_status(pkg, PKG_STAT_TRIGGERSPENDING);
61 pkg_set_status(pkg, PKG_STAT_INSTALLED);
64 debug(dbg_triggersdetail, "post_postinst_tasks - trig_incorporate");
65 trig_incorporate(modstatdb_get_status());
69 post_script_tasks(void)
71 debug(dbg_triggersdetail, "post_script_tasks - ensure_diversions");
74 debug(dbg_triggersdetail, "post_script_tasks - trig_incorporate");
75 trig_incorporate(modstatdb_get_status());
79 cu_post_script_tasks(int argc, void **argv)
85 setexecute(const char *path, struct stat *stab)
87 if ((stab->st_mode & 0555) == 0555)
89 if (!chmod(path, 0755))
91 ohshite(_("unable to set execute permissions on '%.250s'"), path);
95 * Returns the path to the script inside the chroot.
98 maintscript_pre_exec(struct command *cmd)
100 const char *admindir = dpkg_db_get_dir();
101 const char *changedir;
102 size_t instdirlen = strlen(instdir);
104 if (instdirlen > 0 && fc_script_chrootless)
109 if (instdirlen > 0 && !fc_script_chrootless) {
110 if (strncmp(admindir, instdir, instdirlen) != 0)
111 ohshit(_("admindir must be inside instdir for dpkg to work properly"));
112 if (setenv("DPKG_ADMINDIR", admindir + instdirlen, 1) < 0)
113 ohshite(_("unable to setenv for subprocesses"));
114 if (setenv("DPKG_ROOT", "", 1) < 0)
115 ohshite(_("unable to setenv for subprocesses"));
118 ohshite(_("failed to chroot to '%.250s'"), instdir);
120 /* Switch to a known good directory to give the maintainer script
121 * a saner environment, also needed after the chroot(). */
122 if (chdir(changedir))
123 ohshite(_("failed to chdir to '%.255s'"), changedir);
124 if (debug_has_flag(dbg_scripts)) {
125 struct varbuf args = VARBUF_INIT;
126 const char **argv = cmd->argv;
129 varbuf_add_char(&args, ' ');
130 varbuf_add_str(&args, *argv);
132 varbuf_end_str(&args);
133 debug(dbg_scripts, "fork/exec %s (%s )", cmd->filename,
135 varbuf_destroy(&args);
137 if (instdirlen == 0 || fc_script_chrootless)
138 return cmd->filename;
140 assert(strlen(cmd->filename) >= instdirlen);
141 return cmd->filename + instdirlen;
145 * Set a new security execution context for the maintainer script.
147 * Try to create a new execution context based on the current one and the
148 * specific maintainer script filename. If it's the same as the current
149 * one, use the given fallback.
152 maintscript_set_exec_context(struct command *cmd, const char *fallback)
156 #ifdef WITH_LIBSELINUX
157 rc = setexecfilecon(cmd->filename, fallback);
160 return rc < 0 ? rc : 0;
164 maintscript_exec(struct pkginfo *pkg, struct pkgbin *pkgbin,
165 struct command *cmd, struct stat *stab, int warn)
170 setexecute(cmd->filename, stab);
172 push_cleanup(cu_post_script_tasks, ehflag_bombout, NULL, 0, 0);
174 pid = subproc_fork();
177 const char *maintscript_debug;
179 pkg_count = str_fmt("%d", pkgset_installed_instances(pkg->set));
181 maintscript_debug = debug_has_flag(dbg_scripts) ? "1" : "0";
183 if (setenv("DPKG_MAINTSCRIPT_PACKAGE", pkg->set->name, 1) ||
184 setenv("DPKG_MAINTSCRIPT_PACKAGE_REFCOUNT", pkg_count, 1) ||
185 setenv("DPKG_MAINTSCRIPT_ARCH", pkgbin->arch->name, 1) ||
186 setenv("DPKG_MAINTSCRIPT_NAME", cmd->argv[0], 1) ||
187 setenv("DPKG_MAINTSCRIPT_DEBUG", maintscript_debug, 1) ||
188 setenv("DPKG_RUNNING_VERSION", PACKAGE_VERSION, 1))
189 ohshite(_("unable to setenv for maintainer script"));
191 cmd->filename = cmd->argv[0] = maintscript_pre_exec(cmd);
193 if (maintscript_set_exec_context(cmd, "dpkg_script_t") < 0)
194 ohshite(_("cannot set security execution context for "
195 "maintainer script"));
199 subproc_signals_ignore(cmd->name);
200 rc = subproc_reap(pid, cmd->name, warn);
201 subproc_signals_restore();
203 pop_cleanup(ehflag_normaltidy);
209 vmaintscript_installed(struct pkginfo *pkg, const char *scriptname,
210 const char *desc, va_list args)
213 const char *scriptpath;
217 scriptpath = pkg_infodb_get_file(pkg, &pkg->installed, scriptname);
218 sprintf(buf, _("installed %s script"), desc);
220 command_init(&cmd, scriptpath, buf);
221 command_add_arg(&cmd, scriptname);
222 command_add_argv(&cmd, args);
224 if (stat(scriptpath, &stab)) {
225 command_destroy(&cmd);
226 if (errno == ENOENT) {
228 "vmaintscript_installed nonexistent %s",
232 ohshite(_("unable to stat %s '%.250s'"), buf, scriptpath);
234 maintscript_exec(pkg, &pkg->installed, &cmd, &stab, 0);
236 command_destroy(&cmd);
242 * All ...'s in maintscript_* are const char *'s.
246 maintscript_installed(struct pkginfo *pkg, const char *scriptname,
247 const char *desc, ...)
252 va_start(args, desc);
253 rc = vmaintscript_installed(pkg, scriptname, desc, args);
263 maintscript_postinst(struct pkginfo *pkg, ...)
269 rc = vmaintscript_installed(pkg, POSTINSTFILE, "post-installation", args);
279 maintscript_new(struct pkginfo *pkg, const char *scriptname,
280 const char *desc, const char *cidir, char *cidirrest, ...)
287 strcpy(cidirrest, scriptname);
288 sprintf(buf, _("new %s script"), desc);
290 va_start(args, cidirrest);
291 command_init(&cmd, cidir, buf);
292 command_add_arg(&cmd, scriptname);
293 command_add_argv(&cmd, args);
296 if (stat(cidir, &stab)) {
297 command_destroy(&cmd);
298 if (errno == ENOENT) {
300 "maintscript_new nonexistent %s '%s'",
304 ohshite(_("unable to stat %s '%.250s'"), buf, cidir);
306 maintscript_exec(pkg, &pkg->available, &cmd, &stab, 0);
308 command_destroy(&cmd);
315 maintscript_fallback(struct pkginfo *pkg,
316 const char *scriptname, const char *desc,
317 const char *cidir, char *cidirrest,
318 const char *ifok, const char *iffallback)
321 const char *oldscriptpath;
325 oldscriptpath = pkg_infodb_get_file(pkg, &pkg->installed, scriptname);
326 sprintf(buf, _("old %s script"), desc);
328 command_init(&cmd, oldscriptpath, buf);
329 command_add_args(&cmd, scriptname, ifok,
330 versiondescribe(&pkg->available.version, vdew_nonambig),
333 if (stat(oldscriptpath, &stab)) {
334 if (errno == ENOENT) {
336 "maintscript_fallback nonexistent %s '%s'",
337 scriptname, oldscriptpath);
338 command_destroy(&cmd);
341 warning(_("unable to stat %s '%.250s': %s"),
342 cmd.name, oldscriptpath, strerror(errno));
344 if (!maintscript_exec(pkg, &pkg->installed, &cmd, &stab, SUBPROC_WARN)) {
345 command_destroy(&cmd);
350 notice(_("trying script from the new package instead ..."));
352 strcpy(cidirrest, scriptname);
353 sprintf(buf, _("new %s script"), desc);
355 command_destroy(&cmd);
356 command_init(&cmd, cidir, buf);
357 command_add_args(&cmd, scriptname, iffallback,
358 versiondescribe(&pkg->installed.version, vdew_nonambig),
359 versiondescribe(&pkg->available.version, vdew_nonambig),
362 if (stat(cidir, &stab)) {
363 command_destroy(&cmd);
365 ohshit(_("there is no script in the new version of the package - giving up"));
367 ohshite(_("unable to stat %s '%.250s'"), buf, cidir);
370 maintscript_exec(pkg, &pkg->available, &cmd, &stab, 0);
371 notice(_("... it looks like that went OK"));
373 command_destroy(&cmd);