Advanced Commands for Linux users¶
Advanced commands provide greater customization and controls in more specialized situations once you become familiar with basic commands.
Objectives: In this chapter, future Linux administrators will learn:
some useful commands not covered in the previous chapter.
some advanced commands.
user commands, Linux
Knowledge:
Complexity:
Reading time: 20 minutes
uniq
command¶
The uniq
command is a very powerful command, used with the sort
command, especially for log file analysis. It allows you to sort and display entries by removing duplicates.
To illustrate how the uniq
command works, let us use a firstnames.txt
file containing a list of first names:
antoine
xavier
steven
patrick
xavier
antoine
antoine
steven
Note
uniq
requires the input file to be sorted before running because it only compares consecutive lines.
With no argument, the uniq
command will not display identical lines that follow each other in the firstnames.txt
file:
$ sort firstnames.txt | uniq
antoine
patrick
steven
xavier
To display only the rows that appear only once, use the -u
option:
$ sort firstnames.txt | uniq -u
patrick
Conversely, to display only the lines that appear at least twice in the file, use the -d
option:
$ sort firstnames.txt | uniq -d
antoine
steven
xavier
To simply delete lines that appear only once, use the -D
option:
$ sort firstnames.txt | uniq -D
antoine
antoine
antoine
steven
steven
xavier
xavier
Finally, to count the number of occurrences of each line, use the -c
option:
$ sort firstnames.txt | uniq -c
3 antoine
1 patrick
2 steven
2 xavier
$ sort firstnames.txt | uniq -cd
3 antoine
2 steven
2 xavier
xargs
commands¶
The xargs
command allows the construction and execution of command lines from standard input.
The xargs
command reads whitespace or linefeed delimited arguments from standard input, and executes the command (/bin/echo
by default) one or more times using the initial arguments followed by the arguments read from standard input.
A first and simplest example would be the following:
$ xargs
use
of
xargs
<CTRL+D>
use of xargs
The xargs
command waits for an input from the standard stdin input. Three lines are entered. The end of the user input is specified to xargs
by the keystroke sequence Ctrl+D. xargs
then executes the default command echo
followed by the three arguments corresponding to the user input, namely:
$ echo "use" "of" "xargs"
use of xargs
It is possible to specify a command to be run by xargs
.
In the following example, xargs
will run the command ls -ld
on the set of folders specified in the standard input:
$ xargs ls -ld
/home
/tmp
/root
<CTRL+D>
drwxr-xr-x. 9 root root 4096 5 avril 11:10 /home
dr-xr-x---. 2 root root 4096 5 avril 15:52 /root
drwxrwxrwt. 3 root root 4096 6 avril 10:25 /tmp
In practice, the xargs
command executed the ls -ld /home /tmp /root
command.
What happens if the command to be executed does not accept multiple arguments, such as with the find
command?
$ xargs find /var/log -name
*.old
*.log
find: paths must precede expression: *.log
The xargs
command attempted to execute the find
command with multiple arguments behind the -name
option, which caused find
to generate an error:
$ find /var/log -name "*.old" "*.log"
find: paths must precede expression: *.log
In this case, the xargs
command must be forced to execute the find
command several times (once per line entered as standard input). The -L
option followed by an integer allows you to specify the maximum number of entries to be processed with the command at one time:
$ xargs -L 1 find /var/log -name
*.old
/var/log/dmesg.old
*.log
/var/log/boot.log
/var/log/anaconda.yum.log
/var/log/anaconda.storage.log
/var/log/anaconda.log
/var/log/yum.log
/var/log/audit/audit.log
/var/log/anaconda.ifcfg.log
/var/log/dracut.log
/var/log/anaconda.program.log
<CTRL+D>
To specify both arguments on the same line, use the -n 1
option:
$ xargs -n 1 find /var/log -name
*.old *.log
/var/log/dmesg.old
/var/log/boot.log
/var/log/anaconda.yum.log
/var/log/anaconda.storage.log
/var/log/anaconda.log
/var/log/yum.log
/var/log/audit/audit.log
/var/log/anaconda.ifcfg.log
/var/log/dracut.log
/var/log/anaconda.program.log
<CTRL+D>
Case study of a backup with a tar
based on a search:
$ find /var/log/ -name "*.log" -mtime -1 | xargs tar cvfP /root/log.tar
$ tar tvfP /root/log.tar
-rw-r--r-- root/root 1720 2017-04-05 15:43 /var/log/boot.log
-rw-r--r-- root/root 499270 2017-04-06 11:01 /var/log/audit/audit.log
The special feature of the xargs
command is that it places the input argument at the end of the called command. This works very well with the above example since the files passed in will form the list of files to be added to the archive.
Using the example of the cp
command, to copy a list of files in a directory, this list of files will be added at the end of the command... but what the cp
command expects at the end of the command is the destination. To do this, use the -I
option to put the input arguments somewhere else than at the end of the line.
find /var/log -type f -name "*.log" | xargs -I % cp % /root/backup
The -I
option allows you to specify a character (the %
character in the above example) where the input files to xargs
will be placed.
yum-utils
package¶
The yum-utils
package is a collection of utilities, built for yum
by various authors, which make it easier and more powerful to use.
Note
While yum
has been replaced by dnf
in Rocky Linux 8, the package name has remained yum-utils
, although it can be installed as dnf-utils
as well. These are classic YUM utilities implemented as CLI shims on top of DNF to maintain backwards compatibility with yum-3
.
Here are some examples of these utilities.
repoquery
command¶
The repoquery
command is used to query the packages in the repository.
Examples of use:
- Display the dependencies of a package (it can be a software package that has been installed or not installed), equivalent to
dnf deplist <package-name>
repoquery --requires <package-name>
- Display the files provided by an installed package (does not work for packages that are not installed), equivalent to
rpm -ql <package-name>
$ repoquery -l yum-utils
/etc/bash_completion.d
/etc/bash_completion.d/yum-utils.bash
/usr/bin/debuginfo-install
/usr/bin/find-repos-of-install
/usr/bin/needs-restarting
/usr/bin/package-cleanup
/usr/bin/repo-graph
/usr/bin/repo-rss
/usr/bin/repoclosure
/usr/bin/repodiff
/usr/bin/repomanage
/usr/bin/repoquery
/usr/bin/reposync
/usr/bin/repotrack
/usr/bin/show-changed-rco
/usr/bin/show-installed
/usr/bin/verifytree
/usr/bin/yum-builddep
/usr/bin/yum-config-manager
/usr/bin/yum-debug-dump
/usr/bin/yum-debug-restore
/usr/bin/yum-groups-manager
/usr/bin/yumdownloader
…
yumdownloader
command¶
The yumdownloader
command downloads RPM packages from the repositories. Equivalent to dnf download --downloadonly --downloaddir ./ package-name
Note
This command is very useful to quickly build a local repository of a few rpms!
Example: yumdownloader
will download the samba rpm package and all its dependencies:
$ yumdownloader --destdir /var/tmp --resolve samba
or
$ dnf download --downloadonly --downloaddir /var/tmp --resolve samba
Options | Comments |
---|---|
--destdir | The downloaded packages will be stored in the specified folder. |
--resolve | Also downloads the package dependencies. |
psmisc
packages¶
The psmisc
package contains utilities for managing system processes:
pstree
: thepstree
command displays the current processes on the system in a tree-like structure.killall
: thekillall
command sends a kill signal to all processes identified by name.fuser
: thefuser
command identifies thePID
of processes that use the specified files or file systems.
Examples:
$ pstree
systemd─┬─NetworkManager───2*[{NetworkManager}]
├─agetty
├─auditd───{auditd}
├─crond
├─dbus-daemon───{dbus-daemon}
├─firewalld───{firewalld}
├─lvmetad
├─master─┬─pickup
│ └─qmgr
├─polkitd───5*[{polkitd}]
├─rsyslogd───2*[{rsyslogd}]
├─sshd───sshd───bash───pstree
├─systemd-journal
├─systemd-logind
├─systemd-udevd
└─tuned───4*[{tuned}]
# killall httpd
Kill processes (option -k
) that access the /etc/httpd/conf/httpd.conf
file:
# fuser -k /etc/httpd/conf/httpd.conf
watch
command¶
The watch
command regularly executes a command and displays the result in the terminal in full screen.
The -n
option allows you to specify the number of seconds between each execution of the command.
Note
To exit the watch
command, you must type the keys: Ctrl+C to kill the process.
Examples:
- Display the end of the
/etc/passwd
file every 5 seconds:
watch -n 5 tail -n 3 /etc/passwd
Result:
Every 5.0s: tail -n 3 /etc/passwd rockstar.rockylinux.lan: Thu Jul 1 15:43:59 2021
sssd:x:996:993:User for sssd:/:/sbin/nologin
chrony:x:995:992::/var/lib/chrony:/sbin/nologin
sshd:x:74:74:Privilege-separated SSH:/var/empty/sshd:/sbin/nologin
- Monitoring the number of files in a folder:
watch -n 1 'ls -l | wc -l'
- Display a clock:
watch -t -n 1 date
install
command¶
Contrary to what its name might suggest, the install
command is not used to install new packages.
This command combines file copying (cp
) and directory creation (mkdir
), with rights management (chmod
, chown
) and other useful functionalities (like backups).
install source dest
install -t directory source [...]
install -d directory
Options:
Options | Remarks |
---|---|
-b or --backup[=suffix] | creates a backup of destination file |
-d | treats arguments as directory names |
-D | creates all leading components before copying SOURCE to DEST |
-g and -o | sets ownership |
-m | sets permissions |
-p | preserves the timestamps of the sources files |
-t | copies all source arguments to the directory |
Note
There are options for managing the SELinux context (see the manual page).
Examples:
Create a directory with the -d
option:
install -d ~/samples
Copy a file from a source location to a directory:
install src/sample.txt ~/samples/
These two orders could have been carried out with a single command:
$ install -v -D -t ~/samples/ src/sample.txt
install: creating directory '~/samples'
'src/sample.txt' -> '~/samples/sample.txt'
This command already saves time. Combine it with owner, owner group, and rights management to improve the time savings:
sudo install -v -o rocky -g users -m 644 -D -t ~/samples/ src/sample.txt
Note
sudo
is required in this case to make property changes.
You can also create a backup of existing files thanks to the -b
option:
$ install -v -b -D -t ~/samples/ src/sample.txt
'src/sample.txt' -> '~/samples/sample.txt' (archive: '~/samples/sample.txt~')
As you can see, the install
command creates a backup file with a ~
tilde appended to the original file name.
The suffix can be specified thanks to the -S
option:
$ install -v -b -S ".bak" -D -t ~/samples/ src/sample.txt
'src/sample.txt' -> '~/samples/sample.txt' (archive: '~/samples/sample.txt.bak')
tree
command¶
Expand the files or directories in the directory in a tree-like manner.
options | description |
---|---|
-a | All files are listed |
-h | Prints the size in a more human-readable way |
-u | Displays file owner or UID number |
-g | Displays file group owner or GID number |
-p | Print the protections for each file |
For example:
$ tree -hugp /etc/yum.repos.d/
/etc/yum.repos.d/
├── [-rw-r--r-- root root 1.6K] epel-modular.repo
├── [-rw-r--r-- root root 1.3K] epel.repo
├── [-rw-r--r-- root root 1.7K] epel-testing-modular.repo
├── [-rw-r--r-- root root 1.4K] epel-testing.repo
├── [-rw-r--r-- root root 710] Rocky-AppStream.repo
├── [-rw-r--r-- root root 695] Rocky-BaseOS.repo
├── [-rw-r--r-- root root 1.7K] Rocky-Debuginfo.repo
├── [-rw-r--r-- root root 360] Rocky-Devel.repo
├── [-rw-r--r-- root root 695] Rocky-Extras.repo
├── [-rw-r--r-- root root 731] Rocky-HighAvailability.repo
├── [-rw-r--r-- root root 680] Rocky-Media.repo
├── [-rw-r--r-- root root 680] Rocky-NFV.repo
├── [-rw-r--r-- root root 690] Rocky-Plus.repo
├── [-rw-r--r-- root root 715] Rocky-PowerTools.repo
├── [-rw-r--r-- root root 746] Rocky-ResilientStorage.repo
├── [-rw-r--r-- root root 681] Rocky-RT.repo
└── [-rw-r--r-- root root 2.3K] Rocky-Sources.repo
0 directories, 17 files
stat
command¶
The stat
command displays the status of a file or file system.
$ stat /root/anaconda-ks.cfg
File: /root/anaconda-ks.cfg
Size: 1352 Blocks: 8 IO Block: 4096 regular file
Device: 10302h/66306d Inode: 2757097 Links: 1
Access: (0755/-rwxr-xr-x) Uid: ( 0/ root) Gid: ( 0/ root)
Access: 2024-01-20 13:04:57.012033583 +0800
Modify: 2023-09-25 14:04:48.524760784 +0800
Change: 2024-01-24 16:37:34.315995221 +0800
Birth: 2
File
- Displays the path location of the file.Size
- Displays the file size in bytes. If this is a directory, it displays the fixed 4096 bytes occupied by the directory name.Blocks
- Displays the number of allocated blocks. Attention, please! The size of each block in this command is 512 bytes. The default size of each block inls -ls
is 1024 bytes.Device
- Device number in decimal or hexadecimal notation.Inode
- Inode is a unique ID number the Linux kernel assigns to a file or directory.Links
- Number of hard links. Hard links are sometimes referred to as physical links.Access
- The last access time of files and directories, i.e.atime
in GNU/Linux.Modify
- The last modification time of files and directories, i.e.mtime
in GNU/Linux.Change
- The last time the property is changed, i.e.ctime
in GNU/Linux.Birth
- Birth time (Creation time). In some documents, it is abbreviated asbtime
orcrtime
. You need a file system and kernel version higher than a certain version to display the creation time.
For files:
atime - After accessing the file content using commands such as cat
, less
, more
, and head
, the atime
of the file can be updated. Please pay attention! The atime
of the file is not updated in real-time, and for performance reasons, it needs to wait for a period of time before it can be displayed.
mtime - Modifying the file content can update the mtime
of the file (such as appending or overwriting the file content through redirection), because the file size is a property of the file, the ctime
will also be updated simultaneously.
ctime - Changing the owner, group, permissions, file size, and links (soft and hard links) of the file will update ctime.
For directories:
atime - After using the cd
command to enter a new directory that has never been accessed before, you can update and fix the atime
of that directory.
mtime - Performing operations such as creating, deleting, and renaming files in this directory will update the mtime
and ctime
of the directory.
ctime - When the permissions, owner, group, etc. of a directory change, the ctime
of the directory will be updated.
Tip
- If you create a new file or directory, its
atime
,mtime
, andctime
are exactly the same - If the file content is modified, the
mtime
andctime
of the file will inevitably be updated. - If a brand new file is created in the directory, the
atime
,ctime
, andmtime
of that directory will be updated simultaneously. - If the
mtime
of a directory is updated, then thectime
of that directory must be updated.