web-dev-qa-db-ja.com

Linux:宛先ディレクトリーが存在しない場合はコピーして作成する

宛先ディレクトリが存在しない場合はそれを作成するコマンド(またはおそらくcpのオプション)が必要です。

例:

cp -? file /path/to/copy/file/to/is/very/deep/there
262
flybywire
mkdir -p "$d" && cp file "$d"

cpにはそのようなオプションはありません)。

以下の両方が当てはまる場合:

  1. GNUバージョンのcpを使用しています(たとえば、Macバージョンではありません)。
  2. 既存のディレクトリ構造からコピーしているだけで、それを再作成する必要があります。

それからcp--parentsフラグでこれを行うことができます。情報ページ( http://www.gnu.org/software/coreutils/manual/html_node/cp-invocation.html#cp-invocation またはinfo cpまたはman cpで表示可能)から:

--parents
     Form the name of each destination file by appending to the target
     directory a slash and the specified name of the source file.  The
     last argument given to `cp' must be the name of an existing
     directory.  For example, the command:

          cp --parents a/b/c existing_dir

     copies the file `a/b/c' to `existing_dir/a/b/c', creating any
     missing intermediate directories.

例:

/tmp $ mkdir foo
/tmp $ mkdir foo/foo
/tmp $ touch foo/foo/foo.txt
/tmp $ mkdir bar
/tmp $ cp --parents foo/foo/foo.txt bar
/tmp $ ls bar/foo/foo
foo.txt
202
Paul Whipp

そのような古い質問ですが、別の解決策を提案することもできます。

installプログラムを使用してファイルをコピーし、「オンザフライ」で宛先パスを作成することができます。

install -D file /path/to/copy/file/to/is/very/deep/there/file

ただし、考慮すべき点がいくつかあります。

  1. あなたがする必要があります宛先ファイル名も指定、宛先パスだけでなく
  2. 宛先ファイル実行可能になる(少なくとも、私のテストで見た限り)

-mオプションを追加して宛先ファイルにアクセス権を設定することで、簡単に#2を修正できます(例:-m 664は、touchで新しいファイルを作成するのと同じように、アクセス許可rw-rw-r--で宛先ファイルを作成します)。


そしてここでそれは 私が触発された答えへの恥知らずなリンク =)

32
help_asap

あなたが望むことをするシェル関数。ファイルを置くための穴を掘るので「埋め」コピーと呼びます。

bury_copy() { mkdir -p `dirname $2` && cp "$1" "$2"; }
15
Andy Ross

これを行う1つの方法は次のとおりです。

mkdir -p `dirname /path/to/copy/file/to/is/very/deep/there` \
   && cp -r file /path/to/copy/file/to/is/very/deep/there

dirnameはあなたに行き先のディレクトリまたはファイルの親を与えます。 mkdir -p `dirname ...`はそのディレクトリを作成し、cp -rを呼び出したときに正しいベースディレクトリが正しいことを確認します。

この親の利点は、宛先パスの最後の要素がファイル名の場合に機能することです。

そしてそれはOS X上で動作します。

6
Jamie McCrindle

install -D file -m 644 -t /path/to/copy/file/to/is/very/deep/there

5
Spongman

上記の答えを尊重しているので、rsyncを使用することをお勧めします。

$  rsync -a directory_name /path_where_to_inject_your_directory/

例:

$ rsync -a test /usr/local/lib/
2
marcdahan

一行で、再開して完全な実用的な解決策を提示するだけです。あなたがあなたのファイルの名前を変更したい場合は注意してください、あなたはmkdirにきれいなディレクトリパスを提供する方法を含めるべきです。 $ fdstはファイルまたはディレクトリです。次のコードはどんな場合でもうまくいくはずです。

fsrc=/tmp/myfile.unk
fdst=/tmp/dir1/dir2/dir3/myfile.txt
mkdir -p $(dirname ${fdst}) && cp -p ${fsrc} ${fdst}

またはバッシュ特有の

fsrc=/tmp/myfile.unk
fdst=/tmp/dir1/dir2/dir3/myfile.txt
mkdir -p ${fdst%/*} && cp -p ${fsrc} ${fdst}
1
Kaalahaan

あなたの.bashrcに以下を追加するだけです。必要なら微調整してください。 Ubuntuで動作します。

mkcp() {
    test -d "$2" || mkdir -p "$2"
    cp -r "$1" "$2"
}

たとえば、testファイルをコピー先のディレクトリdにコピーしたい場合は、

mkcp test a/b/c/d

mkcp最初にコピー先のディレクトリが存在するかどうかチェックし、存在しない場合はそれを作成してソースファイル/ディレクトリをコピーします。

1
SD.

これは私のためにそれをやります

cp -vaR ./from ./to
1
Leroy Dunn

cpには複数の用途があります。

$ cp --help
Usage: cp [OPTION]... [-T] SOURCE DEST
  or:  cp [OPTION]... SOURCE... DIRECTORY
  or:  cp [OPTION]... -t DIRECTORY SOURCE...
Copy SOURCE to DEST, or multiple SOURCE(s) to DIRECTORY.

@ AndyRossの回答は

cp SOURCE DEST

cpのスタイルが、あなたが使用する場合は間違ったことをします

cp SOURCE... DIRECTORY/

cpのスタイル。

私は "DEST"はこの用法では末尾のスラッシュなしにあいまいであると思います(すなわちターゲットディレクトリがまだ存在していないところ)、それはおそらくcpがこれに対するオプションを追加しなかった理由です。

それで、これがdestディレクトリに末尾のスラッシュを強制するこの関数の私のバージョンです:

cp-p() {
  last=${@: -1}

  if [[ $# -ge 2 && "$last" == */ ]] ; then
    # cp SOURCE... DEST/
    mkdir -p "$last" && cp "$@"
  else
    echo "cp-p: (copy, creating parent dirs)"
    echo "cp-p: Usage: cp-p SOURCE... DEST/"
  fi
}
1
Rich

私はCP(Note大文字)と呼ばれるcpのためのサポートスクリプトを書きました。スクリプトはあなたが入力したパスのエラーをチェックし(最後のものが目的地である場合を除く)、それでも問題なければ、コピーを開始する前にmkdir -pステップを実行して目的のパスを作成します。この時点で、通常のcpユーティリティが引き継ぎ、CPとともに使用するスイッチ(-r、-p、-rpLなど)は直接cpにパイプ接続されます。私のスクリプトを使用する前に、理解しておく必要があることがいくつかあります。

  • ここにあるすべての情報はCP --helpを実行することでアクセスできます。 CP --help-allはcpのスイッチを含みます。
  • 目的のパスが見つからない場合、通常のcpはコピーを行いません。あなたはCPによるタイプミスのためのそのような安全策を持っていません。あなたの目的地が作成されますので、あなたの目的地を/ usrr/share/iconsや/ usr/share/iconのように間違って入力した場合は、それが作成されます。
  • 通常のcpは、既存のパスでの動作をモデル化する傾向があります。cp/a/b/c/dは、dが存在するかどうかによって異なります。 dが既存のフォルダの場合、cpはbをそこにコピーし、/ c/d/bにします。 dが存在しない場合、bはcにコピーされ、dに名前変更されます。 dが存在するがファイルであり、bがファイルである場合、それはbのコピーによって上書きされます。 cが存在しない場合、cpはコピーを実行せずに終了します。

CPは既存の経路から手がかりを得る贅沢を持っていないので、それはいくつかの非常にしっかりした行動パターンを持たなければなりません。 CPは、コピーしようとしている項目がコピー先のパスにドロップされ、コピー先ではないことを前提としています(別名、コピー元のファイル/フォルダのコピー)。意味:

  • dがフォルダの場合、 "CP/a/b/c/d"は/ c/d/bになります。
  • / c/bのbがフォルダーの場合、 "CP/a/b/c/b"は/ c/b/bになります。
  • Bとdの両方がファイルの場合:CP/a/b/c/dは/ c/dになります(dはbのコピーです)。同じ状況のCP/a/b/c/bでも同じです。

このデフォルトのCPの振る舞いは "--rename"スイッチで変更することができます。この場合は、

  • 「CP --rename/a/b/c/d」は、bを/ cにコピーし、そのコピーの名前をdに変更しています。

最後の注意点:cpと同様に、CPは一度に複数の項目をコピーすることができ、リストの最後のパスを宛先と見なします。引用符を使用する限り、スペースを含むパスも処理できます。

CPはあなたが入れたパスをチェックし、コピーをする前にそれらが存在することを確認します。厳密モード(--strictスイッチで使用可能)では、コピーされるすべてのファイル/フォルダが存在する必要があります。そうしないと、コピーは行われません。リラックスモード(--relaxed)では、リストした項目のうち少なくとも1つが存在すればコピーは続行されます。リラックスモードがデフォルトです。スイッチを使用して一時的にモードを変更するか、スクリプトの冒頭で変数easy_goingを設定することで永続的にモードを変更できます。

これをインストールする方法は次のとおりです。

非ルート端末では、次のようにします。

Sudo echo > /usr/bin/CP; Sudo chmod +x /usr/bin/CP; Sudo touch /usr/bin/CP
gedit admin:///usr/bin/CP 

Geditで、CPユーティリティを貼り付けて保存します。

#!/bin/bash
#Regular cp works with the assumption that the destination path exists and if it doesn't, it will verify that it's parent directory does.

#eg: cp /a/b /c/d will give /c/d/b if folder path /c/d already exists but will give /c/d (where d is renamed copy of b) if /c/d doesn't exists but /c does.

#CP works differently, provided that d in /c/d isn't an existing file, it assumes that you're copying item into a folder path called /c/d and will create it if it doesn't exist. so CP /a/b /c/d will always give /c/d/b unless d is an existing file. If you put the --rename switch, it will assume that you're copying into /c and renaming the singl item you're copying from b to d at the destination. Again, if /c doesn't exist, it will be created. So CP --rename /a/b /c/d will give a /c/d and if there already a folder called /c/d, contents of b will be merged into d. 

#cp+ $source $destination
#mkdir -p /foo/bar && cp myfile "$_"

err=0 # error count
i=0 #item counter, doesn't include destination (starts at 1, ex. item1, item2 etc)
m=0 #cp switch counter (starts at 1, switch 1, switch2, etc)
n=1 # argument counter (aka the arguments inputed into script, those include both switches and items, aka: $1 $2 $3 $4 $5)
count_s=0
count_i=0
easy_going=true #determines how you deal with bad pathes in your copy, true will allow copy to continue provided one of the items being copied exists, false will exit script for one bad path. this setting can also be changed via the custom switches: --strict and --not-strict
verbal="-v"


  help="===============================================================================\
    \n         CREATIVE COPY SCRIPT (CP) -- written by thebunnyrules\
    \n===============================================================================\n
    \n This script (CP, note capital letters) is intended to supplement \
    \n your system's regular cp command (note uncapped letters). \n
    \n Script's function is to check if the destination path exists \
    \n before starting the copy. If it doesn't it will be created.\n    
    \n To make this happen, CP assumes that the item you're copying is \
    \n being dropped in the destination path and is not the destination\
    \n itself (aka, a renamed copy of the source file/folder). Meaning:\n 
    \n * \"CP /a/b /c/d\" will result in /c/d/b \
    \n * even if you write \"CP /a/b /c/b\", CP will create the path /a/b, \
    \n   resulting in /c/b/b. \n
    \n Of course, if /c/b or /c/d are existing files and /a/b is also a\
    \n file, the existing destination file will simply be overwritten. \
    \n This behavior can be changed with the \"--rename\" switch. In this\
    \n case, it's assumed that \"CP --rename /a/b /c/d\" is copying b into /c  \
    \n and renaming the copy to d.\n
    \n===============================================================================\
    \n        CP specific help: Switches and their Usages \
    \n===============================================================================\n
    \
    \n  --rename\tSee above. Ignored if copying more than one item. \n
    \n  --quiet\tCP is verbose by default. This quiets it.\n
    \n  --strict\tIf one+ of your files was not found, CP exits if\
    \n\t\tyou use --rename switch with multiple items, CP \
    \n\t\texits.\n
    \n  --relaxed\tIgnores bad paths unless they're all bad but warns\
    \n\t\tyou about them. Ignores in-appropriate rename switch\
    \n\t\twithout exiting. This is default behavior. You can \
    \n\t\tmake strict the default behavior by editing the \
    \n\t\tCP script and setting: \n
    \n\t\teasy_going=false.\n
    \n  --help-all\tShows help specific to cp (in addition to CP)."

cp_hlp="\n\nRegular cp command's switches will still work when using CP.\
    \nHere is the help out of the original cp command... \
    \n\n===============================================================================\
    \n          cp specific help: \
    \n===============================================================================\n"

outro1="\n******************************************************************************\
    \n******************************************************************************\
    \n******************************************************************************\
    \n        USE THIS SCRIPT WITH CARE, TYPOS WILL GIVE YOU PROBLEMS...\
    \n******************************************************************************\
    \n******************************* HIT q TO EXIT ********************************\
    \n******************************************************************************"


#count and classify arguments that were inputed into script, output help message if needed
while true; do
    eval input="\$$n"
    in_=${input::1}

    if [ -z "$input" -a $n = 1 ]; then input="--help"; fi 

    if [ "$input" = "-h" -o "$input" = "--help" -o "$input" = "-?" -o "$input" = "--help-all" ]; then
        if [ "$input" = "--help-all" ]; then 
            echo -e "$help"$cp_hlp > /tmp/cp.hlp 
            cp --help >> /tmp/cp.hlp
            echo -e "$outro1" >> /tmp/cp.hlp
            cat /tmp/cp.hlp|less
            cat /tmp/cp.hlp
            rm /tmp/cp.hlp
        else
            echo -e "$help" "$outro1"|less
            echo -e "$help" "$outro1"
        fi
        exit
    fi

    if [ -z "$input" ]; then
        count_i=$(expr $count_i - 1 ) # remember, last item is destination and it's not included in cound
        break 
    Elif [ "$in_" = "-" ]; then
        count_s=$(expr $count_s + 1 )
    else
        count_i=$(expr $count_i + 1 )
    fi
    n=$(expr $n + 1)
done

#error condition: no items to copy or no destination
    if [ $count_i -lt 0 ]; then 
            echo "Error: You haven't listed any items for copying. Exiting." # you didn't put any items for copying
    Elif [ $count_i -lt 1 ]; then
            echo "Error: Copying usually involves a destination. Exiting." # you put one item and no destination
    fi

#reset the counter and grab content of arguments, aka: switches and item paths
n=1
while true; do
        eval input="\$$n" #input=$1,$2,$3,etc...
        in_=${input::1} #first letter of $input

        if [ "$in_" = "-" ]; then
            if [ "$input" = "--rename" ]; then 
                rename=true #my custom switches
            Elif [ "$input" = "--strict" ]; then 
                easy_going=false #exit script if even one of the non-destinations item is not found
            Elif [ "$input" = "--relaxed" ]; then 
                easy_going=true #continue script if at least one of the non-destination items is found
            Elif [ "$input" = "--quiet" ]; then 
                verbal=""
            else
                #m=$(expr $m + 1);eval switch$m="$input" #input is a switch, if it's not one of the above, assume it belongs to cp.
                switch_list="$switch_list \"$input\""
            fi                                  
        Elif ! [ -z "$input" ]; then #if it's not a switch and input is not empty, it's a path
                i=$(expr $i + 1)
                if [ ! -f "$input" -a ! -d "$input" -a "$i" -le "$count_i" ]; then 
                    err=$(expr $err + 1 ); error_list="$error_list\npath does not exit: \"b\""
                else
                    if [ "$i" -le "$count_i" ]; then 
                        eval item$i="$input" 
                        item_list="$item_list \"$input\""
                    else
                        destination="$input" #destination is last items entered
                    fi
                fi
        else
            i=0
            m=0
            n=1                     
            break
        fi      
        n=$(expr $n + 1)
done

#error condition: some or all item(s) being copied don't exist. easy_going: continue if at least one item exists, warn about rest, not easy_going: exit.
#echo "err=$err count_i=$count_i"
if [ "$easy_going" != true -a $err -gt 0 -a $err != $count_i ]; then 
    echo "Some of the paths you entered are incorrect. Script is running in strict mode and will therefore exit."
    echo -e "Bad Paths: $err $error_list"
    exit
fi

if [ $err = $count_i ]; then
    echo "ALL THE PATHS you have entered are incorrect! Exiting."
    echo -e "Bad Paths: $err $error_list"
fi

#one item to one destination:
#------------------------------
#assumes that destination is folder, it does't exist, it will create it. (so copying /a/b/c/d/firefox to /e/f/firefox will result in /e/f/firefox/firefox
#if -rename switch is given, will assume that the top element of destination path is the new name for the the item being given.

#multi-item to single destination:
#------------------------------
#assumes destination is a folder, gives error if it exists and it's a file. -rename switch will be ignored.

#ERROR CONDITIONS: 
# - multiple items being sent to a destination and it's a file.
# - if -rename switch was given and multiple items are being copied, rename switch will be ignored (easy_going). if not easy_going, exit.
# - rename option but source is folder, destination is file, exit.
# - rename option but source is file and destination is folder. easy_going: option ignored.

if [ -f "$destination" ]; then
    if [ $count_i -gt 1 ]; then 
        echo "Error: You've selected a single file as a destination and are copying multiple items to it. Exiting."; exit
    Elif [ -d "$item1" ]; then
        echo "Error: Your destination is a file but your source is a folder. Exiting."; exit
    fi
fi
if [ "$rename" = true ]; then
    if [ $count_i -gt 1 ]; then
        if [ $easy_going = true ]; then
            echo "Warning: you choose the rename option but are copying multiple items. Ignoring Rename option. Continuing."
        else
            echo "Error: you choose the rename option but are copying multiple items. Script running in strict mode. Exiting."; exit
        fi
    Elif [ -d "$destination" -a -f "$item1" ]; then
        echo -n "Warning: you choose the rename option but source is a file and destination is a folder with the same name. "
        if [ $easy_going = true ]; then
            echo "Ignoring Rename option. Continuing."
        else
            echo "Script running in strict mode. Exiting."; exit
        fi
    else
        dest_jr=$(dirname "$destination")
        if [ -d "$destination" ]; then item_list="$item1/*";fi
        mkdir -p "$dest_jr"
    fi
else
    mkdir -p "$destination"
fi

eval cp $switch_list $verbal $item_list "$destination"

cp_err="$?"
if [ "$cp_err" != 0 ]; then 
    echo -e "Something went wrong with the copy operation. \nExit Status: $cp_err"
else 
    echo "Copy operation exited with no errors."
fi

exit
1
thebunnyrules

同じ問題がありました。私のアプローチは、ファイルをアーカイブのようにtarすることでした。

tar cf your_archive.tar file1 /path/to/file2 path/to/even/deeper/file3

tarは、ファイルをアーカイブ内の適切な構造に自動的に保存します。走れば

tar xf your_archive.tar

ファイルは目的のディレクトリ構造に抽出されます。

1
Chris
rsync file /path/to/copy/file/to/is/very/deep/there

あなたが正しい種類のrsyncを持っているなら、これはうまくいくかもしれません。

0
danuker

Help_asapとspongemanが上記で提案したように、「install」コマンドを使用してファイルを既存のディレクトリにコピーするか、新しい宛先ディレクトリがまだない場合は作成します。

オプション1 install -D filename some/deep/directory/filename
ファイルを新規または既存のディレクトリにコピーし、ファイル名にデフォルトの755権限を付与します

オプション2 install -D filename -m640 some/deep/directory/filename
オプション1に従いますが、ファイル名に640の許可を与えます。

オプション3 install -D filename -m640 -t some/deep/directory/
オプション2に従いますが、ターゲットファイル名をターゲットディレクトリにターゲットするため、ソースとターゲットの両方にファイル名を書き込む必要はありません。

オプション4 install -D filena* -m640 -t some/deep/directory/
オプション3に従いますが、複数のファイルにワイルドカードを使用します。

Ubuntuでうまく機能し、2つのステップ(ディレクトリ作成とファイルコピー)を1つのステップに結合します。

0
Dig

単純な

cp -a * /path/to/dst/

トリックをするべきです。

0
Brad D.

ソースから存在しないパスにコピーする

mkdir –p /destination && cp –r /source/ $_

注:このコマンドはすべてのファイルをコピーします

すべてのフォルダとその内容をコピーするためのcp –r

$_は、最後のコマンドで作成された宛先として機能します

0