sshホストbash -cを介して実行されるコマンドでlsパラメータが機能しないように見えるのはなぜですか?

sshホストbash -cを介して実行されるコマンドでlsパラメータが機能しないように見えるのはなぜですか?

私はUbuntuを実行しているローカルホストとリモートホストがあり、シェルはbashに設定されています。私のホームディレクトリには、ローカルホストとリモートホストに2つのファイルがfile-1あります。各ホームディレクトリにはいくつかの異なるファイルがあり、一致するファイルのみをリストしたいと思います。file-2remotefile-*

ローカルでは、次の予測結果が生成されますfile-1 file-2

$ ls file-*
$ bash -c 'ls file-*'

ただし、このコマンドはリモートホームディレクトリ内のすべてのファイルを返します。そこで何が起こったの?

$ ssh remote bash -c 'ls file-*'
$ ssh remote bash -c 'ls "file-*"'
$ ssh remote bash -c 'ls file-\*'
$ ssh remote bash -c 'ls "file-\*"'

ssh remote 'ls file-*'これが期待した結果だけを生成することを知っていますが、渡されたssh remote bash -c 'ls ...'引数が破棄されたように見えるのはなぜですかls ...? (リモートで実行されたlsの出力もパイプして渡されたため、lsその出力のみ影響を受けるようですssh remote bash -c 'ls file-* | xargs -I {} echo "Got this: {}"'。)

ベストアンサー1

使用時にリモートホストで実行されるコマンドはssh remote bash -c 'ls file-*'次のとおりです。

bash -c ls file-*

これはbash -cスクリプトを実行することを意味しますls。位置引数として、bash -cスクリプトはリモートホストから一致する名前を取得しますfile-*(これらの名前の最初の名前はに置かれるため、$0実際には位置引数の一部ではありません)。引数はコマンドに渡されないため、lsディレクトリ内のすべての名前が一覧表示されます。

ssh最初のレベルの引用符(コマンドラインで使用される一連の外部引用符)を削除し、リモートホストで実行するコマンドを渡します。呼び出されたシェルはsshこれらの引用符を削除し、sshリモートコマンドの引数を区別するために新しい引用符を挿入しません(コマンドで使用される引用符を中断する可能性があるため)。

次のコマンドを使用してこれを確認できますssh -v

[...]
debug1: Sending command: bash -c ls file-*
[...]

表示される他の3つのコマンドは同じように機能しますが、シェルなどではなく$0文字列にのみ設定されます。file-*$1$2bash -c

おそらくあなたが望むのは、完全なコマンドを引用することです。

ssh remote 'bash -c "ls file-*"'

ssh -vデバッグ出力では、次のように報告されます。

[...]
debug1: Sending command: bash -c "ls file-*"
[...]

つまり、リモートコマンドに渡された文字列が引用符を削除した後、ローカルシェルで実行したいコマンドであることを確認する必要があります。

また、使用することができます

ssh remote bash -c \"ls file-\*\"

または

ssh remote bash -c '"ls file-*"'

おすすめ記事