001 /* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017 package org.apache.commons.io; 018 019 import java.io.File; 020 import java.io.IOException; 021 import java.util.ArrayList; 022 import java.util.Collection; 023 import java.util.Stack; 024 025 /** 026 * General filename and filepath manipulation utilities. 027 * <p> 028 * When dealing with filenames you can hit problems when moving from a Windows 029 * based development machine to a Unix based production machine. 030 * This class aims to help avoid those problems. 031 * <p> 032 * <b>NOTE</b>: You may be able to avoid using this class entirely simply by 033 * using JDK {@link java.io.File File} objects and the two argument constructor 034 * {@link java.io.File#File(java.io.File, java.lang.String) File(File,String)}. 035 * <p> 036 * Most methods on this class are designed to work the same on both Unix and Windows. 037 * Those that don't include 'System', 'Unix' or 'Windows' in their name. 038 * <p> 039 * Most methods recognise both separators (forward and back), and both 040 * sets of prefixes. See the javadoc of each method for details. 041 * <p> 042 * This class defines six components within a filename 043 * (example C:\dev\project\file.txt): 044 * <ul> 045 * <li>the prefix - C:\</li> 046 * <li>the path - dev\project\</li> 047 * <li>the full path - C:\dev\project\</li> 048 * <li>the name - file.txt</li> 049 * <li>the base name - file</li> 050 * <li>the extension - txt</li> 051 * </ul> 052 * Note that this class works best if directory filenames end with a separator. 053 * If you omit the last separator, it is impossible to determine if the filename 054 * corresponds to a file or a directory. As a result, we have chosen to say 055 * it corresponds to a file. 056 * <p> 057 * This class only supports Unix and Windows style names. 058 * Prefixes are matched as follows: 059 * <pre> 060 * Windows: 061 * a\b\c.txt --> "" --> relative 062 * \a\b\c.txt --> "\" --> current drive absolute 063 * C:a\b\c.txt --> "C:" --> drive relative 064 * C:\a\b\c.txt --> "C:\" --> absolute 065 * \\server\a\b\c.txt --> "\\server\" --> UNC 066 * 067 * Unix: 068 * a/b/c.txt --> "" --> relative 069 * /a/b/c.txt --> "/" --> absolute 070 * ~/a/b/c.txt --> "~/" --> current user 071 * ~ --> "~/" --> current user (slash added) 072 * ~user/a/b/c.txt --> "~user/" --> named user 073 * ~user --> "~user/" --> named user (slash added) 074 * </pre> 075 * Both prefix styles are matched always, irrespective of the machine that you are 076 * currently running on. 077 * <p> 078 * Origin of code: Excalibur, Alexandria, Tomcat, Commons-Utils. 079 * 080 * @version $Id: FilenameUtils.java 1307462 2012-03-30 15:13:11Z ggregory $ 081 * @since 1.1 082 */ 083 public class FilenameUtils { 084 085 /** 086 * The extension separator character. 087 * @since 1.4 088 */ 089 public static final char EXTENSION_SEPARATOR = '.'; 090 091 /** 092 * The extension separator String. 093 * @since 1.4 094 */ 095 public static final String EXTENSION_SEPARATOR_STR = Character.toString(EXTENSION_SEPARATOR); 096 097 /** 098 * The Unix separator character. 099 */ 100 private static final char UNIX_SEPARATOR = '/'; 101 102 /** 103 * The Windows separator character. 104 */ 105 private static final char WINDOWS_SEPARATOR = '\\'; 106 107 /** 108 * The system separator character. 109 */ 110 private static final char SYSTEM_SEPARATOR = File.separatorChar; 111 112 /** 113 * The separator character that is the opposite of the system separator. 114 */ 115 private static final char OTHER_SEPARATOR; 116 static { 117 if (isSystemWindows()) { 118 OTHER_SEPARATOR = UNIX_SEPARATOR; 119 } else { 120 OTHER_SEPARATOR = WINDOWS_SEPARATOR; 121 } 122 } 123 124 /** 125 * Instances should NOT be constructed in standard programming. 126 */ 127 public FilenameUtils() { 128 super(); 129 } 130 131 //----------------------------------------------------------------------- 132 /** 133 * Determines if Windows file system is in use. 134 * 135 * @return true if the system is Windows 136 */ 137 static boolean isSystemWindows() { 138 return SYSTEM_SEPARATOR == WINDOWS_SEPARATOR; 139 } 140 141 //----------------------------------------------------------------------- 142 /** 143 * Checks if the character is a separator. 144 * 145 * @param ch the character to check 146 * @return true if it is a separator character 147 */ 148 private static boolean isSeparator(char ch) { 149 return ch == UNIX_SEPARATOR || ch == WINDOWS_SEPARATOR; 150 } 151 152 //----------------------------------------------------------------------- 153 /** 154 * Normalizes a path, removing double and single dot path steps. 155 * <p> 156 * This method normalizes a path to a standard format. 157 * The input may contain separators in either Unix or Windows format. 158 * The output will contain separators in the format of the system. 159 * <p> 160 * A trailing slash will be retained. 161 * A double slash will be merged to a single slash (but UNC names are handled). 162 * A single dot path segment will be removed. 163 * A double dot will cause that path segment and the one before to be removed. 164 * If the double dot has no parent path segment to work with, {@code null} 165 * is returned. 166 * <p> 167 * The output will be the same on both Unix and Windows except 168 * for the separator character. 169 * <pre> 170 * /foo// --> /foo/ 171 * /foo/./ --> /foo/ 172 * /foo/../bar --> /bar 173 * /foo/../bar/ --> /bar/ 174 * /foo/../bar/../baz --> /baz 175 * //foo//./bar --> /foo/bar 176 * /../ --> null 177 * ../foo --> null 178 * foo/bar/.. --> foo/ 179 * foo/../../bar --> null 180 * foo/../bar --> bar 181 * //server/foo/../bar --> //server/bar 182 * //server/../bar --> null 183 * C:\foo\..\bar --> C:\bar 184 * C:\..\bar --> null 185 * ~/foo/../bar/ --> ~/bar/ 186 * ~/../bar --> null 187 * </pre> 188 * (Note the file separator returned will be correct for Windows/Unix) 189 * 190 * @param filename the filename to normalize, null returns null 191 * @return the normalized filename, or null if invalid 192 */ 193 public static String normalize(String filename) { 194 return doNormalize(filename, SYSTEM_SEPARATOR, true); 195 } 196 /** 197 * Normalizes a path, removing double and single dot path steps. 198 * <p> 199 * This method normalizes a path to a standard format. 200 * The input may contain separators in either Unix or Windows format. 201 * The output will contain separators in the format specified. 202 * <p> 203 * A trailing slash will be retained. 204 * A double slash will be merged to a single slash (but UNC names are handled). 205 * A single dot path segment will be removed. 206 * A double dot will cause that path segment and the one before to be removed. 207 * If the double dot has no parent path segment to work with, {@code null} 208 * is returned. 209 * <p> 210 * The output will be the same on both Unix and Windows except 211 * for the separator character. 212 * <pre> 213 * /foo// --> /foo/ 214 * /foo/./ --> /foo/ 215 * /foo/../bar --> /bar 216 * /foo/../bar/ --> /bar/ 217 * /foo/../bar/../baz --> /baz 218 * //foo//./bar --> /foo/bar 219 * /../ --> null 220 * ../foo --> null 221 * foo/bar/.. --> foo/ 222 * foo/../../bar --> null 223 * foo/../bar --> bar 224 * //server/foo/../bar --> //server/bar 225 * //server/../bar --> null 226 * C:\foo\..\bar --> C:\bar 227 * C:\..\bar --> null 228 * ~/foo/../bar/ --> ~/bar/ 229 * ~/../bar --> null 230 * </pre> 231 * The output will be the same on both Unix and Windows including 232 * the separator character. 233 * 234 * @param filename the filename to normalize, null returns null 235 * @param unixSeparator {@code true} if a unix separator should 236 * be used or {@code false} if a windows separator should be used. 237 * @return the normalized filename, or null if invalid 238 * @since 2.0 239 */ 240 public static String normalize(String filename, boolean unixSeparator) { 241 char separator = unixSeparator ? UNIX_SEPARATOR : WINDOWS_SEPARATOR; 242 return doNormalize(filename, separator, true); 243 } 244 245 //----------------------------------------------------------------------- 246 /** 247 * Normalizes a path, removing double and single dot path steps, 248 * and removing any final directory separator. 249 * <p> 250 * This method normalizes a path to a standard format. 251 * The input may contain separators in either Unix or Windows format. 252 * The output will contain separators in the format of the system. 253 * <p> 254 * A trailing slash will be removed. 255 * A double slash will be merged to a single slash (but UNC names are handled). 256 * A single dot path segment will be removed. 257 * A double dot will cause that path segment and the one before to be removed. 258 * If the double dot has no parent path segment to work with, {@code null} 259 * is returned. 260 * <p> 261 * The output will be the same on both Unix and Windows except 262 * for the separator character. 263 * <pre> 264 * /foo// --> /foo 265 * /foo/./ --> /foo 266 * /foo/../bar --> /bar 267 * /foo/../bar/ --> /bar 268 * /foo/../bar/../baz --> /baz 269 * //foo//./bar --> /foo/bar 270 * /../ --> null 271 * ../foo --> null 272 * foo/bar/.. --> foo 273 * foo/../../bar --> null 274 * foo/../bar --> bar 275 * //server/foo/../bar --> //server/bar 276 * //server/../bar --> null 277 * C:\foo\..\bar --> C:\bar 278 * C:\..\bar --> null 279 * ~/foo/../bar/ --> ~/bar 280 * ~/../bar --> null 281 * </pre> 282 * (Note the file separator returned will be correct for Windows/Unix) 283 * 284 * @param filename the filename to normalize, null returns null 285 * @return the normalized filename, or null if invalid 286 */ 287 public static String normalizeNoEndSeparator(String filename) { 288 return doNormalize(filename, SYSTEM_SEPARATOR, false); 289 } 290 291 /** 292 * Normalizes a path, removing double and single dot path steps, 293 * and removing any final directory separator. 294 * <p> 295 * This method normalizes a path to a standard format. 296 * The input may contain separators in either Unix or Windows format. 297 * The output will contain separators in the format specified. 298 * <p> 299 * A trailing slash will be removed. 300 * A double slash will be merged to a single slash (but UNC names are handled). 301 * A single dot path segment will be removed. 302 * A double dot will cause that path segment and the one before to be removed. 303 * If the double dot has no parent path segment to work with, {@code null} 304 * is returned. 305 * <p> 306 * The output will be the same on both Unix and Windows including 307 * the separator character. 308 * <pre> 309 * /foo// --> /foo 310 * /foo/./ --> /foo 311 * /foo/../bar --> /bar 312 * /foo/../bar/ --> /bar 313 * /foo/../bar/../baz --> /baz 314 * //foo//./bar --> /foo/bar 315 * /../ --> null 316 * ../foo --> null 317 * foo/bar/.. --> foo 318 * foo/../../bar --> null 319 * foo/../bar --> bar 320 * //server/foo/../bar --> //server/bar 321 * //server/../bar --> null 322 * C:\foo\..\bar --> C:\bar 323 * C:\..\bar --> null 324 * ~/foo/../bar/ --> ~/bar 325 * ~/../bar --> null 326 * </pre> 327 * 328 * @param filename the filename to normalize, null returns null 329 * @param unixSeparator {@code true} if a unix separator should 330 * be used or {@code false} if a windows separtor should be used. 331 * @return the normalized filename, or null if invalid 332 * @since 2.0 333 */ 334 public static String normalizeNoEndSeparator(String filename, boolean unixSeparator) { 335 char separator = unixSeparator ? UNIX_SEPARATOR : WINDOWS_SEPARATOR; 336 return doNormalize(filename, separator, false); 337 } 338 339 /** 340 * Internal method to perform the normalization. 341 * 342 * @param filename the filename 343 * @param separator The separator character to use 344 * @param keepSeparator true to keep the final separator 345 * @return the normalized filename 346 */ 347 private static String doNormalize(String filename, char separator, boolean keepSeparator) { 348 if (filename == null) { 349 return null; 350 } 351 int size = filename.length(); 352 if (size == 0) { 353 return filename; 354 } 355 int prefix = getPrefixLength(filename); 356 if (prefix < 0) { 357 return null; 358 } 359 360 char[] array = new char[size + 2]; // +1 for possible extra slash, +2 for arraycopy 361 filename.getChars(0, filename.length(), array, 0); 362 363 // fix separators throughout 364 char otherSeparator = separator == SYSTEM_SEPARATOR ? OTHER_SEPARATOR : SYSTEM_SEPARATOR; 365 for (int i = 0; i < array.length; i++) { 366 if (array[i] == otherSeparator) { 367 array[i] = separator; 368 } 369 } 370 371 // add extra separator on the end to simplify code below 372 boolean lastIsDirectory = true; 373 if (array[size - 1] != separator) { 374 array[size++] = separator; 375 lastIsDirectory = false; 376 } 377 378 // adjoining slashes 379 for (int i = prefix + 1; i < size; i++) { 380 if (array[i] == separator && array[i - 1] == separator) { 381 System.arraycopy(array, i, array, i - 1, size - i); 382 size--; 383 i--; 384 } 385 } 386 387 // dot slash 388 for (int i = prefix + 1; i < size; i++) { 389 if (array[i] == separator && array[i - 1] == '.' && 390 (i == prefix + 1 || array[i - 2] == separator)) { 391 if (i == size - 1) { 392 lastIsDirectory = true; 393 } 394 System.arraycopy(array, i + 1, array, i - 1, size - i); 395 size -=2; 396 i--; 397 } 398 } 399 400 // double dot slash 401 outer: 402 for (int i = prefix + 2; i < size; i++) { 403 if (array[i] == separator && array[i - 1] == '.' && array[i - 2] == '.' && 404 (i == prefix + 2 || array[i - 3] == separator)) { 405 if (i == prefix + 2) { 406 return null; 407 } 408 if (i == size - 1) { 409 lastIsDirectory = true; 410 } 411 int j; 412 for (j = i - 4 ; j >= prefix; j--) { 413 if (array[j] == separator) { 414 // remove b/../ from a/b/../c 415 System.arraycopy(array, i + 1, array, j + 1, size - i); 416 size -= i - j; 417 i = j + 1; 418 continue outer; 419 } 420 } 421 // remove a/../ from a/../c 422 System.arraycopy(array, i + 1, array, prefix, size - i); 423 size -= i + 1 - prefix; 424 i = prefix + 1; 425 } 426 } 427 428 if (size <= 0) { // should never be less than 0 429 return ""; 430 } 431 if (size <= prefix) { // should never be less than prefix 432 return new String(array, 0, size); 433 } 434 if (lastIsDirectory && keepSeparator) { 435 return new String(array, 0, size); // keep trailing separator 436 } 437 return new String(array, 0, size - 1); // lose trailing separator 438 } 439 440 //----------------------------------------------------------------------- 441 /** 442 * Concatenates a filename to a base path using normal command line style rules. 443 * <p> 444 * The effect is equivalent to resultant directory after changing 445 * directory to the first argument, followed by changing directory to 446 * the second argument. 447 * <p> 448 * The first argument is the base path, the second is the path to concatenate. 449 * The returned path is always normalized via {@link #normalize(String)}, 450 * thus <code>..</code> is handled. 451 * <p> 452 * If <code>pathToAdd</code> is absolute (has an absolute prefix), then 453 * it will be normalized and returned. 454 * Otherwise, the paths will be joined, normalized and returned. 455 * <p> 456 * The output will be the same on both Unix and Windows except 457 * for the separator character. 458 * <pre> 459 * /foo/ + bar --> /foo/bar 460 * /foo + bar --> /foo/bar 461 * /foo + /bar --> /bar 462 * /foo + C:/bar --> C:/bar 463 * /foo + C:bar --> C:bar (*) 464 * /foo/a/ + ../bar --> foo/bar 465 * /foo/ + ../../bar --> null 466 * /foo/ + /bar --> /bar 467 * /foo/.. + /bar --> /bar 468 * /foo + bar/c.txt --> /foo/bar/c.txt 469 * /foo/c.txt + bar --> /foo/c.txt/bar (!) 470 * </pre> 471 * (*) Note that the Windows relative drive prefix is unreliable when 472 * used with this method. 473 * (!) Note that the first parameter must be a path. If it ends with a name, then 474 * the name will be built into the concatenated path. If this might be a problem, 475 * use {@link #getFullPath(String)} on the base path argument. 476 * 477 * @param basePath the base path to attach to, always treated as a path 478 * @param fullFilenameToAdd the filename (or path) to attach to the base 479 * @return the concatenated path, or null if invalid 480 */ 481 public static String concat(String basePath, String fullFilenameToAdd) { 482 int prefix = getPrefixLength(fullFilenameToAdd); 483 if (prefix < 0) { 484 return null; 485 } 486 if (prefix > 0) { 487 return normalize(fullFilenameToAdd); 488 } 489 if (basePath == null) { 490 return null; 491 } 492 int len = basePath.length(); 493 if (len == 0) { 494 return normalize(fullFilenameToAdd); 495 } 496 char ch = basePath.charAt(len - 1); 497 if (isSeparator(ch)) { 498 return normalize(basePath + fullFilenameToAdd); 499 } else { 500 return normalize(basePath + '/' + fullFilenameToAdd); 501 } 502 } 503 504 /** 505 * Determines whether the {@code parent} directory contains the {@code child} element (a file or directory). 506 * <p> 507 * The files names are expected to be normalized. 508 * </p> 509 * 510 * Edge cases: 511 * <ul> 512 * <li>A {@code directory} must not be null: if null, throw IllegalArgumentException</li> 513 * <li>A directory does not contain itself: return false</li> 514 * <li>A null child file is not contained in any parent: return false</li> 515 * </ul> 516 * 517 * @param canonicalParent 518 * the file to consider as the parent. 519 * @param canonicalChild 520 * the file to consider as the child. 521 * @return true is the candidate leaf is under by the specified composite. False otherwise. 522 * @throws IOException 523 * if an IO error occurs while checking the files. 524 * @since 2.2 525 * @see FileUtils#directoryContains(File, File) 526 */ 527 public static boolean directoryContains(final String canonicalParent, final String canonicalChild) 528 throws IOException { 529 530 // Fail fast against NullPointerException 531 if (canonicalParent == null) { 532 throw new IllegalArgumentException("Directory must not be null"); 533 } 534 535 if (canonicalChild == null) { 536 return false; 537 } 538 539 if (IOCase.SYSTEM.checkEquals(canonicalParent, canonicalChild)) { 540 return false; 541 } 542 543 return IOCase.SYSTEM.checkStartsWith(canonicalChild, canonicalParent); 544 } 545 546 //----------------------------------------------------------------------- 547 /** 548 * Converts all separators to the Unix separator of forward slash. 549 * 550 * @param path the path to be changed, null ignored 551 * @return the updated path 552 */ 553 public static String separatorsToUnix(String path) { 554 if (path == null || path.indexOf(WINDOWS_SEPARATOR) == -1) { 555 return path; 556 } 557 return path.replace(WINDOWS_SEPARATOR, UNIX_SEPARATOR); 558 } 559 560 /** 561 * Converts all separators to the Windows separator of backslash. 562 * 563 * @param path the path to be changed, null ignored 564 * @return the updated path 565 */ 566 public static String separatorsToWindows(String path) { 567 if (path == null || path.indexOf(UNIX_SEPARATOR) == -1) { 568 return path; 569 } 570 return path.replace(UNIX_SEPARATOR, WINDOWS_SEPARATOR); 571 } 572 573 /** 574 * Converts all separators to the system separator. 575 * 576 * @param path the path to be changed, null ignored 577 * @return the updated path 578 */ 579 public static String separatorsToSystem(String path) { 580 if (path == null) { 581 return null; 582 } 583 if (isSystemWindows()) { 584 return separatorsToWindows(path); 585 } else { 586 return separatorsToUnix(path); 587 } 588 } 589 590 //----------------------------------------------------------------------- 591 /** 592 * Returns the length of the filename prefix, such as <code>C:/</code> or <code>~/</code>. 593 * <p> 594 * This method will handle a file in either Unix or Windows format. 595 * <p> 596 * The prefix length includes the first slash in the full filename 597 * if applicable. Thus, it is possible that the length returned is greater 598 * than the length of the input string. 599 * <pre> 600 * Windows: 601 * a\b\c.txt --> "" --> relative 602 * \a\b\c.txt --> "\" --> current drive absolute 603 * C:a\b\c.txt --> "C:" --> drive relative 604 * C:\a\b\c.txt --> "C:\" --> absolute 605 * \\server\a\b\c.txt --> "\\server\" --> UNC 606 * 607 * Unix: 608 * a/b/c.txt --> "" --> relative 609 * /a/b/c.txt --> "/" --> absolute 610 * ~/a/b/c.txt --> "~/" --> current user 611 * ~ --> "~/" --> current user (slash added) 612 * ~user/a/b/c.txt --> "~user/" --> named user 613 * ~user --> "~user/" --> named user (slash added) 614 * </pre> 615 * <p> 616 * The output will be the same irrespective of the machine that the code is running on. 617 * ie. both Unix and Windows prefixes are matched regardless. 618 * 619 * @param filename the filename to find the prefix in, null returns -1 620 * @return the length of the prefix, -1 if invalid or null 621 */ 622 public static int getPrefixLength(String filename) { 623 if (filename == null) { 624 return -1; 625 } 626 int len = filename.length(); 627 if (len == 0) { 628 return 0; 629 } 630 char ch0 = filename.charAt(0); 631 if (ch0 == ':') { 632 return -1; 633 } 634 if (len == 1) { 635 if (ch0 == '~') { 636 return 2; // return a length greater than the input 637 } 638 return isSeparator(ch0) ? 1 : 0; 639 } else { 640 if (ch0 == '~') { 641 int posUnix = filename.indexOf(UNIX_SEPARATOR, 1); 642 int posWin = filename.indexOf(WINDOWS_SEPARATOR, 1); 643 if (posUnix == -1 && posWin == -1) { 644 return len + 1; // return a length greater than the input 645 } 646 posUnix = posUnix == -1 ? posWin : posUnix; 647 posWin = posWin == -1 ? posUnix : posWin; 648 return Math.min(posUnix, posWin) + 1; 649 } 650 char ch1 = filename.charAt(1); 651 if (ch1 == ':') { 652 ch0 = Character.toUpperCase(ch0); 653 if (ch0 >= 'A' && ch0 <= 'Z') { 654 if (len == 2 || isSeparator(filename.charAt(2)) == false) { 655 return 2; 656 } 657 return 3; 658 } 659 return -1; 660 661 } else if (isSeparator(ch0) && isSeparator(ch1)) { 662 int posUnix = filename.indexOf(UNIX_SEPARATOR, 2); 663 int posWin = filename.indexOf(WINDOWS_SEPARATOR, 2); 664 if (posUnix == -1 && posWin == -1 || posUnix == 2 || posWin == 2) { 665 return -1; 666 } 667 posUnix = posUnix == -1 ? posWin : posUnix; 668 posWin = posWin == -1 ? posUnix : posWin; 669 return Math.min(posUnix, posWin) + 1; 670 } else { 671 return isSeparator(ch0) ? 1 : 0; 672 } 673 } 674 } 675 676 /** 677 * Returns the index of the last directory separator character. 678 * <p> 679 * This method will handle a file in either Unix or Windows format. 680 * The position of the last forward or backslash is returned. 681 * <p> 682 * The output will be the same irrespective of the machine that the code is running on. 683 * 684 * @param filename the filename to find the last path separator in, null returns -1 685 * @return the index of the last separator character, or -1 if there 686 * is no such character 687 */ 688 public static int indexOfLastSeparator(String filename) { 689 if (filename == null) { 690 return -1; 691 } 692 int lastUnixPos = filename.lastIndexOf(UNIX_SEPARATOR); 693 int lastWindowsPos = filename.lastIndexOf(WINDOWS_SEPARATOR); 694 return Math.max(lastUnixPos, lastWindowsPos); 695 } 696 697 /** 698 * Returns the index of the last extension separator character, which is a dot. 699 * <p> 700 * This method also checks that there is no directory separator after the last dot. 701 * To do this it uses {@link #indexOfLastSeparator(String)} which will 702 * handle a file in either Unix or Windows format. 703 * <p> 704 * The output will be the same irrespective of the machine that the code is running on. 705 * 706 * @param filename the filename to find the last path separator in, null returns -1 707 * @return the index of the last separator character, or -1 if there 708 * is no such character 709 */ 710 public static int indexOfExtension(String filename) { 711 if (filename == null) { 712 return -1; 713 } 714 int extensionPos = filename.lastIndexOf(EXTENSION_SEPARATOR); 715 int lastSeparator = indexOfLastSeparator(filename); 716 return lastSeparator > extensionPos ? -1 : extensionPos; 717 } 718 719 //----------------------------------------------------------------------- 720 /** 721 * Gets the prefix from a full filename, such as <code>C:/</code> 722 * or <code>~/</code>. 723 * <p> 724 * This method will handle a file in either Unix or Windows format. 725 * The prefix includes the first slash in the full filename where applicable. 726 * <pre> 727 * Windows: 728 * a\b\c.txt --> "" --> relative 729 * \a\b\c.txt --> "\" --> current drive absolute 730 * C:a\b\c.txt --> "C:" --> drive relative 731 * C:\a\b\c.txt --> "C:\" --> absolute 732 * \\server\a\b\c.txt --> "\\server\" --> UNC 733 * 734 * Unix: 735 * a/b/c.txt --> "" --> relative 736 * /a/b/c.txt --> "/" --> absolute 737 * ~/a/b/c.txt --> "~/" --> current user 738 * ~ --> "~/" --> current user (slash added) 739 * ~user/a/b/c.txt --> "~user/" --> named user 740 * ~user --> "~user/" --> named user (slash added) 741 * </pre> 742 * <p> 743 * The output will be the same irrespective of the machine that the code is running on. 744 * ie. both Unix and Windows prefixes are matched regardless. 745 * 746 * @param filename the filename to query, null returns null 747 * @return the prefix of the file, null if invalid 748 */ 749 public static String getPrefix(String filename) { 750 if (filename == null) { 751 return null; 752 } 753 int len = getPrefixLength(filename); 754 if (len < 0) { 755 return null; 756 } 757 if (len > filename.length()) { 758 return filename + UNIX_SEPARATOR; // we know this only happens for unix 759 } 760 return filename.substring(0, len); 761 } 762 763 /** 764 * Gets the path from a full filename, which excludes the prefix. 765 * <p> 766 * This method will handle a file in either Unix or Windows format. 767 * The method is entirely text based, and returns the text before and 768 * including the last forward or backslash. 769 * <pre> 770 * C:\a\b\c.txt --> a\b\ 771 * ~/a/b/c.txt --> a/b/ 772 * a.txt --> "" 773 * a/b/c --> a/b/ 774 * a/b/c/ --> a/b/c/ 775 * </pre> 776 * <p> 777 * The output will be the same irrespective of the machine that the code is running on. 778 * <p> 779 * This method drops the prefix from the result. 780 * See {@link #getFullPath(String)} for the method that retains the prefix. 781 * 782 * @param filename the filename to query, null returns null 783 * @return the path of the file, an empty string if none exists, null if invalid 784 */ 785 public static String getPath(String filename) { 786 return doGetPath(filename, 1); 787 } 788 789 /** 790 * Gets the path from a full filename, which excludes the prefix, and 791 * also excluding the final directory separator. 792 * <p> 793 * This method will handle a file in either Unix or Windows format. 794 * The method is entirely text based, and returns the text before the 795 * last forward or backslash. 796 * <pre> 797 * C:\a\b\c.txt --> a\b 798 * ~/a/b/c.txt --> a/b 799 * a.txt --> "" 800 * a/b/c --> a/b 801 * a/b/c/ --> a/b/c 802 * </pre> 803 * <p> 804 * The output will be the same irrespective of the machine that the code is running on. 805 * <p> 806 * This method drops the prefix from the result. 807 * See {@link #getFullPathNoEndSeparator(String)} for the method that retains the prefix. 808 * 809 * @param filename the filename to query, null returns null 810 * @return the path of the file, an empty string if none exists, null if invalid 811 */ 812 public static String getPathNoEndSeparator(String filename) { 813 return doGetPath(filename, 0); 814 } 815 816 /** 817 * Does the work of getting the path. 818 * 819 * @param filename the filename 820 * @param separatorAdd 0 to omit the end separator, 1 to return it 821 * @return the path 822 */ 823 private static String doGetPath(String filename, int separatorAdd) { 824 if (filename == null) { 825 return null; 826 } 827 int prefix = getPrefixLength(filename); 828 if (prefix < 0) { 829 return null; 830 } 831 int index = indexOfLastSeparator(filename); 832 int endIndex = index+separatorAdd; 833 if (prefix >= filename.length() || index < 0 || prefix >= endIndex) { 834 return ""; 835 } 836 return filename.substring(prefix, endIndex); 837 } 838 839 /** 840 * Gets the full path from a full filename, which is the prefix + path. 841 * <p> 842 * This method will handle a file in either Unix or Windows format. 843 * The method is entirely text based, and returns the text before and 844 * including the last forward or backslash. 845 * <pre> 846 * C:\a\b\c.txt --> C:\a\b\ 847 * ~/a/b/c.txt --> ~/a/b/ 848 * a.txt --> "" 849 * a/b/c --> a/b/ 850 * a/b/c/ --> a/b/c/ 851 * C: --> C: 852 * C:\ --> C:\ 853 * ~ --> ~/ 854 * ~/ --> ~/ 855 * ~user --> ~user/ 856 * ~user/ --> ~user/ 857 * </pre> 858 * <p> 859 * The output will be the same irrespective of the machine that the code is running on. 860 * 861 * @param filename the filename to query, null returns null 862 * @return the path of the file, an empty string if none exists, null if invalid 863 */ 864 public static String getFullPath(String filename) { 865 return doGetFullPath(filename, true); 866 } 867 868 /** 869 * Gets the full path from a full filename, which is the prefix + path, 870 * and also excluding the final directory separator. 871 * <p> 872 * This method will handle a file in either Unix or Windows format. 873 * The method is entirely text based, and returns the text before the 874 * last forward or backslash. 875 * <pre> 876 * C:\a\b\c.txt --> C:\a\b 877 * ~/a/b/c.txt --> ~/a/b 878 * a.txt --> "" 879 * a/b/c --> a/b 880 * a/b/c/ --> a/b/c 881 * C: --> C: 882 * C:\ --> C:\ 883 * ~ --> ~ 884 * ~/ --> ~ 885 * ~user --> ~user 886 * ~user/ --> ~user 887 * </pre> 888 * <p> 889 * The output will be the same irrespective of the machine that the code is running on. 890 * 891 * @param filename the filename to query, null returns null 892 * @return the path of the file, an empty string if none exists, null if invalid 893 */ 894 public static String getFullPathNoEndSeparator(String filename) { 895 return doGetFullPath(filename, false); 896 } 897 898 /** 899 * Does the work of getting the path. 900 * 901 * @param filename the filename 902 * @param includeSeparator true to include the end separator 903 * @return the path 904 */ 905 private static String doGetFullPath(String filename, boolean includeSeparator) { 906 if (filename == null) { 907 return null; 908 } 909 int prefix = getPrefixLength(filename); 910 if (prefix < 0) { 911 return null; 912 } 913 if (prefix >= filename.length()) { 914 if (includeSeparator) { 915 return getPrefix(filename); // add end slash if necessary 916 } else { 917 return filename; 918 } 919 } 920 int index = indexOfLastSeparator(filename); 921 if (index < 0) { 922 return filename.substring(0, prefix); 923 } 924 int end = index + (includeSeparator ? 1 : 0); 925 if (end == 0) { 926 end++; 927 } 928 return filename.substring(0, end); 929 } 930 931 /** 932 * Gets the name minus the path from a full filename. 933 * <p> 934 * This method will handle a file in either Unix or Windows format. 935 * The text after the last forward or backslash is returned. 936 * <pre> 937 * a/b/c.txt --> c.txt 938 * a.txt --> a.txt 939 * a/b/c --> c 940 * a/b/c/ --> "" 941 * </pre> 942 * <p> 943 * The output will be the same irrespective of the machine that the code is running on. 944 * 945 * @param filename the filename to query, null returns null 946 * @return the name of the file without the path, or an empty string if none exists 947 */ 948 public static String getName(String filename) { 949 if (filename == null) { 950 return null; 951 } 952 int index = indexOfLastSeparator(filename); 953 return filename.substring(index + 1); 954 } 955 956 /** 957 * Gets the base name, minus the full path and extension, from a full filename. 958 * <p> 959 * This method will handle a file in either Unix or Windows format. 960 * The text after the last forward or backslash and before the last dot is returned. 961 * <pre> 962 * a/b/c.txt --> c 963 * a.txt --> a 964 * a/b/c --> c 965 * a/b/c/ --> "" 966 * </pre> 967 * <p> 968 * The output will be the same irrespective of the machine that the code is running on. 969 * 970 * @param filename the filename to query, null returns null 971 * @return the name of the file without the path, or an empty string if none exists 972 */ 973 public static String getBaseName(String filename) { 974 return removeExtension(getName(filename)); 975 } 976 977 /** 978 * Gets the extension of a filename. 979 * <p> 980 * This method returns the textual part of the filename after the last dot. 981 * There must be no directory separator after the dot. 982 * <pre> 983 * foo.txt --> "txt" 984 * a/b/c.jpg --> "jpg" 985 * a/b.txt/c --> "" 986 * a/b/c --> "" 987 * </pre> 988 * <p> 989 * The output will be the same irrespective of the machine that the code is running on. 990 * 991 * @param filename the filename to retrieve the extension of. 992 * @return the extension of the file or an empty string if none exists or {@code null} 993 * if the filename is {@code null}. 994 */ 995 public static String getExtension(String filename) { 996 if (filename == null) { 997 return null; 998 } 999 int index = indexOfExtension(filename); 1000 if (index == -1) { 1001 return ""; 1002 } else { 1003 return filename.substring(index + 1); 1004 } 1005 } 1006 1007 //----------------------------------------------------------------------- 1008 /** 1009 * Removes the extension from a filename. 1010 * <p> 1011 * This method returns the textual part of the filename before the last dot. 1012 * There must be no directory separator after the dot. 1013 * <pre> 1014 * foo.txt --> foo 1015 * a\b\c.jpg --> a\b\c 1016 * a\b\c --> a\b\c 1017 * a.b\c --> a.b\c 1018 * </pre> 1019 * <p> 1020 * The output will be the same irrespective of the machine that the code is running on. 1021 * 1022 * @param filename the filename to query, null returns null 1023 * @return the filename minus the extension 1024 */ 1025 public static String removeExtension(String filename) { 1026 if (filename == null) { 1027 return null; 1028 } 1029 int index = indexOfExtension(filename); 1030 if (index == -1) { 1031 return filename; 1032 } else { 1033 return filename.substring(0, index); 1034 } 1035 } 1036 1037 //----------------------------------------------------------------------- 1038 /** 1039 * Checks whether two filenames are equal exactly. 1040 * <p> 1041 * No processing is performed on the filenames other than comparison, 1042 * thus this is merely a null-safe case-sensitive equals. 1043 * 1044 * @param filename1 the first filename to query, may be null 1045 * @param filename2 the second filename to query, may be null 1046 * @return true if the filenames are equal, null equals null 1047 * @see IOCase#SENSITIVE 1048 */ 1049 public static boolean equals(String filename1, String filename2) { 1050 return equals(filename1, filename2, false, IOCase.SENSITIVE); 1051 } 1052 1053 /** 1054 * Checks whether two filenames are equal using the case rules of the system. 1055 * <p> 1056 * No processing is performed on the filenames other than comparison. 1057 * The check is case-sensitive on Unix and case-insensitive on Windows. 1058 * 1059 * @param filename1 the first filename to query, may be null 1060 * @param filename2 the second filename to query, may be null 1061 * @return true if the filenames are equal, null equals null 1062 * @see IOCase#SYSTEM 1063 */ 1064 public static boolean equalsOnSystem(String filename1, String filename2) { 1065 return equals(filename1, filename2, false, IOCase.SYSTEM); 1066 } 1067 1068 //----------------------------------------------------------------------- 1069 /** 1070 * Checks whether two filenames are equal after both have been normalized. 1071 * <p> 1072 * Both filenames are first passed to {@link #normalize(String)}. 1073 * The check is then performed in a case-sensitive manner. 1074 * 1075 * @param filename1 the first filename to query, may be null 1076 * @param filename2 the second filename to query, may be null 1077 * @return true if the filenames are equal, null equals null 1078 * @see IOCase#SENSITIVE 1079 */ 1080 public static boolean equalsNormalized(String filename1, String filename2) { 1081 return equals(filename1, filename2, true, IOCase.SENSITIVE); 1082 } 1083 1084 /** 1085 * Checks whether two filenames are equal after both have been normalized 1086 * and using the case rules of the system. 1087 * <p> 1088 * Both filenames are first passed to {@link #normalize(String)}. 1089 * The check is then performed case-sensitive on Unix and 1090 * case-insensitive on Windows. 1091 * 1092 * @param filename1 the first filename to query, may be null 1093 * @param filename2 the second filename to query, may be null 1094 * @return true if the filenames are equal, null equals null 1095 * @see IOCase#SYSTEM 1096 */ 1097 public static boolean equalsNormalizedOnSystem(String filename1, String filename2) { 1098 return equals(filename1, filename2, true, IOCase.SYSTEM); 1099 } 1100 1101 /** 1102 * Checks whether two filenames are equal, optionally normalizing and providing 1103 * control over the case-sensitivity. 1104 * 1105 * @param filename1 the first filename to query, may be null 1106 * @param filename2 the second filename to query, may be null 1107 * @param normalized whether to normalize the filenames 1108 * @param caseSensitivity what case sensitivity rule to use, null means case-sensitive 1109 * @return true if the filenames are equal, null equals null 1110 * @since 1.3 1111 */ 1112 public static boolean equals( 1113 String filename1, String filename2, 1114 boolean normalized, IOCase caseSensitivity) { 1115 1116 if (filename1 == null || filename2 == null) { 1117 return filename1 == null && filename2 == null; 1118 } 1119 if (normalized) { 1120 filename1 = normalize(filename1); 1121 filename2 = normalize(filename2); 1122 if (filename1 == null || filename2 == null) { 1123 throw new NullPointerException( 1124 "Error normalizing one or both of the file names"); 1125 } 1126 } 1127 if (caseSensitivity == null) { 1128 caseSensitivity = IOCase.SENSITIVE; 1129 } 1130 return caseSensitivity.checkEquals(filename1, filename2); 1131 } 1132 1133 //----------------------------------------------------------------------- 1134 /** 1135 * Checks whether the extension of the filename is that specified. 1136 * <p> 1137 * This method obtains the extension as the textual part of the filename 1138 * after the last dot. There must be no directory separator after the dot. 1139 * The extension check is case-sensitive on all platforms. 1140 * 1141 * @param filename the filename to query, null returns false 1142 * @param extension the extension to check for, null or empty checks for no extension 1143 * @return true if the filename has the specified extension 1144 */ 1145 public static boolean isExtension(String filename, String extension) { 1146 if (filename == null) { 1147 return false; 1148 } 1149 if (extension == null || extension.length() == 0) { 1150 return indexOfExtension(filename) == -1; 1151 } 1152 String fileExt = getExtension(filename); 1153 return fileExt.equals(extension); 1154 } 1155 1156 /** 1157 * Checks whether the extension of the filename is one of those specified. 1158 * <p> 1159 * This method obtains the extension as the textual part of the filename 1160 * after the last dot. There must be no directory separator after the dot. 1161 * The extension check is case-sensitive on all platforms. 1162 * 1163 * @param filename the filename to query, null returns false 1164 * @param extensions the extensions to check for, null checks for no extension 1165 * @return true if the filename is one of the extensions 1166 */ 1167 public static boolean isExtension(String filename, String[] extensions) { 1168 if (filename == null) { 1169 return false; 1170 } 1171 if (extensions == null || extensions.length == 0) { 1172 return indexOfExtension(filename) == -1; 1173 } 1174 String fileExt = getExtension(filename); 1175 for (String extension : extensions) { 1176 if (fileExt.equals(extension)) { 1177 return true; 1178 } 1179 } 1180 return false; 1181 } 1182 1183 /** 1184 * Checks whether the extension of the filename is one of those specified. 1185 * <p> 1186 * This method obtains the extension as the textual part of the filename 1187 * after the last dot. There must be no directory separator after the dot. 1188 * The extension check is case-sensitive on all platforms. 1189 * 1190 * @param filename the filename to query, null returns false 1191 * @param extensions the extensions to check for, null checks for no extension 1192 * @return true if the filename is one of the extensions 1193 */ 1194 public static boolean isExtension(String filename, Collection<String> extensions) { 1195 if (filename == null) { 1196 return false; 1197 } 1198 if (extensions == null || extensions.isEmpty()) { 1199 return indexOfExtension(filename) == -1; 1200 } 1201 String fileExt = getExtension(filename); 1202 for (String extension : extensions) { 1203 if (fileExt.equals(extension)) { 1204 return true; 1205 } 1206 } 1207 return false; 1208 } 1209 1210 //----------------------------------------------------------------------- 1211 /** 1212 * Checks a filename to see if it matches the specified wildcard matcher, 1213 * always testing case-sensitive. 1214 * <p> 1215 * The wildcard matcher uses the characters '?' and '*' to represent a 1216 * single or multiple (zero or more) wildcard characters. 1217 * This is the same as often found on Dos/Unix command lines. 1218 * The check is case-sensitive always. 1219 * <pre> 1220 * wildcardMatch("c.txt", "*.txt") --> true 1221 * wildcardMatch("c.txt", "*.jpg") --> false 1222 * wildcardMatch("a/b/c.txt", "a/b/*") --> true 1223 * wildcardMatch("c.txt", "*.???") --> true 1224 * wildcardMatch("c.txt", "*.????") --> false 1225 * </pre> 1226 * N.B. the sequence "*?" does not work properly at present in match strings. 1227 * 1228 * @param filename the filename to match on 1229 * @param wildcardMatcher the wildcard string to match against 1230 * @return true if the filename matches the wilcard string 1231 * @see IOCase#SENSITIVE 1232 */ 1233 public static boolean wildcardMatch(String filename, String wildcardMatcher) { 1234 return wildcardMatch(filename, wildcardMatcher, IOCase.SENSITIVE); 1235 } 1236 1237 /** 1238 * Checks a filename to see if it matches the specified wildcard matcher 1239 * using the case rules of the system. 1240 * <p> 1241 * The wildcard matcher uses the characters '?' and '*' to represent a 1242 * single or multiple (zero or more) wildcard characters. 1243 * This is the same as often found on Dos/Unix command lines. 1244 * The check is case-sensitive on Unix and case-insensitive on Windows. 1245 * <pre> 1246 * wildcardMatch("c.txt", "*.txt") --> true 1247 * wildcardMatch("c.txt", "*.jpg") --> false 1248 * wildcardMatch("a/b/c.txt", "a/b/*") --> true 1249 * wildcardMatch("c.txt", "*.???") --> true 1250 * wildcardMatch("c.txt", "*.????") --> false 1251 * </pre> 1252 * N.B. the sequence "*?" does not work properly at present in match strings. 1253 * 1254 * @param filename the filename to match on 1255 * @param wildcardMatcher the wildcard string to match against 1256 * @return true if the filename matches the wilcard string 1257 * @see IOCase#SYSTEM 1258 */ 1259 public static boolean wildcardMatchOnSystem(String filename, String wildcardMatcher) { 1260 return wildcardMatch(filename, wildcardMatcher, IOCase.SYSTEM); 1261 } 1262 1263 /** 1264 * Checks a filename to see if it matches the specified wildcard matcher 1265 * allowing control over case-sensitivity. 1266 * <p> 1267 * The wildcard matcher uses the characters '?' and '*' to represent a 1268 * single or multiple (zero or more) wildcard characters. 1269 * N.B. the sequence "*?" does not work properly at present in match strings. 1270 * 1271 * @param filename the filename to match on 1272 * @param wildcardMatcher the wildcard string to match against 1273 * @param caseSensitivity what case sensitivity rule to use, null means case-sensitive 1274 * @return true if the filename matches the wilcard string 1275 * @since 1.3 1276 */ 1277 public static boolean wildcardMatch(String filename, String wildcardMatcher, IOCase caseSensitivity) { 1278 if (filename == null && wildcardMatcher == null) { 1279 return true; 1280 } 1281 if (filename == null || wildcardMatcher == null) { 1282 return false; 1283 } 1284 if (caseSensitivity == null) { 1285 caseSensitivity = IOCase.SENSITIVE; 1286 } 1287 String[] wcs = splitOnTokens(wildcardMatcher); 1288 boolean anyChars = false; 1289 int textIdx = 0; 1290 int wcsIdx = 0; 1291 Stack<int[]> backtrack = new Stack<int[]>(); 1292 1293 // loop around a backtrack stack, to handle complex * matching 1294 do { 1295 if (backtrack.size() > 0) { 1296 int[] array = backtrack.pop(); 1297 wcsIdx = array[0]; 1298 textIdx = array[1]; 1299 anyChars = true; 1300 } 1301 1302 // loop whilst tokens and text left to process 1303 while (wcsIdx < wcs.length) { 1304 1305 if (wcs[wcsIdx].equals("?")) { 1306 // ? so move to next text char 1307 textIdx++; 1308 if (textIdx > filename.length()) { 1309 break; 1310 } 1311 anyChars = false; 1312 1313 } else if (wcs[wcsIdx].equals("*")) { 1314 // set any chars status 1315 anyChars = true; 1316 if (wcsIdx == wcs.length - 1) { 1317 textIdx = filename.length(); 1318 } 1319 1320 } else { 1321 // matching text token 1322 if (anyChars) { 1323 // any chars then try to locate text token 1324 textIdx = caseSensitivity.checkIndexOf(filename, textIdx, wcs[wcsIdx]); 1325 if (textIdx == -1) { 1326 // token not found 1327 break; 1328 } 1329 int repeat = caseSensitivity.checkIndexOf(filename, textIdx + 1, wcs[wcsIdx]); 1330 if (repeat >= 0) { 1331 backtrack.push(new int[] {wcsIdx, repeat}); 1332 } 1333 } else { 1334 // matching from current position 1335 if (!caseSensitivity.checkRegionMatches(filename, textIdx, wcs[wcsIdx])) { 1336 // couldnt match token 1337 break; 1338 } 1339 } 1340 1341 // matched text token, move text index to end of matched token 1342 textIdx += wcs[wcsIdx].length(); 1343 anyChars = false; 1344 } 1345 1346 wcsIdx++; 1347 } 1348 1349 // full match 1350 if (wcsIdx == wcs.length && textIdx == filename.length()) { 1351 return true; 1352 } 1353 1354 } while (backtrack.size() > 0); 1355 1356 return false; 1357 } 1358 1359 /** 1360 * Splits a string into a number of tokens. 1361 * The text is split by '?' and '*'. 1362 * Where multiple '*' occur consecutively they are collapsed into a single '*'. 1363 * 1364 * @param text the text to split 1365 * @return the array of tokens, never null 1366 */ 1367 static String[] splitOnTokens(String text) { 1368 // used by wildcardMatch 1369 // package level so a unit test may run on this 1370 1371 if (text.indexOf('?') == -1 && text.indexOf('*') == -1) { 1372 return new String[] { text }; 1373 } 1374 1375 char[] array = text.toCharArray(); 1376 ArrayList<String> list = new ArrayList<String>(); 1377 StringBuilder buffer = new StringBuilder(); 1378 for (int i = 0; i < array.length; i++) { 1379 if (array[i] == '?' || array[i] == '*') { 1380 if (buffer.length() != 0) { 1381 list.add(buffer.toString()); 1382 buffer.setLength(0); 1383 } 1384 if (array[i] == '?') { 1385 list.add("?"); 1386 } else if (list.isEmpty() || 1387 i > 0 && list.get(list.size() - 1).equals("*") == false) { 1388 list.add("*"); 1389 } 1390 } else { 1391 buffer.append(array[i]); 1392 } 1393 } 1394 if (buffer.length() != 0) { 1395 list.add(buffer.toString()); 1396 } 1397 1398 return list.toArray( new String[ list.size() ] ); 1399 } 1400 1401 }