如何在不剥离任何尾随换行符的情况下将 stdin 捕获到变量?

Com*_*ast 12 shell shell-script echo newlines printf

在 shell 脚本中...

如何在不剥离任何尾随换行符的情况下将 stdin 捕获到变量?

现在我已经尝试过:

var=`cat`
var=`tee`
var=$(tee)
Run Code Online (Sandbox Code Playgroud)

在所有情况下$var都不会有输入流的尾随换行符。谢谢。

另外:如果输入中没有尾随换行符,则解决方案不得添加.

根据接受的答案更新:

我在代码中使用的最终解决方案如下:

function filter() {
    #do lots of sed operations
    #see https://github.com/gistya/expandr for full code
}

GIT_INPUT=`cat; echo x`
FILTERED_OUTPUT=$(printf '%s' "$GIT_INPUT" | filter)
FILTERED_OUTPUT=${FILTERED_OUTPUT%x}
printf '%s' "$FILTERED_OUTPUT"
Run Code Online (Sandbox Code Playgroud)

如果您想查看完整代码,请查看expandr的 github 页面,这是我为信息安全目的开发的一个小的开源git 关键字扩展过滤器shell 脚本。根据.gitattributes文件(可以是特定于分支的)和git config 中设置的规则,无论何时检入或检出存储库,git 都会通过expandr.sh shell 脚本通过管道传输每个文件。(这就是为什么保留任何尾随换行符或缺少换行符至关重要。)这使您可以清理敏感信息,并为测试、暂存和实时分支交换不同的环境特定值集。

vin*_*c17 9

在将值存储在变量中之前,尾部的换行符被剥离。您可能想要执行以下操作:

var=`cat; echo x`
Run Code Online (Sandbox Code Playgroud)

并使用${var%x}代替$var. 例如:

printf "%s" "${var%x}"
Run Code Online (Sandbox Code Playgroud)

请注意,这解决了尾随换行符问题,但不能解决空字节问题(如果标准输入不是文本),因为根据 POSIX 命令替换

如果输出包含任何空字节,则行为未指定。

但是 shell 实现可能会保留空字节。


phe*_*mer 5

您可以使用read内置函数来完成此操作:

$ IFS='' read -d '' -r foo < <(echo bar)

$ echo "<$foo>"
<bar
>
Run Code Online (Sandbox Code Playgroud)

对于读取 STDIN 的脚本,只需:

IFS='' read -d '' -r foo
Run Code Online (Sandbox Code Playgroud)

 

我不确定这会在什么 shell 中工作。但在 bash 和 zsh 中都可以正常工作。