]> Creatis software - gdcm.git/blob - src/gdcmDirList.cxx
Add
[gdcm.git] / src / gdcmDirList.cxx
1 /*=========================================================================
2                                                                                 
3   Program:   gdcm
4   Module:    $RCSfile: gdcmDirList.cxx,v $
5   Language:  C++
6   Date:      $Date: 2006/01/10 15:54:03 $
7   Version:   $Revision: 1.58 $
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       const char *str = strerror(errno);
82       gdcmStaticErrorMacro( str );
83       return false;
84    }
85 }
86
87 //-----------------------------------------------------------------------------
88 // Protected
89
90 //-----------------------------------------------------------------------------
91 // Private
92 /**
93  * \brief   Explore a directory with possibility of recursion
94  *          return number of files read
95  * @param  dirpath   directory to explore
96  * @param  recursive whether we want recursion or not
97  */
98 int DirList::Explore(std::string const &dirpath, bool recursive)
99 {
100    int numberOfFiles = 0;
101    std::string fileName;
102    std::string dirName = Util::NormalizePath(dirpath);
103 #ifdef _MSC_VER
104    WIN32_FIND_DATA fileData;
105    //assert( dirName[dirName.size()-1] == '' );
106    HANDLE hFile = FindFirstFile((dirName+"*").c_str(), &fileData);
107
108    for(BOOL b = (hFile != INVALID_HANDLE_VALUE); b;
109        b = FindNextFile(hFile, &fileData))
110    {
111       fileName = fileData.cFileName;
112       if ( fileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY )
113       {
114          // Need to check for . and .. to avoid infinite loop
115          if ( fileName != "." && fileName != ".." && recursive )
116          {
117             numberOfFiles += Explore(dirName+fileName,recursive);
118          }
119       }
120       else
121       {
122          Filenames.push_back(dirName+fileName);
123          numberOfFiles++;
124       }
125    }
126    DWORD dwError = GetLastError();
127    if (hFile != INVALID_HANDLE_VALUE) 
128       FindClose(hFile);
129    if (dwError != ERROR_NO_MORE_FILES) 
130    {
131       LPVOID lpMsgBuf;
132       FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER|
133                     FORMAT_MESSAGE_FROM_SYSTEM|
134                     FORMAT_MESSAGE_IGNORE_INSERTS,
135                     NULL,GetLastError(),
136                     MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
137                     (LPTSTR) &lpMsgBuf,0,NULL);
138
139       gdcmErrorMacro("FindNextFile error. Error is " << (char *)lpMsgBuf
140                    <<" for the directory : "<<dirName);
141       return -1;
142    }
143
144 #else
145   // Real POSIX implementation: scandir is a BSD extension only, and doesn't 
146   // work on debian for example
147
148    DIR* dir = opendir(dirName.c_str());
149    if (!dir)
150    {
151       return 0;
152    }
153
154    // According to POSIX, the dirent structure contains a field char d_name[]
155    // of unspecified size, with at most NAME_MAX characters preceeding the
156    // terminating null character. Use of other fields will harm the  porta-
157    // bility of your programs.
158
159    struct stat buf;
160    dirent *d;
161    for (d = readdir(dir); d; d = readdir(dir))
162    {
163       fileName = dirName + d->d_name;
164       if( stat(fileName.c_str(), &buf) != 0 )
165       {
166          const char *str = strerror(errno);
167          gdcmErrorMacro( str );
168       }
169       if ( S_ISREG(buf.st_mode) )    //is it a regular file?
170       {
171          Filenames.push_back( fileName );
172          numberOfFiles++;
173       }
174       else if ( S_ISDIR(buf.st_mode) ) //directory?
175       {
176          if ( d->d_name[0] != '.' && recursive ) //we also skip hidden files
177          {
178             numberOfFiles += Explore( fileName, recursive);
179          }
180       }
181       else
182       {
183          gdcmErrorMacro( "Unexpected error" );
184          return -1;
185       }
186    }
187    if( closedir(dir) != 0 )
188    {
189       const char *str = strerror(errno);
190       gdcmErrorMacro( str );
191    }
192 #endif
193
194   return numberOfFiles;
195 }
196
197 //-----------------------------------------------------------------------------
198 // Print
199 /**
200  * \brief   Print method
201  * @param os ostream to write to 
202  */
203 void DirList::Print(std::ostream &os, std::string const &)
204 {
205    std::copy(Filenames.begin(), Filenames.end(), 
206              std::ostream_iterator<std::string>(os, "\n"));
207 }
208
209 //-----------------------------------------------------------------------------
210 } // end namespace gdcm