• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one or more
3  * contributor license agreements.  See the NOTICE file distributed with
4  * this work for additional information regarding copyright ownership.
5  * The ASF licenses this file to You under the Apache License, Version 2.0
6  * (the "License"); you may not use this file except in compliance with
7  * the License.  You may obtain a copy of the License at
8  *
9  *      http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  */
17 package org.apache.commons.io;
18 
19 import java.io.BufferedReader;
20 import java.io.IOException;
21 import java.io.InputStream;
22 import java.io.InputStreamReader;
23 import java.io.OutputStream;
24 import java.util.ArrayList;
25 import java.util.Arrays;
26 import java.util.List;
27 import java.util.StringTokenizer;
28 
29 /**
30  * General File System utilities.
31  * <p>
32  * This class provides static utility methods for general file system
33  * functions not provided via the JDK {@link java.io.File File} class.
34  * <p>
35  * The current functions provided are:
36  * <ul>
37  * <li>Get the free space on a drive
38  * </ul>
39  *
40  * @author Frank W. Zammetti
41  * @author Stephen Colebourne
42  * @author Thomas Ledoux
43  * @author James Urie
44  * @author Magnus Grimsell
45  * @author Thomas Ledoux
46  * @version $Id: FileSystemUtils.java 453889 2006-10-07 11:56:25Z scolebourne $
47  * @since Commons IO 1.1
48  */
49 public class FileSystemUtils {
50 
51     /** Singleton instance, used mainly for testing. */
52     private static final FileSystemUtils INSTANCE = new FileSystemUtils();
53 
54     /** Operating system state flag for error. */
55     private static final int INIT_PROBLEM = -1;
56     /** Operating system state flag for neither Unix nor Windows. */
57     private static final int OTHER = 0;
58     /** Operating system state flag for Windows. */
59     private static final int WINDOWS = 1;
60     /** Operating system state flag for Unix. */
61     private static final int UNIX = 2;
62     /** Operating system state flag for Posix flavour Unix. */
63     private static final int POSIX_UNIX = 3;
64 
65     /** The operating system flag. */
66     private static final int OS;
67     static {
68         int os = OTHER;
69         try {
70             String osName = System.getProperty("os.name");
71             if (osName == null) {
72                 throw new IOException("os.name not found");
73             }
74             osName = osName.toLowerCase();
75             // match
76             if (osName.indexOf("windows") != -1) {
77                 os = WINDOWS;
78             } else if (osName.indexOf("linux") != -1 ||
79                 osName.indexOf("sun os") != -1 ||
80                 osName.indexOf("sunos") != -1 ||
81                 osName.indexOf("solaris") != -1 ||
82                 osName.indexOf("mpe/ix") != -1 ||
83                 osName.indexOf("freebsd") != -1 ||
84                 osName.indexOf("irix") != -1 ||
85                 osName.indexOf("digital unix") != -1 ||
86                 osName.indexOf("unix") != -1 ||
87                 osName.indexOf("mac os x") != -1) {
88                 os = UNIX;
89             } else if (osName.indexOf("hp-ux") != -1 ||
90                 osName.indexOf("aix") != -1) {
91                 os = POSIX_UNIX;
92             } else {
93                 os = OTHER;
94             }
95 
96         } catch (Exception ex) {
97             os = INIT_PROBLEM;
98         }
99         OS = os;
100     }
101 
102     /**
103      * Instances should NOT be constructed in standard programming.
104      */
FileSystemUtils()105     public FileSystemUtils() {
106         super();
107     }
108 
109     //-----------------------------------------------------------------------
110     /**
111      * Returns the free space on a drive or volume by invoking
112      * the command line.
113      * This method does not normalize the result, and typically returns
114      * bytes on Windows, 512 byte units on OS X and kilobytes on Unix.
115      * As this is not very useful, this method is deprecated in favour
116      * of {@link #freeSpaceKb(String)} which returns a result in kilobytes.
117      * <p>
118      * Note that some OS's are NOT currently supported, including OS/390,
119      * OpenVMS and and SunOS 5. (SunOS is supported by <code>freeSpaceKb</code>.)
120      * <pre>
121      * FileSystemUtils.freeSpace("C:");       // Windows
122      * FileSystemUtils.freeSpace("/volume");  // *nix
123      * </pre>
124      * The free space is calculated via the command line.
125      * It uses 'dir /-c' on Windows and 'df' on *nix.
126      *
127      * @param path  the path to get free space for, not null, not empty on Unix
128      * @return the amount of free drive space on the drive or volume
129      * @throws IllegalArgumentException if the path is invalid
130      * @throws IllegalStateException if an error occurred in initialisation
131      * @throws IOException if an error occurs when finding the free space
132      * @since Commons IO 1.1, enhanced OS support in 1.2 and 1.3
133      * @deprecated Use freeSpaceKb(String)
134      *  Deprecated from 1.3, may be removed in 2.0
135      */
136     @Deprecated
freeSpace(String path)137     public static long freeSpace(String path) throws IOException {
138         return INSTANCE.freeSpaceOS(path, OS, false);
139     }
140 
141     //-----------------------------------------------------------------------
142     /**
143      * Returns the free space on a drive or volume in kilobytes by invoking
144      * the command line.
145      * <pre>
146      * FileSystemUtils.freeSpaceKb("C:");       // Windows
147      * FileSystemUtils.freeSpaceKb("/volume");  // *nix
148      * </pre>
149      * The free space is calculated via the command line.
150      * It uses 'dir /-c' on Windows, 'df -kP' on AIX/HP-UX and 'df -k' on other Unix.
151      * <p>
152      * In order to work, you must be running Windows, or have a implementation of
153      * Unix df that supports GNU format when passed -k (or -kP). If you are going
154      * to rely on this code, please check that it works on your OS by running
155      * some simple tests to compare the command line with the output from this class.
156      * If your operating system isn't supported, please raise a JIRA call detailing
157      * the exact result from df -k and as much other detail as possible, thanks.
158      *
159      * @param path  the path to get free space for, not null, not empty on Unix
160      * @return the amount of free drive space on the drive or volume in kilobytes
161      * @throws IllegalArgumentException if the path is invalid
162      * @throws IllegalStateException if an error occurred in initialisation
163      * @throws IOException if an error occurs when finding the free space
164      * @since Commons IO 1.2, enhanced OS support in 1.3
165      */
freeSpaceKb(String path)166     public static long freeSpaceKb(String path) throws IOException {
167         return INSTANCE.freeSpaceOS(path, OS, true);
168     }
169 
170     //-----------------------------------------------------------------------
171     /**
172      * Returns the free space on a drive or volume in a cross-platform manner.
173      * Note that some OS's are NOT currently supported, including OS/390.
174      * <pre>
175      * FileSystemUtils.freeSpace("C:");  // Windows
176      * FileSystemUtils.freeSpace("/volume");  // *nix
177      * </pre>
178      * The free space is calculated via the command line.
179      * It uses 'dir /-c' on Windows and 'df' on *nix.
180      *
181      * @param path  the path to get free space for, not null, not empty on Unix
182      * @param os  the operating system code
183      * @param kb  whether to normalize to kilobytes
184      * @return the amount of free drive space on the drive or volume
185      * @throws IllegalArgumentException if the path is invalid
186      * @throws IllegalStateException if an error occurred in initialisation
187      * @throws IOException if an error occurs when finding the free space
188      */
freeSpaceOS(String path, int os, boolean kb)189     long freeSpaceOS(String path, int os, boolean kb) throws IOException {
190         if (path == null) {
191             throw new IllegalArgumentException("Path must not be empty");
192         }
193         switch (os) {
194             case WINDOWS:
195                 return (kb ? freeSpaceWindows(path) / 1024 : freeSpaceWindows(path));
196             case UNIX:
197                 return freeSpaceUnix(path, kb, false);
198             case POSIX_UNIX:
199                 return freeSpaceUnix(path, kb, true);
200             case OTHER:
201                 throw new IllegalStateException("Unsupported operating system");
202             default:
203                 throw new IllegalStateException(
204                   "Exception caught when determining operating system");
205         }
206     }
207 
208     //-----------------------------------------------------------------------
209     /**
210      * Find free space on the Windows platform using the 'dir' command.
211      *
212      * @param path  the path to get free space for, including the colon
213      * @return the amount of free drive space on the drive
214      * @throws IOException if an error occurs
215      */
freeSpaceWindows(String path)216     long freeSpaceWindows(String path) throws IOException {
217         path = FilenameUtils.normalize(path);
218         if (path.length() > 2 && path.charAt(1) == ':') {
219             path = path.substring(0, 2);  // seems to make it work
220         }
221 
222         // build and run the 'dir' command
223         String[] cmdAttribs = new String[] {"cmd.exe", "/C", "dir /-c " + path};
224 
225         // read in the output of the command to an ArrayList
226         List<String> lines = performCommand(cmdAttribs, Integer.MAX_VALUE);
227 
228         // now iterate over the lines we just read and find the LAST
229         // non-empty line (the free space bytes should be in the last element
230         // of the ArrayList anyway, but this will ensure it works even if it's
231         // not, still assuming it is on the last non-blank line)
232         for (int i = lines.size() - 1; i >= 0; i--) {
233             String line = lines.get(i);
234             if (line.length() > 0) {
235                 return parseDir(line, path);
236             }
237         }
238         // all lines are blank
239         throw new IOException(
240                 "Command line 'dir /-c' did not return any info " +
241                 "for path '" + path + "'");
242     }
243 
244     /**
245      * Parses the Windows dir response last line
246      *
247      * @param line  the line to parse
248      * @param path  the path that was sent
249      * @return the number of bytes
250      * @throws IOException if an error occurs
251      */
parseDir(String line, String path)252     long parseDir(String line, String path) throws IOException {
253         // read from the end of the line to find the last numeric
254         // character on the line, then continue until we find the first
255         // non-numeric character, and everything between that and the last
256         // numeric character inclusive is our free space bytes count
257         int bytesStart = 0;
258         int bytesEnd = 0;
259         int j = line.length() - 1;
260         innerLoop1: while (j >= 0) {
261             char c = line.charAt(j);
262             if (Character.isDigit(c)) {
263               // found the last numeric character, this is the end of
264               // the free space bytes count
265               bytesEnd = j + 1;
266               break innerLoop1;
267             }
268             j--;
269         }
270         innerLoop2: while (j >= 0) {
271             char c = line.charAt(j);
272             if (!Character.isDigit(c) && c != ',' && c != '.') {
273               // found the next non-numeric character, this is the
274               // beginning of the free space bytes count
275               bytesStart = j + 1;
276               break innerLoop2;
277             }
278             j--;
279         }
280         if (j < 0) {
281             throw new IOException(
282                     "Command line 'dir /-c' did not return valid info " +
283                     "for path '" + path + "'");
284         }
285 
286         // remove commas and dots in the bytes count
287         StringBuffer buf = new StringBuffer(line.substring(bytesStart, bytesEnd));
288         for (int k = 0; k < buf.length(); k++) {
289             if (buf.charAt(k) == ',' || buf.charAt(k) == '.') {
290                 buf.deleteCharAt(k--);
291             }
292         }
293         return parseBytes(buf.toString(), path);
294     }
295 
296     //-----------------------------------------------------------------------
297     /**
298      * Find free space on the *nix platform using the 'df' command.
299      *
300      * @param path  the path to get free space for
301      * @param kb  whether to normalize to kilobytes
302      * @param posix  whether to use the posix standard format flag
303      * @return the amount of free drive space on the volume
304      * @throws IOException if an error occurs
305      */
freeSpaceUnix(String path, boolean kb, boolean posix)306     long freeSpaceUnix(String path, boolean kb, boolean posix) throws IOException {
307         if (path.length() == 0) {
308             throw new IllegalArgumentException("Path must not be empty");
309         }
310         path = FilenameUtils.normalize(path);
311 
312         // build and run the 'dir' command
313         String flags = "-";
314         if (kb) {
315             flags += "k";
316         }
317         if (posix) {
318             flags += "P";
319         }
320         String[] cmdAttribs =
321             (flags.length() > 1 ? new String[] {"df", flags, path} : new String[] {"df", path});
322 
323         // perform the command, asking for up to 3 lines (header, interesting, overflow)
324         List<String> lines = performCommand(cmdAttribs, 3);
325         if (lines.size() < 2) {
326             // unknown problem, throw exception
327             throw new IOException(
328                     "Command line 'df' did not return info as expected " +
329                     "for path '" + path + "'- response was " + lines);
330         }
331         String line2 = lines.get(1); // the line we're interested in
332 
333         // Now, we tokenize the string. The fourth element is what we want.
334         StringTokenizer tok = new StringTokenizer(line2, " ");
335         if (tok.countTokens() < 4) {
336             // could be long Filesystem, thus data on third line
337             if (tok.countTokens() == 1 && lines.size() >= 3) {
338                 String line3 = lines.get(2); // the line may be interested in
339                 tok = new StringTokenizer(line3, " ");
340             } else {
341                 throw new IOException(
342                         "Command line 'df' did not return data as expected " +
343                         "for path '" + path + "'- check path is valid");
344             }
345         } else {
346             tok.nextToken(); // Ignore Filesystem
347         }
348         tok.nextToken(); // Ignore 1K-blocks
349         tok.nextToken(); // Ignore Used
350         String freeSpace = tok.nextToken();
351         return parseBytes(freeSpace, path);
352     }
353 
354     //-----------------------------------------------------------------------
355     /**
356      * Parses the bytes from a string.
357      *
358      * @param freeSpace  the free space string
359      * @param path  the path
360      * @return the number of bytes
361      * @throws IOException if an error occurs
362      */
parseBytes(String freeSpace, String path)363     long parseBytes(String freeSpace, String path) throws IOException {
364         try {
365             long bytes = Long.parseLong(freeSpace);
366             if (bytes < 0) {
367                 throw new IOException(
368                         "Command line 'df' did not find free space in response " +
369                         "for path '" + path + "'- check path is valid");
370             }
371             return bytes;
372 
373         } catch (NumberFormatException ex) {
374             throw new IOException(
375                     "Command line 'df' did not return numeric data as expected " +
376                     "for path '" + path + "'- check path is valid");
377         }
378     }
379 
380     //-----------------------------------------------------------------------
381     /**
382      * Performs the os command.
383      *
384      * @param cmdAttribs  the command line parameters
385      * @param max The maximum limit for the lines returned
386      * @return the parsed data
387      * @throws IOException if an error occurs
388      */
performCommand(String[] cmdAttribs, int max)389     List<String> performCommand(String[] cmdAttribs, int max) throws IOException {
390         // this method does what it can to avoid the 'Too many open files' error
391         // based on trial and error and these links:
392         // http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4784692
393         // http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4801027
394         // http://forum.java.sun.com/thread.jspa?threadID=533029&messageID=2572018
395         // however, its still not perfect as the JDK support is so poor
396         // (see commond-exec or ant for a better multi-threaded multi-os solution)
397 
398         List<String> lines = new ArrayList<String>(20);
399         Process proc = null;
400         InputStream in = null;
401         OutputStream out = null;
402         InputStream err = null;
403         BufferedReader inr = null;
404         try {
405             proc = openProcess(cmdAttribs);
406             in = proc.getInputStream();
407             out = proc.getOutputStream();
408             err = proc.getErrorStream();
409             inr = new BufferedReader(new InputStreamReader(in));
410             String line = inr.readLine();
411             while (line != null && lines.size() < max) {
412                 line = line.toLowerCase().trim();
413                 lines.add(line);
414                 line = inr.readLine();
415             }
416 
417             proc.waitFor();
418             if (proc.exitValue() != 0) {
419                 // os command problem, throw exception
420                 throw new IOException(
421                         "Command line returned OS error code '" + proc.exitValue() +
422                         "' for command " + Arrays.asList(cmdAttribs));
423             }
424             if (lines.size() == 0) {
425                 // unknown problem, throw exception
426                 throw new IOException(
427                         "Command line did not return any info " +
428                         "for command " + Arrays.asList(cmdAttribs));
429             }
430             return lines;
431 
432         } catch (InterruptedException ex) {
433             throw new IOException(
434                     "Command line threw an InterruptedException '" + ex.getMessage() +
435                     "' for command " + Arrays.asList(cmdAttribs));
436         } finally {
437             IOUtils.closeQuietly(in);
438             IOUtils.closeQuietly(out);
439             IOUtils.closeQuietly(err);
440             IOUtils.closeQuietly(inr);
441             if (proc != null) {
442                 proc.destroy();
443             }
444         }
445     }
446 
447     /**
448      * Opens the process to the operating system.
449      *
450      * @param cmdAttribs  the command line parameters
451      * @return the process
452      * @throws IOException if an error occurs
453      */
openProcess(String[] cmdAttribs)454     Process openProcess(String[] cmdAttribs) throws IOException {
455         return Runtime.getRuntime().exec(cmdAttribs);
456     }
457 
458 }
459