我需要在我的 EC2 实例上添加一个关闭挂钩来做一些资源清理工作。
此外,我还可以手动启动和停止我的实例以进行测试,并且我希望启动和关闭挂钩的触发方式与初始引导程序相同。
然后我决定通过 Cloudformation bash 脚本在 AWS EC2 Ubuntu 16.04 LTS 实例上安装一个脚本作为服务。
这是脚本的第一个简单版本:
UserData:
"Fn::Base64":
!Sub
- |
#!/usr/bin/env bash
BOOTSTRAP_SCRIPT_NAME=bootstrap
BOOTSTRAP_SCRIPT_PATH=/etc/init.d/${BOOTSTRAP_SCRIPT_NAME}
cat > /etc/init.d/boostrap <<EOF
### BEGIN INIT INFO
# Provides: ${BOOTSTRAP_SCRIPT_NAME}
# Required-Start: \\\$local_fs \\\$remote_fs \\\$network \\\$syslog \\\$named
# Required-Stop: \\\$local_fs \\\$remote_fs \\\$network \\\$syslog \\\$named
# Default-Start: 2 3 4 5
# Default-Stop: 0 1 6
# Short-Description: Bootstrap an instance
# Description: Bootstrap an instance
### END INIT INFO
function start() {
echo "STARTUP on $(date)"
}
function stop() {
echo "SHUTDOWN on $(date)"
}
case "\$1" {
start)
start | tee -a /var/log/${BOOTSTRAP_SCRIPT_NAME}.log
;;
stop)
stop | tee -a /var/log/${BOOTSTRAP_SCRIPT_NAME}.log
;;
}
EOF
chmod +x ${BOOTSTRAP_SCRIPT_PATH}
update-rc.d -f ${BOOTSTRAP_SCRIPT_NAME} remove
update-rc.d ${BOOTSTRAP_SCRIPT_NAME} defaults
在这个版本中,引导脚本永远不会启动。
我很快就明白,引导脚本是在 cloud-init 阶段安装的,顺便说一句,是在 linux sysv init 阶段安装的,不会参与当前的 init 阶段......(如果这是错误的,请告诉我 ;-))
然后我决定手动启动它,例如 cloudformation bash 示例中的 apache2。我在脚本末尾添加了以下行。
${BOOTSTRAP_SCRIPT_PATH} start
我再次对其进行了测试,并在此修复后在 bootstrap.log 文件中看到了“STARTUP on XXX”日志。但是当我试图在控制台中停止实例时,bootstrap.log 文件中没有出现“XXX 上的 SHUTDOWN”日志......
我登录到实例并尝试手动启动/停止脚本......所有启动和关闭日志都出现了 8-O。然后我假设由于 boostrap 脚本没有被识别为初始化脚本,因此不会在实例停止或终止时调用停止回调......(如果这是错误的,请告诉我 ;-))
然后,我从 AWS 控制台启动和停止了几次实例,并且 STARTUP 和 SHUTDOWN 消息仍然出现在日志中。这证实了我的假设。日志仅在第一个初始化和关闭周期中丢失。
所以我做了一些奇怪而丑陋的事情......我用这个替换了最后一行 start 命令:
reboot -n
该脚本现在可以按我的需要运行,但我认为应该有一种更简洁的方法来启用我的脚本以进行初始化,或者至少在 cloud-init 期间的关闭阶段启用脚本,而无需重新启动...
有人对此问题有最佳解决方案或更多详细信息吗?
PS:我尝试过init u
,telinit u
而不是重新启动但没有成功