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 / 问题 / 717457
Accepted
PatS
PatS
Asked: 2022-09-16 12:57:12 +0800 CST2022-09-16 12:57:12 +0800 CST 2022-09-16 12:57:12 +0800 CST

为什么 ssh localhost sh -c 'echo "$0"' arg0 不输出任何东西?

  • 772

类似于为什么 echo called as /bin/sh -c echo foo 不输出任何东西?但使用ssh.

为什么这不输出任何东西:

ssh localhost sh -c 'echo "0=$0 1=$1"' arg1 arg2 arg3

但是,如果我将其更改为

ssh localhost sh -c 'true; echo "0=$0 1=$1"' arg1 arg2 arg3
# Output is
0=bash 1= arg1 arg2 arg3

我看到暗示正在运行 echo 命令的行为,但变量替换的工作方式与预期不同。见https://unix.stackexchange.com/a/253424/119816

在没有 ssh 的情况下运行按预期工作

与上面相同,但删除了 ssh 命令

sh -c 'echo "0=$0 1=$1"' arg1 arg2 arg3
# Output is
0=arg1 1=arg2

# Adding true gives same output
sh -c 'true; echo "0=$0 1=$1"' arg1 arg2 arg3
0=arg1 1=arg2

我正在尝试将根可访问文件从一个系统复制到另一个系统

我正在尝试让 ssh 命令将根文件复制到另一个系统。我正在尝试的命令是:

file=/etc/hosts
set -x
ssh host1 sudo cat $file | ssh host2 sudo sh -c 'exec cat "$0"' $file
# Output is
+ ssh host2 sudo sh -c 'exec cat > "$0"' /etc/hosts
+ ssh host1 sudo cat /etc/hosts
bash: : No such file or directory

这对我来说看起来不错,我不确定如何排除故障。

我的解决方案

我的解决方案是回退到我以前使用过的

ssh host1 sudo cat $file | ssh host2 sudo tee $file > /dev/null

以上工作。

寻找解决方案

我遇到了这个问题并问了这个问题:

  • Unix 是否有从标准输入读取并写入文件的命令(如 tee 而不将输出发送到标准输出)?

其他人对命令有问题/疑问sh -c:

  • 为什么 echo call as /bin/sh -c echo foo 不输出任何东西?

但是在使用 ssh 时肯定会发生一个微妙的问题,这会导致额外的 shell 评估。

shell ssh
  • 2 2 个回答
  • 510 Views

2 个回答

  • Voted
  1. roaima
    2022-09-16T13:10:48+08:002022-09-16T13:10:48+08:00

    这与您引用的问题中的问题相同。本地 shell 删除了一层引号。这是正在发生的事情

    初始代码

    ssh localhost sh -c 'echo "0=$0 1=$1"' arg1 arg2 arg3
    

    在本地 shell 扩展之后但在命令执行之前

    ssh localhost sh -c echo "0=$0 1=$1" arg1 arg2 arg3
    #                   ^^^^^^^^^^^^^^^^ a single token
    #             ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ passed to the remote shell
    

    在 shell 扩展之后但在执行之前的远程 shell 中

    sh -c echo 0= 1= arg1 arg2 arg3
    #          ^^^^^ a single token
    #     ^^^^ argument for -c
    

    echo绑定到作为要执行的sh -c命令,其余的是参数。

    你应该得到一个空行。

    就个人而言,我了解到ssh将其 args 直接传递给远程 shell,这就是它被执行的地方。因此,我倾向于编写ssh这样的命令,这提醒我该some command line…部分将逐字传递到远程 shell 以执行,就好像我直接键入它一样:

    ssh remoteHost 'some command line…'
    

    (用单引号将行括起来以避免局部评估,或使用双引号插入局部变量。)


    然后您说实际上您正在尝试让 ssh 命令将根文件复制到另一个系统

    file=/etc/hosts
    ssh -n host1 "scp -p '$file' host2:'/path/to/destination'"
    

    或者,由于文件名是安全的,并且假设您在本地客户端和之间具有根等价性host1,并且 from host1to host2:

    ssh -n root@host1 scp -p /etc/hosts host2:/etc/hosts
    

    有了现代scp,反过来又可以进一步简化:

    scp -OR -p root@host1:/etc/hosts root@host2:/etc/
    
    • 11
  2. Best Answer
    Toby Speight
    2022-09-17T05:57:44+08:002022-09-17T05:57:44+08:00

    如果你添加-v到你的ssh命令,它会告诉你它在做什么:

    debug1: Sending command: sh -c echo "0=$0 1=$1" arg1 arg2 arg3
    

    果然,如果我们在sh本地运行该命令,我们只会得到一个空行输出,因为sh正在运行echo并"0=$0 1=$1"成为第零个参数。

    当我们添加 时true;,会发生这种情况:

    debug1: Sending command: sh -c true; echo "0=$0 1=$1" arg1 arg2 arg3
    

    现在sh只是运行true。然后echo输出它的论点:

    0=bash 1= arg1 arg2 arg3
    

    您可能想要的是sh -c 'echo "0=$0 1=$1"' arg1 arg2 arg3在远端运行。您需要引用整个命令,以便远程 shell 完整接收它:

    ssh localhost "sh -c 'echo \"0=\$0 1=\$1\"' arg1 arg2 arg3"
    

    输出:

    0=arg1 1=arg2
    

    如果你有 GNU printf(在本地),你可以通过使用它来为远端适当地格式化命令字符串来使其更具可读性:

    command='echo "0=$0 1=$1"'
    ssh localhost "sh -c $(printf %q "$command") arg1 arg2 arg3"
    
    • 5

相关问题

  • 这个命令是如何工作的?mkfifo /tmp/f; 猫/tmp/f | /bin/sh -i 2>&1 | 数控 -l 1234 > /tmp/f

  • FreeBSD 的 sh:列出函数

  • 有没有办法让 ls 只显示某些目录的隐藏文件?

  • grep -v grep 有什么作用

  • 如何将带有〜的路径保存到变量中?

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