産卵 - コマンドが見つかりません!


15

Mac OS X 10.9.4を使用しています。ローカルマシンから別のホストにファイルをコピーするスクリプトは次のとおりです。

#!/bin/bash
#!/usr/bin/expect

echo "I will fail if you give junk values!!"
echo " "
echo "Enter file name: "
read filePath
echo " "
echo "Where you want to copy?"
echo "Enter"
echo "1. if Host1"
echo "2. if Host2"
echo "3. if Host3"
read choice
echo " "
if [ $choice -eq "1" ]
then
  spawn scp filePath uname@host1:/usr/tmp   
  expect "password"   
  send "MyPassword\r"
  interact
elif [ $choice -eq "2" ]
then
  spawn scp filePath uname@host2:/usr/tmp   
  expect "password"   
  send "MyPassword\r"
  interact
elif [ $choice -eq "3" ]
then
  spawn scp filePath uname@host3:/usr/tmp   
  expect "password"   
  send "MyPassword\r"
  interact
else
  echo "Wrong input"
fi

このスクリプトを実行しているとき、私は次のようになっています

./rcopy.sh: line 21: spawn: command not found
couldn't read file "password": no such file or directory
./rcopy.sh: line 23: send: command not found
./rcopy.sh: line 24: interact: command not found

私の場合、期待コードの周りにEODマーカーが必要でした。stackoverflow.com/questions/26125036/...
AnneTheAgile

回答:


20

スクリプトは2つのインタープリターを結合しようとしています。との両方が#!/bin/bashあり#!/usr/bin/expectます。それは機能しません。2つのうち1つのみ使用できます。以来bash初めてだった、あなたのスクリプトはbashスクリプトとして実行されています。

ただし、スクリプト内には、やexpectなどのコマンドがあります。スクリプトはではなくによって読み取られているため、これは失敗します。これを回避するには、さまざまなスクリプトを作成してスクリプトから呼び出すか、すべてをに翻訳します。spawnsendbashexpectexpectbashexpect

しかし、最良の方法であり、単純なテキストファイルにプレーンテキストでパスワードを入れるという恐ろしい慣行を回避する方法は、代わりにパスワードなしのsshをセットアップすることです。そうscpすれば、パスワードは不要になり、次のものは必要ありませんexpect

  1. まず、マシン上で公開sshキーを作成します。

    ssh-keygen -t rsa

    各ログイン後にsshコマンドを初めて実行するときに入力するように求められるパスフレーズを求められます。これは、複数のsshまたはscpコマンドの場合、一度入力するだけで済むことを意味します。完全にパスワードなしでアクセスするには、パスフレーズを空のままにします。

  2. 公開鍵を生成したら、それをネットワーク内の各コンピューターにコピーします。

    while read ip; do 
     ssh-copy-id -i ~/.ssh/id_rsa.pub user1@$ip 
    done < IPlistfile.txt

    これIPlistfile.txtは、各行にサーバーの名前またはIPを含むファイルでなければなりません。例えば:

    host1
    host2
    host3

    これを行うのはこれが初めてなので、各IPのパスワードを手動で入力する必要がありますが、それを行うと、簡単な方法でこれらのマシンのいずれかにファイルをコピーできるようになります。

    scp file user@host1:/path/to/file
  3. スクリプトから期待を削除します。パスワードなしでアクセスできるようになったので、次のようにスクリプトを使用できます。

    #!/bin/bash
    echo "I will fail if you give junk values!!"
    echo " "
    echo "Enter file name: "
    read filePath
    echo " "
    echo "Where you want to copy?"
    echo "Enter"
    echo "1. if Host1"
    echo "2. if Host2"
    echo "3. if Host3"
    read choice
    echo " "
    if [ $choice -eq "1" ]
    then
      scp filePath uname@host1:/usr/tmp   
    elif [ $choice -eq "2" ]
    then
      scp filePath uname@host2:/usr/tmp   
    elif [ $choice -eq "3" ]
    then
      scp filePath uname@host3:/usr/tmp   
    else
      echo "Wrong input"
    fi

1
ssh-agentを起動する場合、パスフレーズを入力する必要があるのは1回だけです。それ以外の場合は、毎回入力する必要があります。
グレンジャックマン

5

コードはもっと簡潔にすることができます。

#!/bin/bash

read -p "Enter file name: " filePath
if ! [[ -r $filePath ]]; then
    echo "cannot read $filePath"
    exit 1
fi

PS3="Where you want to copy? "
select host in host1 host2 host3; do
    if [[ -n $host ]]; then
        expect <<END
            spawn scp "$filePath" uname@$host:/usr/tmp   
            expect "password"   
            send "MyPassword\r"
            expect eof
END
        break
    fi
done

提案されているようにsshキーを設定する場合、さらに優れています。

    if [[ -n $host ]]; then
        scp "$filePath" uname@$host:/usr/tmp   
        break
    fi

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.