When learning the Linux command line, a popular warning to come across is that there is no “undo” command. This is especially relevant for things like deleting files using the
rm command, but has implications in all kinds of situations.
In this guide, we will discuss some ways to “undo” some changes made on the command line. There is no single strategy, so the programs and techniques involved vary depending on what exactly you are trying to guard against. We will start with some obvious ideas and move forward.
We will be implementing these on an Ubuntu 12.04 system, but most Linux distributions and versions should be capable of implementing the suggestions with little adjustment.
One of the only ways to restore a file that has been changed inadvertently (or deleted), is to have an extra copy of that file on hand. We will discuss a few ways of ensuring that you have that option.
Of course, the easiest and safest way to be able to revert changes made on your server is to run regular routine backups on your important files.
There are a large number of backup programs that are available on a Linux system. Some guides on how to install and configure backups can be found here. It is important to study the differences between the tools to find out which one best suits your needs. Equally important is regularly validating your backups to make sure they are doing what you want them to do.
Backups provide a very complete way to restore damage to your server. They can handle total data corruption or deletion as long as the copied data is kept in a remote location.
Different levels of backup include full backups (back up all data completely), differential backups (back up every file that has changed since the last full backup), and incremental backups (back up data changes within files since the last full or differential backup).
A combination of these levels are often employed in tandem with each other to completely back up files without the overhead of running full backups every time.
Often, individual files can be restored without having to restore the entire filesystem. This is especially useful if you accidentally delete or modify a file.
DigitalOcean offers a backup plan that will automatically back up your entire server regularly. You can enable this when creating your droplet by checking the box at the bottom of the page.
A strategy that is somewhat similar to backing up is version control. While not an ideal solution for backing up an entire computer, if you are trying to simply revert files to a previous state, version control may be exactly what you are looking for.
Version control systems, like
mercurial, allow you to track changes to files. This means, if you put your configuration directory, like
/etc, under version control, you can easily revert your changes if you made a change that broke something.
We have quite a few articles covering how to use git on your server.
Briefly, you can install git on Ubuntu with the following commands:
sudo apt-get update sudo apt-get install git
Once the installation is complete, you need to set up a few configuration options by typing:
<pre> git config --global user.name “<span class=“highlight”>your_name</span>” git config --global user.email “<span class=“highlight”>your_email</span>” </pre>
After this is done, change to a directory that you would like to track changes. We will use the
/etc directory in this example. Another good place to put under version control is your home directory. We can initialize a git repository by typing:
cd /etc sudo git init
You can then add all of the files in this directory (and subdirectories) by typing:
sudo git add .
Commit the changes by typing something like:
git commit -m "Initial commit"
Your files will now be under version control. As you make changes in files in this directory, you will want to re-run those last two commands (with a different message instead of “Initial commit”).
You can then revert a file to a previous state by finding the commit hash through the log:
<pre> commit 7aca1cf3b5b19c6d37b4ddc6860945e8c644cd4f Author: root root@localhost Date: Thu Jan 23 13:28:25 2014 -0500
commit <span class=“highlight”>4be26a199fd9691dc567412a470d446507885966</span> Author: root root@localhost Date: Thu Jan 23 13:20:38 2014 -0500
Then revert the file by tyipng something like:
<pre> git checkout <span class=“highlight”>commit_hash</span> – <span class=“highlight”>file_to_revert</span> </pre>
This is an easy way to revert changes that you may have made.
Keep in mind that this only works well if you are ready to regularly commit to git as you make modifications. One idea would be to set up a cron job to run this regularly.
Sometimes, you may make some changes using the
apt package manager that you would like to revert. Other times, the package manager can help you restore a package to default settings. We will discuss these situations below.
Sometimes, you install a package only to discover that it is not something that you want to keep. You can remove a package in apt by typing:
<pre> sudo apt-get remove <span class=“highlight”>package</span> </pre>
However, this will leave the configuration files intact. This is sometimes what you want, but if you are trying to completely remove the package from your system, you can use the
purge command instead, like this:
<pre> sudo apt-get purge <span class=“highlight”>package</span> </pre>
This operates in almost entirely the same way, but also removes any configuration files associated with the package. This is useful if you are sure that you don’t need the package anymore or if you haven’t made any modifications and can rely on the default configuration file.
You can uninstall any automatically installed dependencies that are no longer needed by using the autoremove apt command:
sudo apt-get autoremove --purge
Another issue that happens when installing packages with apt is that “meta-packages” can be difficult to remove correctly.
Meta-packages are packages that are simply a list of dependencies. They don’t install anything themselves, but are a list of other packages to pull in. They are also notoriously difficult to remove completely in an automatic way.
One tool that can help is the
deborphan package. Install it like this:
sudo apt-get install deborphan
After you remove a meta-package, you can run the
orphaner command to find orphans that have been left by the package uninstall. This will help you find packages that aren’t removed through regular methods.
Another way of finding stray files is through the
mlocate package. You can install it like this:
sudo apt-get install mlocate
Afterwards, you can update the index of files by issuing this command:
You can then search for the package name to see if there are additional places on the filesystem (outside of the apt indexes) where that package is referenced.
<pre> locate <span class=“highlight”>package_name</span> </pre>
You can also see the files the packages installed by a meta-package by either checking the apt logs:
sudo nano /var/lob/apt/history.log
You can use the information about the packages that were installed and manually remove them if you do not need them anymore.
Sometimes, during configuration, you change configuration files and want to revert back to the default files as packaged by the distribution.
If you want to keep the current configuration file as a backup, you can copy it out of the way by typing:
sudo mv file file.bak
The sudo in the command above is necessary if you do not have write permissions to the directory in question. Otherwise, you can omit that command.
After you have removed the file or moved it out of the way, you can reinstall the package, telling apt to check if any configuration files are missing:
<pre> sudo apt-get -o Dpkg::Options=“–force-confmiss” install --reinstall <span class=“highlight”>package_name</span> </pre>
If you do not know which package is responsible for the configuration file you need to restore, you can use the
dpkg utility to tell you:
<pre> dpkg -S <span class=“highlight”>file_name</span> </pre>
If you simply want to run through the initial package configuration steps that happen during some installations to change some values, you can issue this command:
<pre> dpkg-reconfigure <span class=“highlight”>package_name</span> </pre>
This will relaunch the configuration prompts that happened when you initially installed the program.
Another common situation happens when you modify file permissions. Sometimes, you change the permissions of a file for testing purposes or because you’ve been following some advice only to discover later that it was a bad idea.
It is possible to find out the default permissions of the file as packaged by your distribution by finding out which package owns a file. You can do that by issuing this command:
<pre> dpkg -S <span class=“highlight”>filename</span> </pre>
This will tell you the package associated with that file. For instance, if we want to find out the package owner of the
/etc/deluser.conf file, we could type:
dpkg -S /etc/deluser.conf
As you can see, it tells us that the
adduser package is responsible for that file. We can then check the
.deb file for that package by changing into the apt archive:
In this directory, you will find the
.deb files for many of the packages installed on your system. If you cannot find a file that matches the package you are using, you may have to re-download it from the repositories using this command:
<pre> sudo apt-get download <span class=“highlight”>package</span> </pre>
For instance, if there’s no
.deb for our
adduser package, we can acquire one by typing:
sudo apt-get download adduser
Once the file is in that directory, we can query the default attributes of the files it installs by typing:
<pre> dpkg -c <span class=“highlight”>file.deb</span> </pre>
adduser program, this might look something like this:
dpkg -c adduser_3.113ubuntu2_all.deb
<pre> drwxr-xr-x root/root 0 2011-10-19 18:01 ./ drwxr-xr-x root/root 0 2011-10-19 18:01 ./etc/ <span class=“highlight”>-rw-r–r-- root/root 604 2011-10-19 18:01 ./etc/deluser.conf</span> drwxr-xr-x root/root 0 2011-10-19 18:01 ./usr/ drwxr-xr-x root/root 0 2011-10-19 18:01 ./usr/sbin/ -rwxr-xr-x root/root 35120 2011-10-19 18:01 ./usr/sbin/adduser -rwxr-xr-x root/root 16511 2011-10-19 18:01 ./usr/sbin/deluser . . . </pre>
As you can see, we can verify that the default package sets the permissions to read/write access for the owner (root), and read access for all other users.
You should now have some strategies for reversing mistakes that you have made and have ideas for how to plan ahead to give yourself a contingency plan. Almost all of the concepts mentioned above make use of some sort of record of the previous state of things, whether created by you or available through your distribution’s repositories.
This should reinforce the importance of maintaining copies of files that are important to you. Whether they be data files or configuration parameters, keeping track of what your system looks like when it is in good working order will assist you in repairing things when something goes wrong.
In the comments below, post any other suggestions you have for reverting changes.
<div class=“author”>By Justin Ellingwood</div>
Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.
Join our DigitalOcean community of over a million developers for free! Get help and share knowledge in our Questions & Answers section, find tutorials and tools that will help you grow as a developer and scale your project or business, and subscribe to topics of interest.Sign up now
This textbox defaults to using Markdown to format your answer.
You can type !ref in this text area to quickly search our full set of tutorials, documentation & marketplace offerings and insert the link!
Phil! Thanks for the suggestion, I hadn’t heard of that before!
I guess the main difference is that with stand-alone git, you might have a bit more flexibility, but I really like the fact that it automatically pushes to the repository and can track changes at the permissions level. Great suggestion!
etckeeper is a good program in the same vein as your Git example…
ah this one will come in handy. good one!