Linux: copier et créer destination dir si elle n'existe pas
je veux une commande (ou probablement une option vers 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
16 réponses
mkdir -p "$d" && cp file "$d"
(il n'y a pas d'option pour cp
).
Si les deux conditions suivantes sont remplies:
- vous utilisez la version GNU de
cp
(et pas, par exemple, la version Mac), et - vous copiez à partir d'une structure de répertoire existante et vous avez juste besoin de le recréer
alors vous pouvez le faire avec le drapeau --parents
de cp
. 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
Réponse Courte
copier myfile.txt
à /foo/bar/myfile.txt
, utilisez:
mkdir -p /foo/bar && cp myfile.txt $_
Comment cela fonctionne?
il y a quelques composants à cela, donc je vais couvrir toute la syntaxe étape par étape.
Le mkdir utilité", 1519380920" comme spécifié dans la norme POSIX , fait répertoires. L'argument -p
, selon le docs, causera mkdir à
créer tout composant de chemin intermédiaire manquant
signifie qu'en appelant mkdir -p /foo/bar
, mkdir créera /foo
et /foo/bar
si /foo
n'existe pas déjà. (Sans -p
, il lancera à la place une erreur.
l'opérateur de la liste &&
, tel que documenté dans le standard POSIX (ou le Manuel de Bash si vous préférez), a pour effet que cp myfile.txt $_
n'est exécuté que si mkdir -p /foo/bar
exécute avec succès. Cela signifie que la commande cp
n'essaiera pas de s'exécuter si mkdir
échoue pour l'une des nombreuses raisons pour lesquelles elle pourrait échouer .
enfin, le $_
nous passons comme second argument à cp
est un " paramètre spécial" qui peut être pratique pour éviter de répéter de longs arguments (comme des chemins de fichier) sans avoir à les stocker dans une variable. Par le manuel de Bash , elle:
s'étend jusqu'au 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
s'étend à cp myfile.txt /foo/bar
, qui copie myfile.txt
dans le nouveau /foo/bar
répertoire.
notez que $_
est et non fait partie de la norme POSIX , donc théoriquement une variante Unix pourrait avoir un shell qui ne supporte pas cette construction. Cependant, je ne connais pas de coquilles modernes qui ne supportent pas $_
; certainement Bash, Dash, et zsh tous le font.
Une dernière remarque: la commande que j'ai donné au début de cette réponse suppose que votre répertoire de noms n'ont pas d'espaces. Si vous avez affaire à des noms avec des espaces, vous aurez besoin de les citer de sorte que les différents mots ne sont pas traités comme des arguments différents de mkdir
ou cp
. Donc votre commande ressemblerait en fait à:
mkdir -p "/my directory/name with/spaces" && cp "my filename with spaces.txt" "$_"
une si vieille question, mais peut-être que je peux 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 quelques aspects à prendre en considération:
- vous devez spécifier également le nom du fichier de destination , non seulement le chemin de destination
- le fichier de destination sera exécutable (du moins, d'après mes tests)
vous pouvez facilement modifier le #2 en ajoutant l'option -m
pour définir les permissions sur le fichier de destination (exemple: -m 664
va créer le fichier de destination avec les permissions rw-rw-r--
, tout comme la création d'un nouveau fichier avec touch
).
et voici le lien honteux à la réponse j'ai été inspiré par =)
fonction Shell qui fait ce que vous voulez, l'appelant une copie "bury" parce qu'il creuse un trou pour le fichier à vivre:
bury_copy() { mkdir -p `dirname ` && cp "" ""; }
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 ...'va ensuite créer ce répertoire en s'assurant que lorsque vous appelez cp -r le répertoire de base correct est en place.
l'avantage de ceci sur --parents est que cela fonctionne pour le cas où le dernier élément dans le chemin de destination est un nom de fichier.
et ça marchera sur OS X.
install -D file -m 644 -t /path/to/copy/file/to/is/very/deep/there
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/
il suffit d'ajouter ce qui suit dans votre .bashrc, tweak si vous avez besoin. Travaille à Ubuntu.
mkcp() {
test -d "" || mkdir -p ""
cp -r "" ""
}
e. g Si vous voulez copier le fichier' test ' dans le répertoire de destination 'd' Use,
mkcp test a/b/c/d
mkcp vérifiera d'abord si le répertoire de destination existe ou non, sinon le fera et copiera le fichier/répertoire source.
cp
a de multiples 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.
@AndyRoss réponse travaille pour la
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 dans cet usage (i.e. là où le répertoire cible n'existe pas encore), ce qui explique peut-être pourquoi cp
n'a jamais ajouté une option pour cela.
voici donc ma version de cette fonction qui impose une barre oblique sur le dir le plus sombre:
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
}
juste pour reprendre et donner une solution de travail complète, en une ligne. Faites attention si vous voulez renommer votre fichier, vous devriez inclure une façon de fournir un chemin dir propre à mkdir. $fdst peut être file ou dir. Le prochain code 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}
rsync file /path/to/copy/file/to/is/very/deep/there
cela pourrait fonctionner, si vous avez la bonne sorte de rsync
.
copier de la source à 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
$_
travail comme destination qui est créé dans la dernière commande
j'ai écrit un script de support pour cp, appelé CP (notez les majuscules) qui est destiné à faire exactement cela. Script vérifiera les erreurs dans le chemin que vous avez mis (sauf le dernier qui est la destination) et si tout va bien, il fera une étape mkdir-p pour créer le chemin de destination avant de lancer la copie. À ce moment, l'utilitaire cp régulier prend la relève et tous les commutateurs que vous utilisez avec CP (comme-r,- p,- rpL sont acheminés directement à cp). Avant d'utiliser mon script, il y a quelques choses que vous devez comprendre.
- toutes les informations ici peuvent être consultées en faisant CP --help. CP -- aide-tous les commutateurs de cp inclus.
- 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 CP. Vous êtes destination sera créé, donc si vous mal orthographiez votre destination comme /usrr/share/icons ou /usr/share/icon bien c'est ce qui va être créé.
- régulier cp 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 copie 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 b est un fichier, il sera remplacé par b de la copie. Si c n'existe pas, cp ne fait pas la copie et sort.
CP n'a pas le luxe de prendre des repères à partir des chemins existants, il doit donc avoir un certain très des comportements fermes. CP suppose que l'article que vous copiez est abandonné dans le chemin de destination et n'est pas la destination elle-même (aka, une copie renommée du fichier/dossier source). Ce qui signifie:
- "CP /a/b /c/d" entraînera /c/d/b si d est un dossier
- "CP /a/b /c/b" entraînera /c/b/b, si b /c/b est un dossier.
- Si b et d sont des fichiers: CP /A/B / C / d résultera en / c / D (où d est une copie de b). Idem pour le document CP /A/b / c / b dans les mêmes circonstances.
ce comportement de CP par défaut peut être modifié avec le commutateur "--rename". Dans ce cas, il est supposé que
- " CP --rename /a/b /c/d " copie b en /C et renomme la copie en D.
quelques remarques finales: comme avec cp, CP peut copier plusieurs articles à la fois avec le dernier chemin étant énuméré supposé être la destination. Il peut également Gérer les chemins avec des espaces aussi longtemps que vous utilisez des guillemets.
CP va vérifier les chemins que vous avez mis et s'assurer qu'ils existent avant de faire la copie. En mode strict (disponible via --strict switch), tous les fichiers/dossiers copiés doivent exister ou aucune copie ne doit avoir lieu. En mode détendu (--détendu), copie continuera si au moins l'un des éléments énumérés existe. Le mode Relax est le mode par défaut, vous pouvez changer le mode Temporairement via les commutateurs ou de façon permanente en réglant le variable easy_ going au début du script.
Voici comment l'installer:
Dans un non-terminal en 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, pâte de CP de l'utilitaire et de l'enregistrer:
#!/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: )
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=,,,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
disons que vous faites quelque chose comme
cp fichier1.txt A/B/C/D / file.txt
où A/B/C / D sont des répertoires qui n'existent pas encore
une solution possible est la suivante
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
espère que ça aide!