Linux: yoksa hedef dizini kopyalayın ve oluşturun


348

Hedef dizini yoksa bir komut (veya muhtemelen cp için bir seçenek) istiyorum.

Misal:

cp -? file /path/to/copy/file/to/is/very/deep/there


5
Kabul edilen cevap yanlış (cp için böyle bir seçenek var). Paul Whipp'in ikinci cevabına bakınız.
Ronenz

1
@Ronenz, GNU'da bahsetmeye değer bir seçenek olduğunu kabul ediyorum cp(Paul Whipp tarafından yanıtlandığı gibi), ancak OP'nin talep ettiği kadar genel değil. Bu kopyalayabilirsiniz a/foo/bar/fileiçin b/foo/bar/fileancak kopyalayamazsınız fileiçin b/foo/bar/file. (yani yalnızca ilk dosya durumunda zaten var olan üst dizinleri oluşturmak için çalışır, ancak rastgele dizinler oluşturamaz)
Sudo Bash

Bu cp için güzel bir özellik isteği gibi görünüyor. Yakın gelecekte böyle bir komut satırı anahtarını alma şansımız var mı?
Arnaud

Yanıtlar:


328
mkdir -p "$d" && cp file "$d"

(böyle bir seçenek yoktur cp).


62
Ihtiyacınız olduğunu sanmıyorum test -d: mkdir -pdizin zaten olsa bile hala başarılı.
ephemient

Üzgünüz, doğru. Ama sonra, test bir bash yerleşik olabilir (özellikle [[-d "$ d"]] olarak
yazılırsa

1
mkdir
BusyBox'ta

7
@holms, $dmuhtemelen söz konusu dizini içeren bir değişkendir. Yani, üç kez tekrarlamanız gerekiyorsa, önce değişkene atayabilirsiniz.
Michael Krelin - hacker

237

Aşağıdakilerin her ikisi de doğruysa:

  1. GNU sürümünü kullanıyorsunuz cp(örneğin Mac sürümünü kullanmıyorsunuz ) ve
  2. Varolan bir dizin yapısından kopyalama yapıyorsunuz ve yeniden oluşturulmasına ihtiyacınız var

sonra bunu --parentsbayrağı ile yapabilirsiniz cp. Bilgi sayfasından (görüntülenebilir de http://www.gnu.org/software/coreutils/manual/html_node/cp-invocation.html#cp-invocation ile ya info cpya 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.

Misal:

/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

4
Bu Mac OS X üzerinde çalışmaz, bu yüzden sanırım Linux'a özgüdür.
olt

7
Gnu'ya özgü diyebilirim. Eğer varsa macportscoreutils ve onun yükleyebilirsiniz gcp.
Michael Krelin - hacker

16
Adam, linux her şeye
Ziggy

19
Aradığım cevap (ve bu yüzden benim önerdiğim cevap) olmasına rağmen, bu biraz farklı bir sorunun cevabı gibi hissediyorum. Sanırım bu, hedef üst dizinlerin başlangıç ​​üst dizinleri ile aynı olmasını istediğinizi varsayar, bu da muhtemelen bunu okuyan çoğu insanın ilginç olduğu kullanım durumudur.
David Winiecki

7
Bu, 'bar' dizininin zaten var olduğunu varsayar, ancak orijinal soru, hedef olarak sağlandığında ve zaten mevcut olmadığında 'bar'ın otomatik olarak nasıl oluşturulacağını ima eder. Bunun cevabı @ MichaelKrelin-hacker tarafından sağlanır.
thdoan

70

Kısa cevap

Kopyalamak myfile.txtiçin şunu /foo/bar/myfile.txtkullanın:

mkdir -p /foo/bar && cp myfile.txt $_

Bu nasıl çalışıyor?

Bunun birkaç bileşeni var, bu yüzden tüm sözdizimini adım adım ele alacağım.

Mkdir yarar, POSIX standardında belirtildiği şekilde , dizinleri yapar. -pArgüman, docs başına neden olur mkdir için

Eksik ara yol adı bileşenleri oluşturma

yani arama yaparken mkdir -p /foo/bar, mkdir oluşturulur /foo ve /foo/bar eğer /foozaten mevcut değilse. (Yok -p, bunun yerine bir hata atar.

&&Liste operatörü, belgelenen olarak POSIX standardı (veya Bash kılavuzunda isterseniz), etkiye sahiptir cp myfile.txt $_yalnızca çalıştırılmaktadır mkdir -p /foo/barbaşarıyla yürütür. Bu, cpkomutun mkdirbaşarısız olmasının birçok nedeninden biri nedeniyle başarısız olursa çalıştırmaya çalışmadığı anlamına gelir .

Son $_olarak, ikinci argüman cpolarak geçiyoruz, bir değişkende saklamak zorunda kalmadan uzun argümanları (dosya yolları gibi) tekrarlamaktan kaçınmak için kullanışlı olabilecek bir "özel parametre" dir. Başına Bash kılavuzda , bu:

önceki komutun son argümanına genişler

Bu durumda, /foo/barbiz geçtik mkdir. Böylece cpkomut , yeni oluşturulan dizine cp myfile.txt /foo/barkopyalanacak myfile.txtşekilde genişler /foo/bar.

Not $_olan olup POSIX standart bir parçası , bu nedenle, teorik olarak, bir UNIX varyantı Bu konstraktı desteklemeyen bir kabuk olabilir. Ancak, desteklemeyen hiçbir modern mermi bilmiyorum $_; kesinlikle Bash, Dash ve zsh hepsi yapar.


Son bir not: Bu cevabın başında verdiğim komut, dizin adlarınızın boşluk içermediğini varsayar. Boşluk içeren adlarla uğraşıyorsanız, bunları farklı kelimelerle mkdirveya için farklı argümanlar olarak ele alınmaz cp. Yani komutunuz aslında şöyle görünecektir:

mkdir -p "/my directory/name with/spaces" && cp "my filename with spaces.txt" "$_"

17
Bir yana: Yığın Taşması'ndaki Bash veya Unix kabuk komutları hakkındaki soruların ayrıntılarındaki eksiklikten genellikle hayal kırıklığına uğradım. Ben burada ters aşırı gitmek için çalıştı ve her ikisi de sözdiziminin her detayını açıklamak ve bu şeylerin nasıl çalıştığı hakkında belgeler bulmak için uygun yerlere bağlantı. Umarım bu en azından bazı insanlara yardımcı olur. Ayrıca, şu durumlarda kredi: Bu yaklaşımı bu yanıttan yinelenen bir soruya çektim: stackoverflow.com/a/14085147/1709587
Mark Amery

Daha önce hiç $ _ görmedim. bu neye karşılık gelir? son komutta kullanılan klasör?
thebunnyrules

10
@thebunnyrules Ama ... ama ... "Bu nasıl çalışır?" cevabımda, tam olarak sormuş olduğunuz soruya ayrılmış birden fazla paragraf içeren bölüm. Göz ardı edildiğini ve sevilmediğini hissediyor. :(
Mark Amery

Bunun için üzgünüm. Kesinlikle haklısın. Cevabınızdaki diğer her şeyi zaten biliyordum, bu yüzden hızlıca taradım. Daha dikkatli okumalıydım.
thebunnyrules

Vay canına, bu cevaba gerçekten çok fazla iş yaptınız ... Bunun için teşekkürler. Öğrenmek güzeldi: $ _. Bundan sonra kendimi biraz kullandığımı görebiliyorum. Tüm süreci otomatikleştirmek ve bu konuya koymak için bir senaryo yazdım. Bir
deneyin

39

Böyle eski bir soru, ama belki alternatif bir çözüm önerebilirim.

Sen kullanabilirsiniz installDosyanızı kopyalamak için program ve "anında" hedef yolu oluşturmak.

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

Bununla birlikte, dikkate alınması gereken bazı yönler vardır:

  1. yalnızca hedef yolu değil , hedef dosya adını da belirtmeniz gerekir
  2. hedef dosya çalıştırılabilir olacak (en azından testlerimde gördüğüm kadarıyla)

-mHedef dosyada izinleri ayarlama seçeneğini ekleyerek # 2'yi kolayca değiştirebilirsiniz (örnek: -m 664hedef dosyayı rw-rw-r--, yeni bir dosya oluşturmak gibi izinlerle oluşturur touch).


Ve işte burada ilham aldığım cevaba utanmaz bağlantı =)


Gerçekten benim kullanım ama serin hile için işe yaramadı! Aklımda tutacağım.
thebunnyrules

bu komutun 755( rwxr-xr-x) izinsiz hedef dosyaları oluşturduğunu unutmayın , bu da büyük olasılıkla istenmez. -manahtarla başka bir şey belirtebilirsiniz , ancak sadece dosya izinlerini tutmanın bir yolunu bulamadım :(
törzsmókus

19

İstediğiniz şeyi yapan kabuk işlevi, dosyanın içinde yaşamak için bir delik açtığı için "gömme" kopyası olarak adlandırılır:

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

1
dirname $2Çok tırnak işareti olmalı
Tarnay Kálmán

4
@Kalmi, uygun fiyat teklifi $2için argüman olarak da alıntı yapmak istersiniz dirname. Gibi mkdir -p "$(dirname "$2")". Bu alıntı olmadan $2içinde cpyararsız;)
Michael Krelin - Hacker

3
Bu cevabın $ 2'nin hedef dizin olmadığını varsayarsınız, aksi takdirde fonksiyon gövdesi olarak "mkdir -p" $ 2 "&& cp" $ 1 "" $ 2 "
istersiniz

@ User467257'nin işaret ettiği gibi bunun bir hata olduğunu düşünüyorum. $ 2 bu senaryoda hedef dir ve hedef dosya arasında belirsiz olduğundan, bu düzeltilebilir olduğunu sanmıyorum. Buna hitap eden alternatif bir cevap gönderdim.
Zengin

@Rich, bunun düzeltilebilir olmadığını kabul etmiyorum. Aşağıdaki dosyalar ve dizinler için iyi çalışır: mkdir -p dirname "$2"&& cp -r "$ 1" "$ 2"; Etraftaki backtick'lerin görünmediğini unutmayın, dirname $2çünkü SO bunları kod işaretleri olarak ayrıştırır.
Yury

11

İşte bunu yapmanın bir yolu:

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

dirnamesize hedef dizinin veya dosyanın üst öğesini verecektir. mkdir -p `dirname ...` daha sonra cp -r'yi çağırdığınızda doğru temel dizinin yerinde olmasını sağlayarak bu dizini oluşturur.

Bu fazla parçaların avantajı, hedef yoldaki son öğenin bir dosya adı olduğu durumda çalışmasıdır.

Ve OS X üzerinde çalışacak.


6

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


1
aslında, hiç de aynı değil. dosya adını iki kez geçirmenizi gerektirir (bu nedenle '-t' bayrağı) ve dosyadaki yürütme bitlerini ayarlar (dolayısıyla '-m' örneği). Onun gerektiğini değil benim yapar oysa - aslında soruya cevap vermediğinden, üst cevabı.
Spongman

1
Bu tek bir dosya için çalışır. Sadece thereson dosya değil son dosya olduğunu dikkate alın .
kvantour

6

yukarıdaki cevaplara tüm saygımla rsync'i aşağıdaki gibi kullanmayı tercih ederim:

$  rsync -a directory_name /path_where_to_inject_your_directory/

misal:

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

Ben denedim ve bu sonuca gitmek: rsync -a boğa / bazı / çember / ti / doop / ploop SONUÇ rsync: mkdir "/ bazı / çember / ti / doop / ploop" başarısız oldu: Böyle bir dosya veya dizin (2) rsync hatası yok : main.c (675) dosyasında IO (kod 11) hatası [Alıcı = 3.1.2]
thebunnyrules

@thebunnyrules yolu pwd komutuyla kontrol etmeniz (bakınız: [link] ( access.redhat.com/documentation/en-US/Red_Hat_Enterprise_Linux/… )) ve talimatınıza doğru şekilde eklemeniz gerekir
marcdahan

Önerinizi doğru bir şekilde anlarsam, komutumda sadece dosya adı: bull? Şey, hata kaynağı (boğa) değil, rsync hedef dizin oluşturma ile. Bana göre rsync bir mkdir -p değil, basit bir mkdir kullanır.
thebunnyrules

1
Bu arada harika bir yardımcı program, komut dosyalarında kullanmaya başlayabilirim. Tavsiye ettiğiniz için teşekkürler.
thebunnyrules

1
rsync daha tahmin edilebilir cp. Örneğin, eğer cp /from/somedir /to-dircp /to-dirzaten varsa farklı davranırsa : Eğer /to-dirvarsa, o cpzaman yaratacaktır /to-dir/some-dir, aksi takdirde sadece /from/somediriçine yerleştirilir /to-dir. Ancak, rsyncyani, durumunda aynı şekilde davranır /to-dir/some-diredecek hep oluşturulabilir.
JoeAC

5

Sadece bir satırda devam etmek ve eksiksiz bir çalışma çözümü vermek. Dosyanızı yeniden adlandırmak istiyorsanız dikkatli olun, mkdir yoluna temiz bir dizin yolu sağlamanız gerekir. $ fdst dosya veya dizin olabilir. Sonraki kod her durumda çalışmalıdır.

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

veya bash'a özgü

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

Teşekkürler! Buna ihtiyacım vardı. Benim durumumda hedef dizin olup olmadığını bilmiyorum ve dizin var ya da değil bu kod bana o zaman yoksa ben güvenle oluşturabilirsiniz ana dizin verir
xbmono

4

Aşağıdakileri .bashrc'nize ekleyin, gerekirse ince ayar yapın. Ubuntu'da çalışıyor.

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

Örneğin, 'test' dosyasını 'd' Kullanılacak hedef dizine kopyalamak istiyorsanız,

mkcp test a/b/c/d

mkcp ilk olarak hedef dizinin var olup olmadığını kontrol eder, eğer değilse yapar ve kaynak dosya / dizini kopyalar.


Diğerleri diğer yanıta yorum yaptığı gibi, aramadan önce dizinin var olup olmadığını test etmenize gerek yoktur mkdir -p. Var olsa bile başarılı olacaktır.
bfontaine

3

Bu benim için yapar

cp -vaR ./from ./to

Katılıyorum. In man cp:-R If source_file designates a directory, cp copies the directory and the entire subtree connected at that point.
borracciaBlu

3

CP için tam olarak bunu yapmayı amaçlayan CP (büyük harfleri not et) adlı bir destek komut dosyası yazdım. Komut dosyası, girdiğiniz yoldaki hataları (hedef olan sonuncusu hariç) kontrol eder ve her şey yolundaysa, kopyaya başlamadan önce hedef yol oluşturmak için bir mkdir -p adımı yapar. Bu noktada normal cp yardımcı programı devreye girer ve CP ile kullandığınız tüm anahtarlar (-r, -p, -rpL gibi doğrudan cp'ye bağlanır). Senaryomu kullanmadan önce, anlamanız gereken birkaç şey var.

  • Buradaki tüm bilgilere CP --help yapılarak erişilebilir. CP --help-all include'ın cp's anahtarları.
  • Normal cp, hedef yolu bulamazsa kopyayı yapmaz. CP'li yazım hataları için böyle bir güvenlik ağınız yok. Hedefiniz oluşturulacaktır, bu nedenle hedefinizi / usrr / share / icons veya / usr / share / icon olarak yanlış yazarsanız, oluşturulacak olan budur.
  • normal cp, var olan yoldaki davranışını modelleme eğilimindedir: cp / a / b / c / d, d'nin var olup olmamasına göre değişir. d varolan bir klasörse, cp b'yi içine kopyalar ve / c / d / b yapar. D yoksa, b c'ye kopyalanacak ve d olarak yeniden adlandırılacaktır. D varsa ancak bir dosya ve b bir dosyaysa, b'nin kopyası tarafından üzerine yazılır. C yoksa, cp kopyayı yapmaz ve çıkar.

CP, mevcut yollardan ipucu alma lüksüne sahip değildir, bu nedenle çok sağlam davranış kalıplarına sahip olmak zorundadır. CP, kopyaladığınız öğenin hedef yolda bırakıldığını ve hedefin kendisi olmadığını varsayar (diğer bir deyişle, kaynak dosyanın / klasörün yeniden adlandırılmış bir kopyası). Anlamı:

  • D bir klasörse "CP / a / b / c / d" / c / d / b ile sonuçlanır
  • "CP / a / b / c / b", / c / b içindeki bir klasörse / c / b / b ile sonuçlanır.
  • Hem b hem de d dosyalarsa: CP / a / b / c / d / c / d ile sonuçlanır (burada d, b'nin bir kopyasıdır). Aynı durumda CP / a / b / c / b için aynıdır.

Bu varsayılan CP davranışı "--rename" anahtarıyla değiştirilebilir. Bu durumda,

  • "CP --rename / a / b / c / d", b'yi / c'ye kopyalar ve kopyayı d olarak yeniden adlandırır.

Birkaç kapanış notu: CP gibi, CP de bir seferde birden fazla öğeyi kopyalayabilir ve listelenen son yol hedef olarak kabul edilir. Tırnak işaretleri kullandığınız sürece boşluklu yolları da işleyebilir.

CP, koyduğunuz yolları kontrol eder ve kopyayı yapmadan önce var olduklarından emin olur. Katı modda (--strict anahtarı ile kullanılabilir), kopyalanan tüm dosya / klasörlerin var olması veya kopya olmaması gerekir. Rahat modda (--relaxed), listelediğiniz öğelerden en az biri varsa kopyalama devam eder. Rahat mod varsayılan moddur, modu anahtarlar aracılığıyla geçici olarak veya komut dosyasının başlangıcında easy_going değişkenini ayarlayarak kalıcı olarak değiştirebilirsiniz.

Nasıl yükleyeceğiniz aşağıda açıklanmıştır:

Root olmayan bir terminalde şunları yapın:

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

Gedit'te CP yardımcı programını yapıştırın ve şunları kaydedin:

#!/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

Bu belki de aşırıya kaçan, ama oldukça verimli, beklendiği gibi çalışıyor, teşekkür ederim efendim.
Xedret

2

cp birden çok kullanıma sahiptir:

$ 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'un cevabı

cp SOURCE DEST

tarzı cp, ama kullanırsanız yanlış bir şey yapar

cp SOURCE... DIRECTORY/

tarzı cp.

Ben "DEST" bu kullanımda (yani hedef dizinin henüz mevcut değil) bir eğik çizgi olmadan belirsiz olduğunu düşünüyorum, bu belki de neden cphiç bir seçenek eklemedi.

Yani burada dest dir üzerinde bir eğik çizgi zorlayan bu fonksiyon benim sürümü:

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
}

2

Sadece aynı sorunu vardı. Benim yaklaşım sadece böyle bir arşive dosyaları katran oldu:

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

tar dosyaları otomatik olarak arşiv içindeki uygun yapıda saklar. Eğer koşarsan

tar xf your_archive.tar

dosyalar istenen dizin yapısına çıkarılır.


1

Yukarıda help_asap ve spongeman tarafından önerildiği gibi, dosyaları mevcut dizinlere kopyalamak veya henüz yoksa yeni hedef dizinler oluşturmak için 'install' komutunu kullanabilirsiniz.

1. Seçenek install -D filename some/deep/directory/filename
dosyayı yeni veya mevcut bir dizine kopyalar ve dosya adı varsayılan 755 izinlerini verir

Seçenek 2'ye install -D filename -m640 some/deep/directory/filename
göre Seçenek 2, ancak dosya adı 640 izinleri verir.

Seçenek install -D filename -m640 -t some/deep/directory/
2'ye göre Seçenek 3, ancak dosya adını hedef dizine hedefler, böylece dosya adının hem kaynakta hem de hedefte yazılması gerekmez.

Seçenek 3 install -D filena* -m640 -t some/deep/directory/
uyarınca Seçenek 4, ancak birden çok dosya için joker karakter kullanır.

Ubuntu'da güzel çalışıyor ve iki adımı (dizin oluşturma sonra dosya kopyalama) tek bir adımda birleştiriyor.


1

Bu çok geç ama bir yerlerde çaylak yardımcı olabilir. 'Otomatik' klasör oluşturmanız gerekiyorsa rsync en iyi arkadaşınız olmalıdır. rsync / path / to / sourcefile / path / to / tragetdir / thatdoestexist /


0
rsync file /path/to/copy/file/to/is/very/deep/there

Doğru türde bir şey varsa, bu işe yarayabilir rsync.


Bu benim için işe yaramıyor - dest dizininde "Böyle bir dosya veya dizin yok" var
Zengin

0

Kaynaktan mevcut olmayan bir yola kopyalama

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

NOT: bu komut tüm dosyaları kopyalar

cp –r tüm klasörleri ve içeriğini kopyalamak için

$_ Son komutta oluşturulan hedef olarak çalış


0

Basit

cp -a * /path/to/dst/

hile yapmalı.


Öyle değil. '/ Path / to / dst /' önceden var olmalıdır.
Shital Shah

-1

Diyelim ki böyle bir şey yapıyorsun

cp file1.txt A / B / C / D / file.txt

A / B / C / D henüz mevcut olmayan dizinlerdir

Olası bir çözüm aşağıdaki gibidir

DIR=$(dirname A/B/C/D/file.txt)
# DIR= "A/B/C/D"
mkdir -p $DIR
cp file1.txt A/B/C/D/file.txt

umarım yardımcı olur!

Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.