如何使 sysv 服务依赖于 systemd 服务?

如何使 sysv 服务依赖于 systemd 服务?

我在 Ubuntu 18.04 上安装了 Redis 和 Tomcat 的发行版。我尝试在成功启动 Redis 后才启动 Tomcat,但到目前为止还没有成功。

  • tomcat8 包在 /etc/init.d/tomcat8 中有一个 sysv init 脚本,我看到了生成的 /run/systemd/generator.late/tomcat8.service。

  • Redis-server 软件包在 /lib/systemd/system/redis-server.service 中有服务单元

我已经在以下目录中创建了到 redis-server 的符号链接:

ln -s /lib/systemd/system/redis-server.service  /etc/systemd/system/tomcat8.service.requires/redis-server.service
ln -s /lib/systemd/system/redis-server.service  /etc/systemd/system/tomcat8.service.after/redis-server.service

但是如果我故意导致 Redis 无法启动,Tomcat 还是会启动。我希望在 Redis 失败的情况下 Tomcat 不会启动。

我看到 Tomcats 服务状态命令的输出为“LSB 依赖失败”,但 Tomcat 仍然启动了:

root@elkarel:/home/elkarel# service tomcat8 status
  tomcat8.service - LSB: Start Tomcat.
   Loaded: loaded (/etc/init.d/tomcat8; generated)
   Active: active (running) since Mon 2019-07-01 14:57:51 CEST; 2min 7s ago
     Docs: man:systemd-sysv-generator(8)
  Process: 690 ExecStart=/etc/init.d/tomcat8 start (code=exited, status=0/SUCCESS)
    Tasks: 44 (limit: 2340)
   CGroup: /system.slice/tomcat8.service
           └─737 /usr/lib/jvm/java-8-oracle/bin/java -Djava.util.logging.config.file=/var/lib/tomcat8/conf/logging.properties -Djava.util.logging.manager=org.apache.juli.ClassLoaderLogManager -Djava.awt.head

Jul 01 14:57:45 elkarel systemd[1]: Starting LSB: Start Tomcat....
Jul 01 14:57:45 elkarel systemd[1]: Dependency failed for LSB: Start Tomcat..
Jul 01 14:57:45 elkarel systemd[1]: tomcat8.service: Job tomcat8.service/start failed with result 'dependency'.
Jul 01 14:57:46 elkarel tomcat8[690]:  * Starting Tomcat servlet engine tomcat8
Jul 01 14:57:51 elkarel tomcat8[690]:    ...done.

systemctl daemon-reload每次做出任何改变时都会执行。

我没有改变包文件并且我也不想这样做,但是现在我怀疑是否有可能仅使用符号链接来声明这种依赖关系。

答案1

没有.after/,只有.wants/.requires/。因此,您的两个服务都排队等待同时启动。失败的 Requires= 依赖项应该导致 tomcat8 再次停止,但是... SysV 生成的服务在这方面很奇怪。

您需要将附加设置添加到 .conf 文件中<unit>.d/(例如使用systemctl edit tomcat8):

# /etc/systemd/system/tomcat8.service.d/deps.conf

[Unit]
Requires=redis.service
After=redis.service

注意:您应避免使用 .wants/ 或 .requires/ 进行常规自定义,因为这些目录中的符号链接可能会在“systemctl enable/disable”期间意外删除。为其他所有内容创建 .conf 文件或完整单元覆盖。

相关内容