综合实践样例

根据Linux笔记中已经学习的命令,综合运用于实际实践中。

04 执行任务并定时关闭

首行是标记使用的脚本的类型,此处为bash脚本。
两个命令间用&连接,前面的命令执行后挂在后台并立即开始执行后面的命令。用&&连接,只有前面的命令执行完毕并退出后才会执行后面的命令。
第一行开启roscore后,计时两秒后结束roscorekill -2相当于Ctrl+C-2就是Ctrl+C发出的siginit

#!/bin/bash
roscore &
sleep 3 && kill -2

05 解压文件名有规律的系列文件

这一系列的文件名为“17.7z”、“18.7z”…“21.7z”。解压密码为“blue”。

#!/bin/bash

set -u

for i in {17..21}
do
  filename=$i".7z"
  7z x $filename -pblue
done

解压所在文件夹内的所有以“.7z”结尾的文件,解压密码为“blue”。

ls | while read line
do
  file=$line
  if echo $file | grep -q -E '\.7z$'
  then
    echo $file
    7z x $file -pblue
  fi
done

07 修改某文件夹下的特定文件

遍历所在指定文件夹及其所有子文件夹中的markdown文档,并把第四行修改为<日期+时间+时区>

#!/bin/bash

files=$(find "." -type f)

# 遍历文件
for file in $files; do
    # 检查文件名是否符合特定首尾条件
    if [[ "$file" == *_posts* && "$file" == *.md ]]; then
        # 获取文件的最近修改时间
        last_modified=$(stat -c "%Y" "$file")

        # 将时间戳转换为日月年时分秒的格式
        last_modified_formatted=$(date -d @"$last_modified" +"%Y-%m-%d %H:%M:%S %z")

        # 替换文件的第四行为最近修改时间
        sed -i "4s/.*/date:   $last_modified_formatted/" "$file"

        echo "Updated date in $file to $last_modified_formatted"
    fi
done

09 批量删除某个名称的文件夹

find . -type d -name .git -prune -exec rm -r {} \;
  • .: 指定查找的范围,即在呢个文件夹内查找
  • -name: find命令的参数,后面接要查找的文件夹名称
  • .git: 我们想要删除的文件夹的名称,此处我想删除的文件夹为“.git”
  • -exec:
  • {}: can be read as “for each matching file/ folder”
  • \; is a terminator for the -exec clause

下面的这个命令也可以使用。

rm -rf `find . -type d -name a`

11 统计文件名

ls | while read line
do
  file=$line
  if echo $file | grep -q -E '\.bag$'
  then
    if [ 18 = `echo ${#file}` ] 
      then
      # time=`rosbag info $file | grep start`
      # str=`date -d @${time: 0-14: 13} "+%Y-%m-%d %H:%M:%S"`
      time=${file: 0: 14}
      echo $time >> date_t.txt
    fi
  fi
done

13 query ros topics

topic_name[0]="/image"
topic_name[1]="/lidar"
topic_name[2]="/rtk"

for topic in ${topic_name[@]};
do
  echo $topic
  timeout 3 rostopic hz $topic
  echo "----------------------"
done

14 bind several processes via service

We can create several peocesses and kill them in one time via service. Here we create a user service, which is different from system service.
First, # create a folder, which will contain service file and shell script. Assume username is “bs”.

mkdir -p /home/bs/.config/systemd/user/ 
cd /home/bs/.config/systemd/user/

Second, create a service file named myservice.service like below.

[Unit]
Description=Discribe your service here
# This service starts your custom script after the network is available. If this service id fully independent and does not rely on the state of any other services, you can delete this line
After=network.target

[Service]
Type=simple
# Use `ExecStart` to call a wrapper script to ensure that the script handles the termination of all its child processes upon receiving a signal to stop.
# shell script, it must be absolute path
ExecStart=/home/bs/.config/systemd/user/shell.sh
# ensures that all processes started by the service (including any child processes started by the script) will be terminated when the service itself is stopped.
KillMode=control-group
# on-failure: restart this service if this service dies; no: Don't restart the service automatically
Restart=on-failure

[Install]
WantedBy=default.target
# This service will be started at the default runlevel

Third, reload the user-level systemd configuration.

systemctl --user daemon-reload

Fourth, enable the service.

systemctl --user enable myservice.service

Fifth, don’t forget to create shell script. Now let’s create a script named shell.sh like below. Here I take ros as example.

#!/bin/bash

# Function to terminate all roslaunch processes
cleanup() {
    echo "Stopping all ROS nodes..."
    kill -SIGINT "$roslaunch1_pid" "$roslaunch2_pid" "$roslaunch3_pid"
}

# The trap command in the script sets up a handler for SIGTERM, which is the signal sent by systemd when the service is stopped. This handler function, cleanup, will explicitly kill all the background processes.
trap cleanup SIGTERM

# Start roslaunch commands in background
sleep 1 && roslaunch your_package launch_file1.launch & roslaunch1_pid=$!
sleep 2 && roslaunch your_package launch_file2.launch & roslaunch2_pid=$!
sleep 3 && roslaunch your_package launch_file3.launch & roslaunch3_pid=$!

# The wait command makes the main script wait for all the background processes, which means the main service process remains active as long as the background processes are running.
wait $roslaunch1_pid $roslaunch2_pid $roslaunch3_pid

We have completed service and shell file. We can use this service now.

systemctl --user start myservice.service # start service
systemctl --user status myservice.service # check service
systemctl --user restart myservice.service # restart service
systemctl --user stop myservice.service # stop service

Note that if you make some changes to service file, don’t forget to run this cmd systemctl --user daemon-reload to make it effect.

15 update date from local ntp server

create ntp server
Assume that ip of ntp server is 192.168.1.100 and host name is “bs”. Run cmds below:

sudo apt install ntp # install ntp
sudo systemctl start ntp # start ntp

set client
Run cmd below on client that need update date and time.

sudo nano /etc/systemd/timesyncd.conf # you can also use vim or gedit

Edit like below

[Time]
NTP=bs 192.168.1.100
FallbackNTP=bs 192.168.1.100

Then save and close the file. Restart the timesyncd service with cmd below:

sudo systemctl restart systemd-timesyncd

Check that the NTP synchronization is working correctly with:

timedatectl status

If everything is going well, you will see System lock synchronized is yes. You can also run date and check whether the result is right.

16 Creating a Script to Run at Boot with Cron

  1. Prepare Your Script: Ensure your script in /usr/local/func/ is executable. If it’s not, make it executable:
sudo chmod +x /usr/local/func/myscript
  1. Edit the Crontab:
crontab -e

Add the following line to schedule your script to run at every system boot:

@reboot /usr/local/func/myscript

This line tells cron to run /usr/local/func/myscript every time the system boots up.

  1. Save and Exit: After adding the line, save and exit the editor. cron will automatically install the new crontab.

  2. Verify: To ensure your crontab is set up correctly, you can list your current user’s crontab entries:

crontab -l

This command will show all scheduled cron jobs, including your new @reboot job.

  1. Additional Considerations Environment: Cron jobs run with a limited environment, meaning many of the environmental variables available in a full shell session (like those started by logging in) may not be available. If your script relies on certain environment variables, you might need to define them in the script or in the crontab entry. Or you can run source /home/${user}/.bahsrc to make script run like in a terminal.

Logging: Since cron jobs don’t have a terminal, you might want to redirect your script’s output to a file for debugging:

@reboot /usr/local/func/myscript >> /var/log/myscript.log 2>&1

This setup ensures that your script runs at system boot without involving system-wide services or requiring systemd, fitting your preference for a simpler, less formal method.

17 split and convert string

#!/bin/bash

# colorful output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[0;33m'
BLUE='\033[0;34m'
NC='\033[0m'

if [ "$#" -eq 0 ]; then
  echo -e "${RED}Should contain project name.${NC}"
  exit
fi

# (absolute path (relative path))
SHELL_DIR=$(realpath $(dirname "${BASH_SOURCE[0]}"))

name=$1

# whether file exists
# if [ -e "$name" ]; then
#   echo -e "${BLUE}File ${name} exists, skip generatation.${NC}"
#   exit
# fi

# if [ -d "$name" ]; then
#     echo -e "${YELLOW}Directory ${name} exists, skip creation.${NC}"
#     exit
# fi

# Internal Field Separator
IFS='_' 
# Convert string into an array
read -ra PARTS <<< "$name"
# Reset IFS if needed
unset IFS

# STRING="part1_part2_part3"
# # Get specific parts
# FIRST_PART=$(echo "$STRING" | cut -d '_' -f 1)
# SECOND_PART=$(echo "$STRING" | cut -d '_' -f 2)
# THIRD_PART=$(echo "$STRING" | cut -d '_' -f 3)

# traverse an array 
for PART in "${PARTS[@]}"; do
  # echo "$PART"
  # Convert first character to uppercase
  Name=${Name}$(echo "$PART" | sed 's/./\U&/') 
  # convert all charactera to uppercase
  NAME=${NAME}${PART^^} 
  NA_ME=${NA_ME}${PART^^}"_"
done

mkdir -p $name
cd $name

echo $Name
echo $NAME
echo $NA_ME

# "eval" allows you to construct and execute commands dynamically.
eval "${NAME}VERSION=1.0"
echo "${COVFERVERSION}"

date "+%Y-%m-%d %H:%M" # use date as separator

var_name="${NAME}VERSION"
# Create the variable dynamically
declare "$var_name=2.0"
# Access the newly created variable using indirect reference
echo "${!var_name}"  # Correct way to reference it