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