Linux: скопируйте и создайте каталог назначения, если он не существует

246

Мне нужна команда (или, возможно, опция cp), которая создает целевой каталог, если он не существует.

Пример:

cp -? file /path/to/copy/file/to/is/very/deep/there
  • 4
    Принятый ответ неверен (есть такая опция для cp). Смотрите второй ответ Пола Уиппа.
Показать ещё 1 комментарий
Теги:
cp

17 ответов

207
Лучший ответ
mkdir -p "$d" && cp file "$d"

(нет такой опции для cp).

  • 58
    Я не думаю, что вам нужен test -d : mkdir -p все еще успешно, даже если каталог уже существует.
  • 0
    упс, верно. Но тогда test может быть встроенным в bash (особенно если он написан как [[-d "$ d"]]), а mkdir не может ;-)
Показать ещё 4 комментария
201

Если выполняется одно из следующих условий:

  • Вы используете версию GNU cp (а не, например, версию Mac), и
  • Вы копируете какую-либо существующую структуру каталогов, и вам просто нужно ее воссоздать

то вы можете сделать это с помощью флага --parents cp. На информационной странице (можно просмотреть 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
  • 3
    Это не работает в Mac OS X, так что я думаю, что это специфично для Linux.
  • 5
    Я бы сказал, что GNU-конкретных. Если у вас есть macports вы можете установить coreutils и его gcp .
Показать ещё 6 комментариев
36

Краткий ответ

Чтобы скопировать myfile.txt в /foo/bar/myfile.txt, используйте:

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

Как это работает?

Вот несколько компонентов для этого, поэтому я покрою весь синтаксис шаг за шагом.

Утилита mkdir, как указано в стандарте POSIX, создает каталоги. Аргумент -p для каждого документа будет вызывать mkdir

Создайте любые отсутствующие компоненты промежуточного пути

означает, что при вызове mkdir -p /foo/bar mkdir создаст /foo и /foo/bar, если /foo еще не существует. (Без -p вместо этого он выдает ошибку.

Оператор списка &&, как описано в стандарте POSIX (или Bash manual если вы предпочитаете), имеет эффект, что cp myfile.txt $_ запускается только в том случае, если mkdir -p /foo/bar выполняется успешно. Это означает, что команда cp не будет пытаться выполнить, если mkdir не удалось выполнить одну из многих причин, по которым она может потерпеть неудачу.

Наконец, $_ мы передаем как второй аргумент cp - это "специальный параметр", который может быть полезен для избежания повторения длинных аргументов (например, путей к файлам) без необходимости их хранения в переменной. Per руководство Bash, оно:

расширяется до последнего аргумента предыдущей команды

В этом случае, что /foo/bar мы перешли к mkdir. Таким образом, команда cp расширяется до cp myfile.txt /foo/bar, которая копирует myfile.txt во вновь созданный каталог /foo/bar.

Обратите внимание, что $_ не является частью стандарта POSIX, поэтому теоретически вариант Unix может иметь оболочку, которая не поддерживает эту конструкцию. Однако я не знаю никаких современных оболочек, которые не поддерживают $_; конечно Bash, Dash и zsh все делают.


Последнее замечание: команда, которую я дал в начале этого ответа, предполагает, что имена ваших каталогов не имеют пробелов. Если вы имеете дело с именами с пробелами, вам нужно процитировать их так, чтобы разные слова не рассматриваются как разные аргументы для mkdir или cp. Таким образом, ваша команда будет выглядеть так:

mkdir -p "/my directory/name with/spaces" && cp "my filename with spaces.txt" "$_"
  • 10
    Кроме того: я, как правило, разочарован отсутствием подробностей в вопросах о командах оболочки Bash или Unix в Stack Overflow, которые часто выбрасывают сложную и чрезвычайно плотную однострочную строку, которую трудно распознать, если вы еще не являетесь мастером Unix. Я попытался пойти на противоположную крайность здесь и объяснил каждую деталь синтаксиса и ссылку на соответствующие места, чтобы найти документацию о том, как этот материал работает. Я надеюсь, что это поможет, по крайней мере, некоторым людям. Кроме того, кредит в должное время: я использовал этот подход из этого ответа на двойной вопрос: stackoverflow.com/a/14085147/1709587
  • 0
    Я никогда не видел $ _ раньше. что оно относится к? папка, использованная в последней команде?
Показать ещё 4 комментария
31

Такой старый вопрос, но, возможно, я могу предложить альтернативное решение.

Вы можете использовать программу install для копирования вашего файла и создания пути назначения "на лету".

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

Есть несколько аспектов, которые следует принять во внимание:

  • вам нужно указать также имя целевого файла, а не только путь назначения
  • целевой файл будет исполняемым (по крайней мере, насколько я видел из своих тестов)

Вы можете легко изменить # 2, добавив параметр -m для установки разрешений для целевого файла (пример: -m 664 создаст целевой файл с разрешениями rw-rw-r--, так же, как создание нового файла с touch).


И вот это бесстыдная ссылка на ответ, на который я был вдохновлен =)

  • 0
    Это отлично работает на Amazon Linux. Спасибо!
  • 0
    Не очень подходит для моего использования, но крутой трюк! Я буду иметь в виду.
Показать ещё 1 комментарий
15

Функция оболочки, которая делает то, что вы хотите, называя ее копией "похоронить", потому что она выкапывает отверстие для файла, в котором он находится:

bury_copy() { mkdir -p `dirname $2` && cp "$1" "$2"; }
  • 1
    Вы должны иметь кавычки вокруг dirname $2 тоже
  • 4
    @Kalmi, для правильной цитаты вы также хотели бы указать $2 в качестве аргумента для dirname . Например, mkdir -p "$(dirname "$2")" . Без этого цитирование $2 в cp бесполезно;)
Показать ещё 4 комментария
6

Вот один из способов сделать это:

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 будет создан правильный базовый каталог.

Преимущество этих над -parents заключается в том, что он работает для случая, когда последний элемент в пути назначения является именем файла.

И он будет работать на OS X.

5

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

  • 0
    на самом деле, это совсем не одно и то же. Он требует, чтобы вы дважды передали имя файла (отсюда и флаг '-t'), и он установил биты выполнения в файле (отсюда и пример '-m'). его не должно быть главным ответом, так как он на самом деле не отвечает на вопрос - в то время как мой ответ.
  • 0
    Это работает для одного файла. Просто учтите, что there конечный файл, а не последний подкаталог.
2

со всем уважением к ответам выше, я предпочитаю использовать rsync следующим образом:

$  rsync -a directory_name /path_where_to_inject_your_directory/

пример:

$ rsync -a test /usr/local/lib/
  • 0
    Я попытался и получаю следующий результат: rsync -a bull / some / hoop / ti / doop / ploop РЕЗУЛЬТАТ rsync: mkdir "/ some / hoop / ti / doop / ploop" завершилась неудачей: нет такого файла или каталога (2) Ошибка rsync : ошибка в файле ввода-вывода (код 11) в main.c (675) [Receiver = 3.1.2]
  • 0
    @thebunnyrules, вам нужно проверить путь с помощью команды pwd (посмотрите: [link] ( access.redhat.com/documentation/en-US/Red_Hat_Enterprise_Linux/… )) и правильно вставьте его в вашу инструкцию
Показать ещё 2 комментария
1

Просто чтобы возобновить и дать полное рабочее решение, в одну строку. Будьте осторожны, если вы хотите переименовать ваш файл, вы должны включить способ предоставить чистый путь к каталогу mkdir. $ fdst может быть файлом или каталогом. Следующий код должен работать в любом случае.

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

или специфичный для bash

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

Это делает это для меня

cp -vaR ./from ./to
1

Просто добавьте следующее в свой .bashrc, настройте, если вам нужно. Работает в Ubuntu.

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

например Если вы хотите скопировать файл "test" в целевой каталог "d", Используйте

mkcp test a/b/c/d

mkcp сначала проверит, существует ли каталог назначения, или нет, если нет, то сделайте это и скопируйте исходный файл/каталог.

  • 0
    Как другие прокомментировали другой ответ, вам не нужно проверять, существует ли каталог, прежде чем вызывать mkdir -p . Это удастся, даже если он уже существует.
1

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 dir:

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
}
0

Просто была такая же проблема. Мой подход состоял в том, чтобы просто заархивировать файлы в архив следующим образом: tar cf your_archive.tar file1/path/to/file2 path/to/even/deeper/file3

tar автоматически сохраняет файлы в соответствующей структуре в архиве. Если вы запустите tar xf your_archive.tar, файлы будут извлечены в нужную структуру каталогов.

0

Я написал вспомогательный скрипт для cp, называемый CP (примечание заглавными буквами), который предназначен именно для этого. Скрипт проверит наличие ошибок в пути, который вы указали (за исключением последнего, который является пунктом назначения), и, если все в порядке, он выполнит шаг mkdir -p, чтобы создать путь назначения перед началом копирования. В этот момент обычная утилита cp вступает во владение, и все переключатели, которые вы используете с CP (например, -r, -p, -r pL, передаются напрямую в cp). Прежде чем использовать мой сценарий, необходимо кое-что понять.

  • Вся информация здесь может быть доступна с помощью CP --help. CP --help - включает в себя 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/a/b/c/d" приведет к /c/d/b, если d - папка
  • "CP/a/b/c/b" приведет к /c/b/b, если b в /c/b является папкой.
  • Если оба b и d являются файлами: CP/a/b/c/d приведет к /c/d (где d - копия b). То же самое для CP/a/b/c/b в тех же обстоятельствах.

Это поведение CP по умолчанию можно изменить с помощью переключателя "- -r ename". В этом случае предполагалось, что

  • "CP - -r ename/a/b/c/d" копирует b в /c и переименовывает копию в d.

Несколько заключительных замечаний: Как и в случае cp, CP может копировать несколько элементов одновременно, причем последний путь в списке считается местом назначения. Он также может обрабатывать пути с пробелами, если вы используете кавычки.

CP проверит введенные вами пути и удостоверится, что они существуют, прежде чем копировать. В строгом режиме (доступном через переключатель --strict) все копируемые файлы/папки должны существовать, иначе копирование не производится. В расслабленном режиме (- -r elaxed) копирование будет продолжено, если хотя бы один из перечисленных вами пунктов существует. Режим по умолчанию является расслабленным, вы можете изменить режим временно с помощью переключателей или навсегда, установив переменную easy_going в начале скрипта.

Вот как это установить:

В терминале, отличном от -r, выполните:

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 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 regular cp command (note uncapped letters). \n
    \n Script 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 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 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 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 not one of the above, assume it belongs to cp.
                switch_list="$switch_list \"$input\""
            fi                                  
        elif ! [ -z "$input" ]; then #if it not a switch and input is not empty, it 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 a file. -rename switch will be ignored.

#ERROR CONDITIONS: 
# - multiple items being sent to a destination and it 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
  • 0
    Это, возможно, излишнее, но довольно эффективное, работает как положено, спасибо, сэр.
0

Скажем, вы делаете что-то вроде

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

где A/B/C/D - это каталоги, которые еще не существуют

Возможное решение таково:

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

надеюсь, что это поможет!

0

Копировать из источника в несуществующий путь

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

ПРИМЕЧАНИЕ: эта команда копирует все файлы

cp –r для копирования всех папок и их содержимого

$_ работает как пункт назначения, который создается в последней команде

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

Это может сработать, если у вас есть правильный тип rsync.

  • 0
    Это не работает для меня - у меня есть "Нет такого файла или каталога" в директории dest

Ещё вопросы

Сообщество Overcoder
Наверх
Меню