]> Creatis software - gdcm.git/blob - src/gdcmDirList.cxx
Port Mathieu's modif from gdcm1.2
[gdcm.git] / src / gdcmDirList.cxx
1 /*=========================================================================
2                                                                                 
3   Program:   gdcm
4   Module:    $RCSfile: gdcmDirList.cxx,v $
5   Language:  C++
6   Date:      $Date: 2006/05/12 09:36:09 $
7   Version:   $Revision: 1.59 $
8                                                                                 
9   Copyright (c) CREATIS (Centre de Recherche et d'Applications en Traitement de
10   l'Image). All rights reserved. See Doc/License.txt or
11   http://www.creatis.insa-lyon.fr/Public/Gdcm/License.html for details.
12                                                                                 
13      This software is distributed WITHOUT ANY WARRANTY; without even
14      the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15      PURPOSE.  See the above copyright notices for more information.
16                                                                                 
17 =========================================================================*/
18
19 #include "gdcmDirList.h"
20 #include "gdcmUtil.h"
21 #include "gdcmDebug.h"
22
23 #include <iterator>
24 #include <assert.h>
25 #include <errno.h>
26 #include <sys/stat.h>  //stat function
27
28 #ifdef _MSC_VER
29    #include <windows.h> 
30    #include <direct.h>
31 #else
32    #include <dirent.h>   
33    #include <sys/types.h>
34 #endif
35
36 namespace gdcm
37 {
38 //-----------------------------------------------------------------------------
39 // Constructor / Destructor
40 /**
41  * \brief Constructor  
42  * @param  dirName root directory name
43  * @param  recursive whether we want to explore recursively or not 
44  */
45 DirList::DirList(std::string const &dirName, bool recursive)
46 {
47    DirName = dirName;
48    Explore(dirName, recursive);
49 }
50
51 /**
52  * \brief  Destructor
53  */
54 DirList::~DirList()
55 {
56 }
57
58 //-----------------------------------------------------------------------------
59 // Public
60 /**
61  * \brief Tells us if file name corresponds to a Directory   
62  * @param  dirName file name to check
63  * @return true if the file IS a Directory
64  */
65 bool DirList::IsDirectory(std::string const &dirName)
66 {
67    struct stat fs;
68    // std::cout << "dirName[dirName.size()-1] [" << dirName[dirName.size()-1] << "]"
69    //           << std::endl;
70    //assert( dirName[dirName.size()-1] != GDCM_FILESEPARATOR );
71    if ( stat(dirName.c_str(), &fs) == 0 )
72    {
73 #if _WIN32
74       return ((fs.st_mode & _S_IFDIR) != 0);
75 #else
76       return S_ISDIR(fs.st_mode);
77 #endif
78    }
79    else
80    {
81       gdcmStaticErrorMacro( strerror(errno) );
82       return false;
83    }
84 }
85
86 //-----------------------------------------------------------------------------
87 // Protected
88
89 //-----------------------------------------------------------------------------
90 // Private
91 /**
92  * \brief   Explore a directory with possibility of recursion
93  *          return number of files read
94  * @param  dirpath   directory to explore
95  * @param  recursive whether we want recursion or not
96  */
97 int DirList::Explore(std::string const &dirpath, bool recursive)
98 {
99    int numberOfFiles = 0;
100    std::string fileName;
101    std::string dirName = Util::NormalizePath(dirpath);
102 #ifdef _MSC_VER
103    WIN32_FIND_DATA fileData;
104    //assert( dirName[dirName.size()-1] == '' );
105    HANDLE hFile = FindFirstFile((dirName+"*").c_str(), &fileData);
106
107    for(BOOL b = (hFile != INVALID_HANDLE_VALUE); b;
108        b = FindNextFile(hFile, &fileData))
109    {
110       fileName = fileData.cFileName;
111       if ( fileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY )
112       {
113          // Need to check for . and .. to avoid infinite loop
114          if ( fileName != "." && fileName != ".." && recursive )
115          {
116             numberOfFiles += Explore(dirName+fileName,recursive);
117          }
118       }
119       else
120       {
121          Filenames.push_back(dirName+fileName);
122          numberOfFiles++;
123       }
124    }
125    DWORD dwError = GetLastError();
126    if (hFile != INVALID_HANDLE_VALUE) 
127       FindClose(hFile);
128    if (dwError != ERROR_NO_MORE_FILES) 
129    {
130       LPVOID lpMsgBuf;
131       FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER|
132                     FORMAT_MESSAGE_FROM_SYSTEM|
133                     FORMAT_MESSAGE_IGNORE_INSERTS,
134                     NULL,GetLastError(),
135                     MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
136                     (LPTSTR) &lpMsgBuf,0,NULL);
137
138       gdcmErrorMacro("FindNextFile error. Error is " << (char *)lpMsgBuf
139                    <<" for the directory : "<<dirName);
140       return -1;
141    }
142
143 #else
144   // Real POSIX implementation: scandir is a BSD extension only, and doesn't 
145   // work on debian for example
146
147    DIR* dir = opendir(dirName.c_str());
148    if (!dir)
149    {
150       return 0;
151    }
152
153    // According to POSIX, the dirent structure contains a field char d_name[]
154    // of unspecified size, with at most NAME_MAX characters preceeding the
155    // terminating null character. Use of other fields will harm the  porta-
156    // bility of your programs.
157
158    struct stat buf;
159    dirent *d;
160    for (d = readdir(dir); d; d = readdir(dir))
161    {
162       fileName = dirName + d->d_name;
163       if( stat(fileName.c_str(), &buf) != 0 )
164       {
165          gdcmErrorMacro( strerror(errno) );
166       }
167       if ( S_ISREG(buf.st_mode) )    //is it a regular file?
168       {
169          Filenames.push_back( fileName );
170          numberOfFiles++;
171       }
172       else if ( S_ISDIR(buf.st_mode) ) //directory?
173       {
174          if ( d->d_name[0] != '.' && recursive ) //we also skip hidden files
175          {
176             numberOfFiles += Explore( fileName, recursive);
177          }
178       }
179       else
180       {
181          gdcmErrorMacro( "Unexpected error" );
182          return -1;
183       }
184    }
185    if( closedir(dir) != 0 )
186    {
187       gdcmErrorMacro( strerror(errno) );
188    }
189 #endif
190
191   return numberOfFiles;
192 }
193
194 //-----------------------------------------------------------------------------
195 // Print
196 /**
197  * \brief   Print method
198  * @param os ostream to write to 
199  */
200 void DirList::Print(std::ostream &os, std::string const &)
201 {
202    std::copy(Filenames.begin(), Filenames.end(), 
203              std::ostream_iterator<std::string>(os, "\n"));
204 }
205
206 //-----------------------------------------------------------------------------
207 } // end namespace gdcm