Estou usando isso no diretório de um cliente para renomear arquivos com a primeira letra de cada palavra sendo maiúscula conforme sua solicitação:
rename 's/\b(\w)/\u$1/g' *
Isso funciona bem, mas também coloca em maiúscula a primeira letra da extensão, então eu uso isso para corrigir isso:
rename 's/\.Txt$/.txt/' *.Txt
o que funciona bem se a maioria dos arquivos em uma pasta tiver a mesma extensão (geralmente verdadeira), mas é uma dor se eles estiverem muito misturados.
Para contornar esse problema, criei um pequeno script parecido com este (não ria!):
#!/bin/bash
rename 's/\.Txt$/.txt/' *.Txt
rename 's/\.Doc$/.doc/' *.Doc
rename 's/\.Docx$/.docx/' *.Docx
...
rename 's/\.Xlsx$/.xlsx/' *.Xlsx
Eu ignoro os erros 'Não é possível renomear *.Txt *.txt: nenhum arquivo ou diretório' e, se encontrar uma extensão que está faltando, apenas adiciono essa linha ao meu script.
Acho que isso não importa, mas os arquivos estão em um compartilhamento de servidor Windows, mas estou acessando usando o Ubuntu 16.04LTS. Se isso importa, posso copiá-los para minha unidade local primeiro, executar um comando no Ubuntu e mover os arquivos de volta, se necessário.
Existe alguma maneira de alterar o primeiro comando renomear para ignorar as extensões e deixá-las em minúsculas? Posso executar o novo comando em um diretório de nível superior e fazer com que ele seja recursivo em todos os subdiretórios?
Uma forma de o conseguir poderá ser a utilização do Negative Lookbehind para corresponder apenas ao limite da palavra - sequência de caracteres da palavra quando não for precedida por um ponto literal, por exemplo, dado
então
Supondo que você também queira evitar capitalizar o plural
s
, podemos modificar isso paraou mesmo
Isso colocará em maiúscula a primeira letra de cada palavra, exceto a extensão final:
Testei com esses arquivos:
E ele irá renomeá-los como:
O truque é primeiro colocar todas as primeiras letras em maiúsculas, ignorando as extensões, e depois voltar e colocar a extensão em minúsculas:
s/\b(.+?)\b/\u$1/g;
: O.+?
é um padrão não ganancioso, o que significa que encontrará a correspondência mais curta possível. Uma vez que é ancorado por limites de palavras (\b
), ele encontrará todas as palavras (tudo por causa do finalg
). Estes são então substituídos pela versão em maiúscula (primeira letra maiúscula) de si mesmos (\l$1
).s/(.+)\.(.)/$1\.\l$2/
: o.+
é ganancioso, por isso encontrará a correspondência mais longa possível. Isso significa a string mais longa até um final.
, após o qual será a extensão (se houver). Substituímos a correspondência por tudo antes da extensão ($1
), a.
e a extensão pela primeira letra minúscula novamente (\l$2
).A recursão também é bastante fácil. Se o seu shell for bash (se você não sabe, provavelmente é), você pode usar a opção globstar que faz
**
corresponder 0 ou mais subdiretórios:Agora, execute o comando renomear assim (isso também funcionará em qualquer arquivo em seu diretório atual):
Para limitá-lo apenas a arquivos ou diretórios com extensões, use
**/*.*
em vez de**
.Como alternativa, use
find
:E, para limitar esses arquivos e diretórios com uma extensão:
Observe que todas essas soluções irão renomear alegremente diretórios, bem como arquivos . Se você não quiser isso, tenha cuidado onde você diz para recursar ou dê a ele um padrão mais específico como
*.txt
.Em todos os exemplos, remova o -n para fazê-los realmente fazer algo. As
-n
causasrename
para simplesmente imprimir o que faria e não fazer nada. Útil para testes.A maneira mais sensata seria lidar com todas as "palavras" (com o uso do limite da palavra
\b
e referindo-se$1
a qualquer primeiro caractere), incluindo a extensão, mas, em seguida, diminua a própria extensão:Observe que isso não funciona para nomes de arquivos com sublinhados, ou seja, limites de "palavras" são considerados em espaços em branco (tabulações, espaços, novas linhas - que esperamos que não devam estar no nome do arquivo de qualquer maneira).
Observe o uso de
prename
para portabilidade paraksh
, onderename
na verdade é um comando interno, não operl
executável autônomo.Basta usar
rename 's/\b(\w)/\u$1/' *
SEM o sinalizador g.A bandeira g significa "global = faça várias vezes". Você faz a 1ª vez no nome do arquivo e não quer fazer letras maiúsculas na 2ª vez, certo?