web-dev-qa-db-fra.com

Linux: copier et créer le répertoire de destination s'il n'existe pas

Je veux une commande (ou probablement une option de cp) qui crée le répertoire de destination s'il n'existe pas.

Exemple:

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

(Il n'y a pas une telle option pour cp).

254

Si les deux conditions suivantes sont vraies:

  1. Vous utilisez la version GNU de cp (et non, par exemple, la version Mac), et
  2. Vous copiez à partir d'une structure de répertoire existante et vous avez simplement besoin de la recréer

vous pouvez alors faire cela avec le drapeau --parents de cp. À partir de la page d’information (consultable à http://www.gnu.org/software/coreutils/manual/html_node/cp-invocation.html#cp-invocation ou avec info cp ou 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.

Exemple:

/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
210
Paul Whipp

Réponse courte

Pour copier myfile.txt dans /foo/bar/myfile.txt, utilisez:

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

Comment cela marche-t-il?

Cela comporte quelques composants, je vais donc couvrir toute la syntaxe étape par étape.

L'utilitaire mkdir , spécifié dans la norme POSIX , permet de créer des répertoires. L'argument -p, conformément à la documentation, entraînera mkdir

Créez tout composant de chemin d'accès intermédiaire manquant

ce qui signifie que lorsque vous appelez mkdir -p /foo/bar, mkdir créera /foo et /foo/bar si /foo n'existe pas déjà. (Sans -p, il générera une erreur.

L'opérateur de liste &&, comme indiqué dans le standard POSIX (ou le manuel de Bash si vous préférez), a pour effet que cp myfile.txt $_ ne reçoit que exécuté si mkdir -p /foo/bar s'exécute avec succès. Cela signifie que la commande cp ne tentera pas de s'exécuter si mkdir échoue pour ne des nombreuses raisons pour lesquelles il peut échouer .

Enfin, le $_ que nous passons comme deuxième argument à cp est un "paramètre spécial" qui peut être pratique pour éviter de répéter des arguments longs (comme des chemins de fichiers) sans avoir à les stocker dans une variable. Per le manuel de Bash , il:

étend le dernier argument de la commande précédente

Dans ce cas, c'est le /foo/bar que nous avons passé à mkdir. Ainsi, la commande cp se développe en cp myfile.txt /foo/bar, qui copie myfile.txt dans le répertoire /foo/bar nouvellement créé.

Notez que $_ est ( ne fait pas partie du standard POSIX) , donc une variante Unix pourrait avoir un shell qui ne supporte pas cette construction. Cependant, je ne connais aucun shell moderne qui ne supporte pas $_; certainement Bash, Dash et zsh font tous.


Une dernière remarque: la commande que j'ai donnée au début de cette réponse suppose que vos noms de répertoire ne contiennent pas d'espaces. Si vous traitez des noms avec des espaces, vous devez les citer de manière à ce que les différents mots ne sont pas traités comme des arguments différents pour mkdir ou cp. Donc, votre commande ressemblerait en réalité à:

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

Une question si ancienne, mais je peux peut-être proposer une solution alternative.

Vous pouvez utiliser le programme install pour copier votre fichier et créer le chemin de destination "à la volée".

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

Il y a cependant certains aspects à prendre en compte:

  1. vous devez spécifier également le nom du fichier de destination , pas uniquement le chemin de destination
  2. le fichier de destination sera exécutable (au moins, d'après ce que j'ai vu de mes tests)

Vous pouvez facilement modifier le n ° 2 en ajoutant l'option -m pour définir les autorisations sur le fichier de destination (par exemple: -m 664 créera le fichier de destination avec les autorisations rw-rw-r--, comme si vous créiez un nouveau fichier. avec touch).


Et voici le lien sans vergogne vers la réponse qui m'a inspiré =)

33
help_asap

Fonction Shell qui fait ce que vous voulez, appelez-la une copie "enterrée", car elle creuse un trou pour le fichier:

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

Voici une façon de le faire:

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 vous donnera le parent du répertoire ou du fichier de destination. mkdir -p `dirname ...` créera alors ce répertoire en s'assurant que lorsque vous appelez cp -r, le bon répertoire de base est en place.

L'avantage de cette option --parents est que cela fonctionne dans le cas où le dernier élément du chemin de destination est un nom de fichier.

Et ça va marcher sur OS X.

8
Jamie McCrindle

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

5
Spongman

avec tout mon respect pour les réponses ci-dessus, je préfère utiliser rsync comme suit:

$  rsync -a directory_name /path_where_to_inject_your_directory/

exemple:

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

Juste pour reprendre et donner une solution de travail complète, dans une ligne. Faites attention si vous souhaitez renommer votre fichier, vous devez inclure un moyen de fournir un chemin de répertoire propre à mkdir. $ fdst peut être file ou dir. Le code suivant devrait fonctionner dans tous les cas.

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

ou bash spécifique

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

Ça le fait pour moi

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

Ajoutez simplement ce qui suit dans votre .bashrc, modifiez si vous en avez besoin. Fonctionne dans Ubuntu.

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

Si vous souhaitez copier le fichier "test" dans le répertoire de destination "d",

mkcp test a/b/c/d

mkcp va d'abord vérifier si le répertoire de destination existe ou non, sinon, créez-le et copiez le fichier/répertoire source.

1
SD.

cp a plusieurs usages:

$ 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.

La réponse de @ AndyRoss fonctionne pour le

cp SOURCE DEST

style de cp, mais fait la mauvaise chose si vous utilisez le

cp SOURCE... DIRECTORY/

style de cp.

Je pense que "DEST" est ambigu sans une barre oblique finale dans cette utilisation (c'est-à-dire lorsque le répertoire cible n'existe pas encore), ce qui explique peut-être pourquoi cp n'a jamais ajouté d'option pour cela.

Alors voici ma version de cette fonction qui impose une barre oblique de fin sur le répertoire de destination:

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

J'ai écrit un script de support pour cp, appelé CP (note majuscule), destiné à faire exactement cela. Le script recherchera les erreurs dans le chemin que vous avez entré (sauf le dernier qui est la destination) et si tout va bien, il effectuera une étape mkdir -p pour créer le chemin de destination avant de commencer la copie. À ce stade, l'utilitaire cp standard prend le relais et tous les commutateurs que vous utilisez avec CP (tels que -r, -p, -rpL sont directement acheminés vers cp). Avant d’utiliser mon script, vous devez comprendre certaines choses.

  • toutes les informations ici peuvent être consultées en faisant CP --help. CP --help-all include les commutateurs cp.
  • le cp régulier ne fera pas la copie s'il ne trouve pas le chemin de destination. Vous n'avez pas un tel filet de sécurité pour les fautes de frappe avec le CP. Votre destination sera créée. Par conséquent, si vous orthographiez mal votre destination sous/usrr/share/icons ou/usr/share/icon, c'est ce qui va être créé.
  • cp régulier a tendance à modéliser son comportement sur le chemin existant: cp/a/b/c/d variera selon que d existe ou non. si d est un dossier existant, cp y copiera b en faisant/c/d/b. Si d n'existe pas, b sera copié dans c et renommé d. Si d existe mais est un fichier et que b est un fichier, il sera écrasé par la copie de b. Si c n'existe pas, cp ne fait pas la copie et quitte.

Le CP n'a pas le luxe de s'inspirer des sentiers existants, il doit donc adopter des comportements très fermes. CP suppose que l'élément que vous copiez est déposé dans le chemin de destination et n'est pas la destination elle-même (c'est-à-dire une copie renommée du fichier/dossier source). Sens:

  • "CP/a/b/c/d" donnera le répertoire/c/d/b si d est un dossier
  • "CP/a/b/c/b" donnera/c/b/b si b dans/c/b est un dossier.
  • Si b et d sont tous deux des fichiers: CP/a/b/c/d donnera/c/d (où d est une copie de b). Même chose pour CP/a/b/c/b dans les mêmes circonstances.

Ce comportement par défaut du CP peut être modifié avec le commutateur "--rename". Dans ce cas, on suppose que

  • "CP --rename/a/b/c/d" copie b dans/c et renomme la copie en d.

Quelques remarques de clôture: Comme cp, le CP peut copier plusieurs éléments à la fois, le dernier chemin indiqué étant supposé être la destination. Il peut également gérer des chemins avec des espaces tant que vous utilisez des guillemets.

Le CP vérifiera les chemins que vous avez mis et s'assurera qu'ils existent avant de faire la copie. En mode strict (disponible via --strict switch), tous les fichiers/dossiers en cours de copie doivent exister ou aucune copie n’a lieu. En mode relâché (--relaxed), la copie se poursuivra si au moins un des éléments listés existe. Le mode Relaxé est le mode par défaut. Vous pouvez modifier le mode temporairement via les commutateurs ou de manière permanente en définissant la variable easy_going au début du script.

Voici comment l'installer:

Dans un terminal non root, faites:

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

Dans gedit, collez l'utilitaire CP et enregistrez:

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

Je viens d'avoir le même problème. Mon approche consistait simplement à archiver les fichiers dans une archive de la manière suivante:

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

tar stocke automatiquement les fichiers dans la structure appropriée de l'archive. Si tu cours

tar xf your_archive.tar

les fichiers sont extraits dans la structure de répertoires souhaitée.

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

Cela pourrait fonctionner si vous avez le bon type de rsync.

0
danuker

Comme suggéré ci-dessus par help_asap et spongeman, vous pouvez utiliser la commande 'install' pour copier des fichiers dans des répertoires existants ou créer de nouveaux répertoires de destination s'ils n'existent pas déjà.

Option 1 install -D filename some/deep/directory/filename
copie le fichier dans un répertoire nouveau ou existant et donne les autorisations 755 par défaut pour le nom de fichier

Option 2 install -D filename -m640 some/deep/directory/filename
Selon l’option 1, mais donne les autorisations 640 au nom de fichier.

Option 3 install -D filename -m640 -t some/deep/directory/
Selon l'option 2, mais nomfichier est ciblé dans le répertoire cible de sorte que le nomfichier n'ait pas besoin d'être écrit à la fois dans la source et la cible.

Option 4 install -D filena* -m640 -t some/deep/directory/
selon l'option 3 mais utilise un caractère générique pour plusieurs fichiers.

Cela fonctionne bien dans Ubuntu et combine deux étapes (création de répertoire puis copie de fichier) en une seule étape.

0
Dig

Facile

cp -a * /path/to/dst/

devrait faire l'affaire.

0
Brad D.

Copier de la source vers un chemin non existant

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

NOTE: cette commande copie tous les fichiers

cp –r pour copier tous les dossiers et leur contenu

$_ fonctionne comme destination qui est créée dans la dernière commande

0
developerprashant