bash - 检查命名管道/FIFO 是否打开写入

标签 bash named-pipes lsof

我已经为其他一些进程创建了一个命名管道来写入并想检查其他进程是否正确启动,但不知道它的 PID。上下文是 running a command in screen ,确保命令正确启动。我希望这会起作用:

mkfifo /tmp/foo
echo hello > /tmp/foo &
lsof /tmp/foo

遗憾的是,lsof 不报告 echoinotifywait 可能是另一种选择,但并不总是安装,我真的只想轮询一次,而不是阻塞直到发生某些事件。

有什么方法可以检查命名管道是否打开写入?甚至一般开放?


更新:

一旦两端都连接好,lsof 似乎就可以工作了。这实际上解决了我的问题,但为了这个问题,我很想知道是否有可能在没有阅读器的情况下检测到命名管道的初始重定向。

> mkfifo /tmp/foo
> yes > /tmp/foo &
> lsof /tmp/foo
> cat /tmp/foo > /dev/null &
> lsof /tmp/foo
COMMAND   PID     USER   FD   TYPE DEVICE SIZE/OFF     NODE     NAME
yes     16915     user    1w  FIFO   8,18      0t0 16660270 /tmp/foo
cat     16950     user    3r  FIFO   8,18      0t0 16660270 /tmp/foo

最佳答案

更新 2:在使用了 inotify-tools 之后,似乎没有办法获得命名管道已打开写入且正在阻塞的通知。这可能就是为什么 lsof 在有读取器和写入器之前不显示管道的原因。

更新:在研究了命名管道之后,我不相信有任何方法可以单独使用命名管道。 推理:

  • 没有办法限制命名管道的写入者数量(不求助于锁定)
  • 如果没有读者,所有作者都阻塞
  • 如果有读取器,则不会有写入器阻塞(大概只要内核缓冲区未满)

您可以尝试在短时间内不向管道写入任何内容。如果超时到期,则写入被阻塞,表明有人已经打开管道进行写入。

注意:正如评论中所指出的,如果读者存在并且可能足够快,我们的测试写入将不会阻塞并且测试基本上会失败。注释掉下面的 cat 行来测试它。

#!/bin/bash

is_named_pipe_already_opened_for_writing() {
    local named_pipe="$1"
    # Make sure it's a named pipe
    if ! [ -p "$named_pipe" ]; then
        return 1
    fi
    # Try to write zero bytes in the background
    echo -n > "$named_pipe" &
    pid=$!
    # Wait a short amount of time
    sleep 0.1
    # Kill the background process. If kill succeeds, then
    # the write was blocked indicating that someone
    # else is already writing to the named pipe.
    kill $pid 2>/dev/null
}

PIPE=/tmp/foo

# Ignore any bash messages from killing below
trap : TERM

mkfifo $PIPE
# a writer
yes > $PIPE &
# a reader
cat $PIPE >/dev/null &

if is_named_pipe_already_opened_for_writing "$PIPE"; then
    echo "$PIPE is already being written to by another process"
else
    echo "$PIPE is NOT being written to by another process"
fi

jobs -pr | kill 2>/dev/null
rm -f $PIPE

关于bash - 检查命名管道/FIFO 是否打开写入,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42075387/

相关文章:

Linux lsof 需要帮助解析输出

java - JVM 的 lsof 对应物?

linux - 使用 lsof 的脚本在 shell 上运行良好,而不是在 cron 上运行良好

python - 主程序+配置模块的代码设计

bash - 带大括号的 bash 循环?

linux - 如何在 bash 中的某一行之后对文件进行排序?

bash - 无法在后台 shell 中捕获 SIGINT 信号

c# - 少量数据中的 NamedPipe 与 TCP/IP

php - C++ 和 PHP 中的命名管道

bash - cd 进入 while 循环中的目录不起作用