bash - 将预先准备好的输入参数字符串作为输入参数传递给程序(mailx)
问题描述
我正在创建一个工具来解析输入文件以重新构建 mailx 命令(为 mailx 创建数据的服务器无法发送电子邮件,因此我需要将数据存储到文件中,以便另一台服务器可以重建命令并发送它) . 我可以将整个命令输出到一个文件并在另一台服务器上执行该文件,但这几乎不安全/安全......任何人都可以拦截该文件并插入将以 root 身份运行的恶意内容 - 这个解析工具正在检查每个分钟用于使用 systemd 计时器和服务解析和发送电子邮件的任何文件。
我已经使用这种格式的“标记/分隔符”创建了文件:
-------MESSAGE START-------
Email Body Text
Goes Here
-------MESSAGE END-------
-------SUBJECT START-------
Email Subject
-------SUBJECT END-------
-------ATTACHEMENT START-------
path to file to attach if supplied
-------ATTACHEMENT END-------
-------S OPTS START-------
list of mailx '-S' options eg from=EMAILNAME <email@b.c> or sendwait etc each one on a new line
-------S OPTS END-------
-------EMAIL LIST START-------
string of recipient emails comma separated eg. email1,email2,email3 etc..
-------EMAIL LIST END-------
我有一个程序来解析这个文件并重建,然后运行 mailx 命令:
#!/bin/bash
## Using systemd logging to journal for this as its now being called as part of a service
## See: https://serverfault.com/questions/573946/how-can-i-send-a-message-to-the-systemd-journal-froma-the-command-line (kkm answer)
start_time="$(date +[%c])"
exec 4>&2 2> >(while read -r REPLY; do printf >&4 '<3>%s\n' "$REPLY"; done)
echo >&4 "<5>$start_time -- Started gpfs_flag_email.sh"
trap_exit(){
exec >2&
}
trap 'trap_exit' EXIT
email_flag_path="<PATH TO LOCATION>/email_flags/"
mailx_message_start="-------MESSAGE START-------"
mailx_message_end="-------MESSAGE END-------"
mailx_subject_start="-------SUBJECT START-------"
mailx_subject_end="-------SUBJECT END-------"
mailx_attachement_start="-------ATTACHEMENT START-------"
mailx_attachement_end="-------ATTACHEMENT END-------"
mailx_s_opts_start="-------S OPTS START-------"
mailx_s_opts_end="-------S OPTS END-------"
mailx_to_email_start="-------EMAIL LIST START-------"
mailx_to_email_end="-------EMAIL LIST END-------"
no_attachment=false
no_additional_opts=false
additional_args_switch="-S "
num_files_in_flag_path="$(find $email_flag_path -type f | wc -l)"
if [[ $num_files_in_flag_path -gt 0 ]]; then
for file in $email_flag_path*; do
email_message="$(awk "/$mailx_message_start/,/$mailx_message_end/" $file | egrep -v -- "$mailx_message_start|$mailx_message_end")"
email_subject="$(awk "/$mailx_subject_start/,/$mailx_subject_end/" $file | egrep -v -- "$mailx_subject_start|$mailx_subject_end")"
email_attachment="$(awk "/$mailx_attachement_start/,/$mailx_attachement_end/" $file | egrep -v -- "$mailx_attachement_start|$mailx_attachement_end")"
email_additional_opts="$(awk "/$mailx_s_opts_start/,/$mailx_s_opts_end/" $file | egrep -v -- "$mailx_s_opts_start|$mailx_s_opts_end")"
email_addresses="$(awk "/$mailx_to_email_start/,/$mailx_to_email_end/" $file | egrep -v -- "$mailx_to_email_start|$mailx_to_email_end" | tr -d '\n')"
if [[ -z "$email_message" || -z "$email_subject" || -z "$email_addresses" ]]; then
echo >&2 "MISSING DETAILS IN INPUT FILE $file.... Exiting With Error"
exit 1
fi
if [[ -z "$email_attachment" ]]; then
no_attachment=true
fi
if [[ -z "$email_additional_opts" ]]; then
no_additional_opts=true
else
additional_opts_string=""
while read -r line; do
if [[ ! $line =~ [^[:space:]] ]]; then
continue
else
additional_opts_string="$additional_opts_string \"${additional_args_switch} '$line'\""
fi
done <<<"$(echo "$email_additional_opts")"
additional_opts_string="$(echo ${additional_opts_string:1} | tr -d '\n')"
fi
if [[ $no_attachment = true ]]; then
if [[ $no_additional_opts = true ]]; then
echo "$email_message" | mailx -s "$email_subject" $email_addresses
else
echo "$email_message" | mailx -s "$email_subject" $additional_opts_string $email_addresses
fi
else
if [[ $no_additional_opts = true ]]; then
echo "$email_message" | mailx -s "$email_subject" -a $email_attachment $email_addresses
else
echo "$email_message" | mailx -s "$email_subject" -a $email_attachment $additional_opts_string $email_addresses
fi
fi
done
fi
find $email_flag_path -type f -delete
exit 0
然而,上面有一个问题,我只能解决...... -S 选择完全搞砸了电子邮件标题,我最终将电子邮件发送给错误的人(我已经设置了回复和来自选项,但电子邮件标题混乱,回复电子邮件最终在 to: 字段中),如下所示:
To: Me <a@b.com>, sendwait@a.lan , -S@a.lan <-s@a.lan>, another-email <another@b.com>
我要做的就是重建命令,就像我在 CLI 中键入它一样:
echo "EMAIL BODY MESSAGE" | mailx -s "EMAIL SUBJECT" -S "from=EMAILNAME <email@b.c>" -S "replyto=EMAILNAME <email@b.c>" -S sendwait my.email@b.com
我试过引用''和“”引用它周围的其他mailx参数等等......我已经编写了其他将变量作为输入参数传递的工具,所以我无法理解我是如何搞砸的...... .
任何帮助,将不胜感激...
编辑
多亏了 Gordon Davisson 的真正有用的评论,我不仅能够修复它,而且能够理解修复以及使用数组并适当地引用变量......关于使用 printf 的提示真的非常有助于帮助我理解我做错了什么以及如何纠正它:P
declare -a mailx_args_array
...
num_files_in_flag_path="$(find $email_flag_path -type f | wc -l)"
if [[ $num_files_in_flag_path -gt 0 ]]; then
for file in $email_flag_path*; do
....
mailx_args_array+=( -s "$email_subject" )
if [[ ! -z "$email_attachment" ]]; then
mailx_args_array+=( -a "$email_attachment" )
fi
if [[ ! -z "$email_additional_s_opts" ]]; then
while read -r s_opt_line; do
mailx_args_array+=( -S "$s_opt_line" )
done < <(echo "$email_additional_s_opts")
fi
mailx_args_array+=( "$email_addresses" )
echo "$email_message" | mailx "${mailx_args_array[@]}"
done
fi
解决方案
推荐阅读
- c++ - 无法在 unordered_set 中找到()结构属性
- c# - Revit API - 项目中未加载任何空间标签系列
- php - Laravel 从 URL 中检索 id
- r - R:使用 dplyr 改变多列
- sql - 查找所有使用需要更改类型的列的主键和外键
- php - Laravel db:种子成功但我的数据库中没有数据
- sql-server - 向上导航并返回顶部项目的 SQL 递归查询
- ios - 将字符串转换为日期总是晚 2 小时
- python - Pandas:在顺序无关紧要的子集中查找重复项并将它们组合起来
- java - 如何使用apache poi在ms word中的两个项目符号列表之间插入文本