Chamada de sistema mkdir() recursiva no Unix
Pergunta
Depois de ler a página man mkdir(2) para a chamada do sistema Unix com esse nome, parece que a chamada não cria diretórios intermediários em um caminho, apenas o último diretório no caminho.Existe alguma maneira (ou outra função) de criar todos os diretórios no caminho sem recorrer à análise manual da string do meu diretório e à criação individual de cada diretório?
Solução
Verifique se todos os ativos usados na página mestra e o layout da página são verificados e publicados.Isso inclui CSS, imagens, javascript, etc. Se houver algum código personalizado, verifique se ele também não referência a ativos não publicados ou outras listas / bibliotecas que têm segurança restritiva.
Outras dicas
Hmm eu pensei que mkdir -p faz isso?
mkdir -p this/is/a/completo/path/of/syths
ms tem um API de fala e agora tem uma interface gerenciada facilmente consumida.
Aqui está outra opinião sobre mkpath()
, usando a recursão, que é pequena e legível. Faz uso de strdupa()
para evitar alterar o dado dir
argumento de string diretamente e para evitar usar malloc()
& free()
. Certifique -se de compilar com -D_GNU_SOURCE
ativar strdupa()
... o que significa que esse código funciona apenas nas bibliotecas C Glibc, Eglibc, UCLIBC e outras bibliotecas C compatíveis com GLIBC.
int mkpath(char *dir, mode_t mode)
{
if (!dir) {
errno = EINVAL;
return 1;
}
if (strlen(dir) == 1 && dir[0] == '/')
return 0;
mkpath(dirname(strdupa(dir)), mode);
return mkdir(dir, mode);
}
Após a entrada aqui e de Valery Frolov, no projeto Inadyn, a seguinte versão revisada de mkpath()
agora foi empurrado para libite
int mkpath(char *dir, mode_t mode)
{
struct stat sb;
if (!dir) {
errno = EINVAL;
return 1;
}
if (!stat(dir, &sb))
return 0;
mkpath(dirname(strdupa(dir)), mode);
return mkdir(dir, mode);
}
Ele usa mais um syscall, mas otoh o código é mais legível agora.
Dê uma olhada no código -fonte Bash aqui, e procure especificamente exemplos/carregáveis/mkdir.c especialmente linhas 136-210. Se você não quer fazer isso, aqui está algumas das fontes que lidam com isso (retirado do Tar.gz que vinculei):
/* Make all the directories leading up to PATH, then create PATH. Note that
this changes the process's umask; make sure that all paths leading to a
return reset it to ORIGINAL_UMASK */
static int
make_path (path, nmode, parent_mode)
char *path;
int nmode, parent_mode;
{
int oumask;
struct stat sb;
char *p, *npath;
if (stat (path, &sb) == 0)
{
if (S_ISDIR (sb.st_mode) == 0)
{
builtin_error ("`%s': file exists but is not a directory", path);
return 1;
}
if (chmod (path, nmode))
{
builtin_error ("%s: %s", path, strerror (errno));
return 1;
}
return 0;
}
oumask = umask (0);
npath = savestring (path); /* So we can write to it. */
/* Check whether or not we need to do anything with intermediate dirs. */
/* Skip leading slashes. */
p = npath;
while (*p == '/')
p++;
while (p = strchr (p, '/'))
{
*p = '\0';
if (stat (npath, &sb) != 0)
{
if (mkdir (npath, parent_mode))
{
builtin_error ("cannot create directory `%s': %s", npath, strerror (errno));
umask (original_umask);
free (npath);
return 1;
}
}
else if (S_ISDIR (sb.st_mode) == 0)
{
builtin_error ("`%s': file exists but is not a directory", npath);
umask (original_umask);
free (npath);
return 1;
}
*p++ = '/'; /* restore slash */
while (*p == '/')
p++;
}
/* Create the final directory component. */
if (stat (npath, &sb) && mkdir (npath, nmode))
{
builtin_error ("cannot create directory `%s': %s", npath, strerror (errno));
umask (original_umask);
free (npath);
return 1;
}
umask (original_umask);
free (npath);
return 0;
}
Você provavelmente pode se safar com uma implementação menos geral.
Aparentemente não, minhas duas sugestões são:
char dirpath[80] = "/path/to/some/directory";
sprintf(mkcmd, "mkdir -p %s", dirpath);
system(mkcmd);
Ou se você não quiser usar system()
tente olhar para os coreutils mkdir
código -fonte e veja como eles implementaram o -p
opção.
Na verdade você pode simplesmente usar:
mkdir -p ./some/directories/to/be/created/
Não tenho permissão para comentar a primeira (e aceita) resposta (representação insuficiente), então postarei meus comentários como código em uma nova resposta.O código abaixo é baseado na primeira resposta, mas corrige vários problemas:
- Se chamado com um caminho de comprimento zero, não lê nem grava o caractere antes do início do array
opath[]
(sim, "por que você chamaria assim?", mas por outro lado "por que você não corrigiria a vulnerabilidade?") - o tamanho de
opath
é agoraPATH_MAX
(o que não é perfeito, mas é melhor que uma constante) - se o caminho for tão longo ou mais longo que
sizeof(opath)
então ele será encerrado corretamente quando copiado (o questrncpy()
não faz) - você pode especificar o modo do diretório escrito, assim como você pode com o padrão
mkdir()
(embora se você especificar não gravável pelo usuário ou não executável pelo usuário, a recursão não funcionará) - main() retorna o int (obrigatório?)
- removeu alguns desnecessários
#include
é - Eu gosto mais do nome da função;)
// Based on http://nion.modprobe.de/blog/archives/357-Recursive-directory-creation.html
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>
#include <limits.h>
static void mkdirRecursive(const char *path, mode_t mode) {
char opath[PATH_MAX];
char *p;
size_t len;
strncpy(opath, path, sizeof(opath));
opath[sizeof(opath) - 1] = '\0';
len = strlen(opath);
if (len == 0)
return;
else if (opath[len - 1] == '/')
opath[len - 1] = '\0';
for(p = opath; *p; p++)
if (*p == '/') {
*p = '\0';
if (access(opath, F_OK))
mkdir(opath, mode);
*p = '/';
}
if (access(opath, F_OK)) /* if path is not terminated with / */
mkdir(opath, mode);
}
int main (void) {
mkdirRecursive("/Users/griscom/one/two/three", S_IRWXU);
return 0;
}
Minha maneira recursiva de fazer isso:
#include <libgen.h> /* Only POSIX version of dirname() */
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
static void recursive_mkdir(const char *path, mode_t mode)
{
char *spath = NULL;
const char *next_dir = NULL;
/* dirname() modifies input! */
spath = strdup(path);
if (spath == NULL)
{
/* Report error, no memory left for string duplicate. */
goto done;
}
/* Get next path component: */
next_dir = dirname(spath);
if (access(path, F_OK) == 0)
{
/* The directory in question already exists! */
goto done;
}
if (strcmp(next_dir, ".") == 0 || strcmp(next_dir, "/") == 0)
{
/* We reached the end of recursion! */
goto done;
}
recursive_mkdir(next_dir, mode);
if (mkdir(path, mode) != 0)
{
/* Report error on creating directory */
}
done:
free(spath);
return;
}
Editar: Corrigido meu antigo trecho de código, relato de insetos por Namchester
Eu acho que jfreechart é a melhor solução ..
Minha solução:
int mkrdir(const char *path, int index, int permission)
{
char bf[NAME_MAX];
if(*path == '/')
index++;
char *p = strchr(path + index, '/');
int len;
if(p) {
len = MIN(p-path, sizeof(bf)-1);
strncpy(bf, path, len);
bf[len]=0;
} else {
len = MIN(strlen(path)+1, sizeof(bf)-1);
strncpy(bf, path, len);
bf[len]=0;
}
if(access(bf, 0)!=0) {
mkdir(bf, permission);
if(access(bf, 0)!=0) {
return -1;
}
}
if(p) {
return mkrdir(path, p-path+1, permission);
}
return 0;
}
Aqui está minha chance de uma solução mais geral:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
typedef int (*dirhandler_t)( const char*, void* );
/// calls itfunc for each directory in path (except for . and ..)
int iterate_path( const char* path, dirhandler_t itfunc, void* udata )
{
int rv = 0;
char tmp[ 256 ];
char *p = tmp;
char *lp = tmp;
size_t len;
size_t sublen;
int ignore_entry;
strncpy( tmp, path, 255 );
tmp[ 255 ] = '\0';
len = strlen( tmp );
if( 0 == len ||
(1 == len && '/' == tmp[ 0 ]) )
return 0;
if( tmp[ len - 1 ] == '/' )
tmp[ len - 1 ] = 0;
while( (p = strchr( p, '/' )) != NULL )
{
ignore_entry = 0;
*p = '\0';
lp = strrchr( tmp, '/' );
if( NULL == lp ) { lp = tmp; }
else { lp++; }
sublen = strlen( lp );
if( 0 == sublen ) /* ignore things like '//' */
ignore_entry = 1;
else if( 1 == sublen && /* ignore things like '/./' */
'.' == lp[ 0 ] )
ignore_entry = 1;
else if( 2 == sublen && /* also ignore things like '/../' */
'.' == lp[ 0 ] &&
'.' == lp[ 1 ] )
ignore_entry = 1;
if( ! ignore_entry )
{
if( (rv = itfunc( tmp, udata )) != 0 )
return rv;
}
*p = '/';
p++;
lp = p;
}
if( strcmp( lp, "." ) && strcmp( lp, ".." ) )
return itfunc( tmp, udata );
return 0;
}
mode_t get_file_mode( const char* path )
{
struct stat statbuf;
memset( &statbuf, 0, sizeof( statbuf ) );
if( NULL == path ) { return 0; }
if( 0 != stat( path, &statbuf ) )
{
fprintf( stderr, "failed to stat '%s': %s\n",
path, strerror( errno ) );
return 0;
}
return statbuf.st_mode;
}
static int mymkdir( const char* path, void* udata )
{
(void)udata;
int rv = mkdir( path, S_IRWXU );
int errnum = errno;
if( 0 != rv )
{
if( EEXIST == errno &&
S_ISDIR( get_file_mode( path ) ) ) /* it's all good, the directory already exists */
return 0;
fprintf( stderr, "mkdir( %s ) failed: %s\n",
path, strerror( errnum ) );
}
// else
// {
// fprintf( stderr, "created directory: %s\n", path );
// }
return rv;
}
int mkdir_with_leading( const char* path )
{
return iterate_path( path, mymkdir, NULL );
}
int main( int argc, const char** argv )
{
size_t i;
int rv;
if( argc < 2 )
{
fprintf( stderr, "usage: %s <path> [<path>...]\n",
argv[ 0 ] );
exit( 1 );
}
for( i = 1; i < argc; i++ )
{
rv = mkdir_with_leading( argv[ i ] );
if( 0 != rv )
return rv;
}
return 0;
}
Uma solução muito simples, basta passar o input: mkdir dirname
void execute_command_mkdir(char *input)
{
char rec_dir[500];
int s;
if(strcmp(input,"mkdir") == 0)
printf("mkdir: operand required");
else
{
char *split = strtok(input," \t");
while(split)
{
if(strcmp(split,"create_dir") != 0)
strcpy(rec_dir,split);
split = strtok(NULL, " \t");
}
char *split2 = strtok(rec_dir,"/");
char dir[500];
strcpy(dir, "");
while(split2)
{
strcat(dir,split2);
strcat(dir,"/");
printf("%s %s\n",split2,dir);
s = mkdir(dir,0700);
split2 = strtok(NULL,"/");
}
strcpy(output,"ok");
}
if(s < 0)
printf(output,"Error!! Cannot Create Directory!!");
}
Bastante direto.Este pode ser um bom ponto de partida
int makeDir(char *fullpath, mode_t permissions){
int i=0;
char *arrDirs[20];
char aggrpaz[255];
arrDirs[i] = strtok(fullpath,"/");
strcpy(aggrpaz, "/");
while(arrDirs[i]!=NULL)
{
arrDirs[++i] = strtok(NULL,"/");
strcat(aggrpaz, arrDirs[i-1]);
mkdir(aggrpaz,permissions);
strcat(aggrpaz, "/");
}
i=0;
return 0;
}
Você analisa esta função um caminho completo mais as permissões desejadas, ou seja S_IRUSR, para obter uma lista completa de modos, clique aqui https://techoverflow.net/2013/04/05/how-to-use-mkdir-from-sysstat-h/
A string fullpath será dividida pelo caractere "/" e diretórios individuais serão anexados ao aggrpaz corda um de cada vez.Cada iteração do loop chama a função mkdir, passando a ela o caminho agregado até o momento mais as permissões.Este exemplo pode ser melhorado, não estou verificando a saída da função mkdir e esta função só funciona com caminhos absolutos.