AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 560167
Accepted
ghborrmann
ghborrmann
Asked: 2020-01-04 08:09:11 +0800 CST2020-01-04 08:09:11 +0800 CST 2020-01-04 08:09:11 +0800 CST

如何在关机期间写入控制台?

  • 772

我试图阻止用户在没有从机器上物理移除闪存卡的情况下关闭或重新启动。为此,我编写了一个 SystemD 服务 removeflash.service:

[Unit]
Description=Prompt user to remove flash card

[Service]
ExecStop=/usr/lib/systemd/flashshutdown.sh
Type=oneshot
RemainAfterExit=yes

[Install]
WantedBy=multi-user.target
Requires=rsyslog.service

flashshutdown.sh 是一个 bash 脚本,如下:

#! /bin/bash
#
while [ -e /dev/flash ] ; do
    echo "Please remove flash card"
    logger "GHB: Please remove flash card"
    sleep 5
done

我没想到 echo 在关机时会做任何事情,但事实并非如此。但是,我希望 logger 命令能够工作。没有Requires=rsyslog.service,我的服务在rsyslog.service关闭后退出;我插入了防止这种情况发生的要求,但唯一的区别是 removeflash.service 的关闭在关闭顺序中更早。幸运的是,systemd 本身向控制台输出一条消息,表明它正在运行一个作业,提示用户移除闪存卡,从而挽救了服务的目的。

向控制台发出消息的正确方法是什么?

systemd console
  • 2 2 个回答
  • 923 Views

2 个回答

  • Voted
  1. Best Answer
    JdeBP
    2020-01-04T11:45:40+08:002020-01-04T11:45:40+08:00

    服务管理下的服务的标准输出和错误——无论是 s6、runit、perp、daemontools、nosh 服务管理还是 systemd——都不是控制台。它是连接到某种形式的日志写入器的管道。

    对于 systemd 服务,如果您想读取(但您不想)以及写入,则需要在 .INI 文件中使用 aTTYPath=/dev/console和 aStandardOutput=tty来更改此设置。见证 systemd 预先提供的.StandardInput=ttydebug-shell.service

    这是一个一般原则,不是系统特定的。守护进程上下文涉及(除其他外)没有控制终端并且没有终端的打开文件描述符,并且在适当的服务管理下(例如所有 daemontools 系列)这是一个开始的地方,服务进程开始的状态在主管/服务经理分叉时。因此,要使用控制台,服务必须显式打开它。

    在 systemd 中,上述TTYPath和StandardInput设置会导致分叉的子进程在正确执行服务程序之前打开控制台。这隐藏在 systemd 中,您并没有真正看到它。在run类似 nosh 服务的run程序中,程序在执行主程序之前显式使用了一些 nosh 工具集链式加载工具来做同样的事情(emergency-login在本例中):

    % cat /etc/service-bundles/services/emergency-login@console/service/run
    #!/bin/nosh
    #紧急超级用户登录控制台
    设置
    vc-get-tty 控制台
    开放控制-tty
    vc-reset-tty --hard-reset
    line-banner "紧急模式登录。"
    紧急登录
    %
    

    具有讽刺意味的是,您不需要该logger命令或任何 syslog 依赖项。将此交互式提示写入日志是没有意义的。但原则上,您确实应该无特权地运行此服务。对于它所做的任何事情,它都不需要超级用户权限。

    #!/bin/bash在另一个原则上,除非你真的要使用 Bashisms ,否则不要使用你的脚本。在过去的几十年中,Debian Linux 和 Ubuntu Linux 系统启动/关闭的最大加速之一是/bin/sh从 Bourne Again shell 切换到 Debian Almquist shell。如果您要编写像这样简单的脚本,请保持它符合 POSIX 并使用#!/bin/sh ,即使您不使用 Debian/Ubuntu,并且在 Debian/Ubuntu 上,您将获得 Debian Almquist shell 的好处作为奖励。

    此外,如果您决定使用不只是玻璃 TTY 消息,使用类似的工具dialog,您将需要设置TERM环境变量,以便您的程序可以查找正确的转义和控制序列以在 terminfo 数据库中发出。再次,见证debug-shell.service。(在上述run程序中,为了比较,vc-get-tty工具集TERM。)

    同样,您将希望记录脚本错误。所以标准错误应该指向带有StandardError=journal. 这是一个 nosh 服务run程序,它说明了这个等价物,并且还显示了为一个真正不需要它们的程序删除用户权限,在 systemd .INI 文件中将是User=daemon:

    % cat /etc/service-bundles/services/monitor-fsck-progress/service/run
    #!/bin/nosh
    #local socket 用于 monitor-fsck-progress
    local-stream-socket-listen --systemd-compatibility --backlog 2 --mode 0644 /run/fsck.progress
    设置
    setlogin -- 守护进程
    vc-get-tty 控制台
    fdmove -c 4 2
    开放控制-tty
    fdmove 2 4
    setuidgid -- 守护进程
    。/服务
    %
    

    在这种情况下运行的程序会./service在控制台上显示全屏 TUI,同时将其错误发送到日志服务。这是一个人需要做的事情,在一般的服务管理器下,为了运行诸如服务之类的程序,与控制台对话。

    当然,任何这样的全屏 TUI 程序都会与 systemd 的“A stop job is running”冲突,同样写入控制台。但那是你的问题。☺

    进一步阅读

    • https://unix.stackexchange.com/a/468457/5132
    • https://unix.stackexchange.com/a/250965/5132
    • https://unix.stackexchange.com/a/499148/5132
    • https://unix.stackexchange.com/a/233855/5132
    • 鞭尾或对话
    • 4
  2. K7AAY
    2020-01-04T08:48:01+08:002020-01-04T08:48:01+08:00

    你很亲密。在脚本中早期使用echo之前,请在其上方使用reset。例子:

    # echo ^G
    # reset
    # echo ^G MESSAGE TO PUT ON THE TERMINAL
    

    发送 ^G 到 echo 会响起终端铃声,即哔哔声。

    • 0

相关问题

  • journalctl 中的区分级别

  • 将默认编辑器更改为 vim for _ sudo systemctl edit [unit-file] _

  • systemd:如何在服务启动时运行脚本,而不编辑服务定义

  • 使用 systemd 看门狗支持重新启动应用程序

  • 使用键盘快捷键启动/停止 systemd 服务 [关闭]

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve