You pretty much have to do it through a SystemD unit nowadays.  I’ve found the backwards-compatibility bits to be mildly unreliable, to put it, er, mildly…

Repeating stuff you probably already know, but saying it out loud for everyone’s benefit.

 

You need two pieces – a script, and a systemd unit file.

The unit file goes in /etc/systemd/system/ and looks like any other unit, pointing to a script that from systemd’s POV is just another executable.  For example:

 

/etc/systemd/system/last-gasp.service

[Unit]

Description=script that does stuff

DefaultDependencies=no

Before=shutdown.target

 

[Service]

Type=oneshot

ExecStart=/some/where/lastgasp.sh

TimeoutStartSec=0

 

[Install]

WantedBy=shutdown.target

(Unit file syntax reference docs: systemd.service (www.freedesktop.org) and systemd.unit (www.freedesktop.org))

 

Make sure the script is executable or this will fail (more or less) silently.

 

Run “systemctl daemon-reload” to make it notice your new unit file.

Run “systemctl enable last-gasp” to ask systemd to, y’know, actually *do* something with it at shutdown.

 

Test with “reboot”.  If the script takes long enough to run (hint: add “sleep 30” to make it run long enough!) you’ll see systemd print something on screen about waiting for the task to complete.

I think the /etc/systemd/system/ path is correct on pretty much any Linux, but the proof is in the pudding – if the “enable” step works, then the unit file is in the correct location, or at least a correct-enough location.

 

Note that systemd may have unmounted the filesystem where your script is located before it tries to run the script; put the script somewhere on the root filesystem to work around this, if needed.

In theory you can add “RequiresMountsFor=/some/where” to the [Unit] section of the unit to ensure the filesystem doesn’t get unmounted until the script exits – never tried it.

Also, even if the filesystem is still mounted, it may be mounted read-only at this point, so you may not be able to write anything to disk.  This appears to be undefined, I can’t find anything that documents whether RequiresMountsFor leaves you with read-write or read-only mounts.

 

IN THEORY you can also just drop your script into /usr/lib/systemd/system-shutdown/ and magic will happen, but… dunno, I’ve never tried that.  Let us know if it actually works?  (See systemd-poweroff.service (www.freedesktop.org).)  This approach likely happens too late in the process to be useful, but read the docs and assess for yourself.

-Adam

 

From: Roundtable <roundtable-bounces@muug.ca> On Behalf Of vsankar@foretell.ca
Sent: Saturday, February 4, 2023 12:40 PM
To: roundtable@muug.ca
Subject: [RndTbl] How can one execute a script just before shutdown on Ubuntu

 

Sorry to bother you all with this question — can you please point me to a helpful document or give me some instructions on how to execute a script just before shutting down an Ubuntu system (Linux vijay-iMac 5.15.0-58-generic #64-Ubuntu SMP Thu Jan 5 11:43:13 UTC 2023 x86_64 x86_64 x86_64 GNU/Linux)?

 

I created a simple script 

 

-rwxr-xr-x 1 root root 51 Jan 30 11:46 /etc/init.d/before-shutdown

 

Then did a link so that I had

 

lrwxrwxrwx 1 root root 25 Jan 30 11:47 K04before-shutdown -> ../init.d/before-shutdown

 

Nothing happened!! So I wasted a lot of time reading up on systemd etc., and set up unit files to execute before shutdown.target, network.target, power off.target, as well as halt.target. Since these did not work either and life is too short, I went back to what I thought was the old way of doing things and did a 

 

lrwxrwxrwx 1 root root 25 Feb  4 11:58 /etc/rc6.d/K99before-shutdown -> ../init.d/before-shutdown

 

This was a complete fail as well. 

 

Would really appreciate any suggestions on how to make this work. 

 

Thanks very much,

 

Vijay

 

 

Vijay Sankar

ForeTell Technologies Limited
vsankar@foretell.ca