Skip to content

Commit 50d91a3

Browse files
author
Brian Burkhalter
committedNov 1, 2022
8057113: (fs) Path should have a method to obtain the filename extension
Reviewed-by: rriggs, lancea, mr, alanb
1 parent d17bf51 commit 50d91a3

File tree

2 files changed

+145
-2
lines changed

2 files changed

+145
-2
lines changed
 

‎src/java.base/share/classes/java/nio/file/Path.java

+60-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2007, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2007, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -31,6 +31,7 @@
3131
import java.nio.file.spi.FileSystemProvider;
3232
import java.util.Iterator;
3333
import java.util.NoSuchElementException;
34+
import java.util.Objects;
3435

3536
/**
3637
* An object that may be used to locate a file in a file system. It will
@@ -49,7 +50,7 @@
4950
* file system. {@code Path} defines the {@link #getFileName() getFileName},
5051
* {@link #getParent getParent}, {@link #getRoot getRoot}, and {@link #subpath
5152
* subpath} methods to access the path components or a subsequence of its name
52-
* elements.
53+
* elements, and {@link #getExtension() getExtension} to obtain its extension.
5354
*
5455
* <p> In addition to accessing the components of a path, a {@code Path} also
5556
* defines the {@link #resolve(Path) resolve} and {@link #resolveSibling(Path)
@@ -248,6 +249,63 @@ public static Path of(URI uri) {
248249
*/
249250
Path getFileName();
250251

252+
/**
253+
* Returns the file extension of this path's file name as a {@code String}.
254+
* The extension is derived from this {@code Path} by obtaining the
255+
* {@linkplain #getFileName file name element}, deriving its {@linkplain
256+
* #toString string representation}, and then extracting a substring
257+
* determined by the position of a period character ('.', U+002E FULL STOP)
258+
* within the file name string. If the file name element is {@code null},
259+
* or if the file name string does not contain a period character, or if
260+
* the only period in the file name string is its first character, then
261+
* the extension is {@code null}. Otherwise, the extension is the substring
262+
* after the last period in the file name string. If this last period is
263+
* also the last character in the file name string, then the extension is
264+
* {@linkplain String#isEmpty empty}.
265+
*
266+
* @implSpec
267+
* The default implementation is equivalent for this path to:
268+
* <pre>{@code
269+
* int lastPeriod = fileName.lastIndexOf('.');
270+
* if (lastPeriod <= 0)
271+
* return null;
272+
* return (lastPeriod == fileName.length() - 1)
273+
* ? ""
274+
* : fileName.substring(lastPeriod + 1);
275+
* }</pre>
276+
*
277+
* @return the file name extension of this path, which might be the
278+
* empty string, or {@code null} if no extension is found
279+
*
280+
* @since 20
281+
*/
282+
default String getExtension() {
283+
Path fileName = getFileName();
284+
if (fileName == null)
285+
return null;
286+
287+
String fileNameString = fileName.toString();
288+
int length = fileNameString.length();
289+
290+
// An empty or unity length file name string has a null extension
291+
if (length > 1) {
292+
int lastPeriodIndex = fileNameString.lastIndexOf('.');
293+
294+
// Indeterminate if there is no period character or
295+
// only the first character is a period character
296+
if (lastPeriodIndex > 0) {
297+
if (lastPeriodIndex == length - 1) {
298+
// empty string
299+
return "";
300+
} else {
301+
return fileNameString.substring(lastPeriodIndex + 1);
302+
}
303+
}
304+
}
305+
306+
return null;
307+
}
308+
251309
/**
252310
* Returns the <em>parent path</em>, or {@code null} if this path does not
253311
* have a parent.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.nio.file.Path;
25+
26+
import org.testng.Assert;
27+
import org.testng.annotations.DataProvider;
28+
import org.testng.annotations.Test;
29+
30+
/*
31+
* @test
32+
* @bug 8057113
33+
* @summary Verify getExtension method
34+
* @run testng Extensions
35+
*/
36+
public class Extensions {
37+
/**
38+
* Returns path name string and expected extension pairs.
39+
*
40+
* @return {@code {{"pathname", "extension"},...}}
41+
*/
42+
@DataProvider
43+
static Object[][] getProvider() {
44+
Object[][] pairs = {
45+
{"", null},
46+
{".", null},
47+
{"..", ""},
48+
{"...", ""},
49+
{"....", ""},
50+
{".....", ""},
51+
{"aa", null},
52+
{"a.", ""},
53+
{".a", null},
54+
{"..a", "a"},
55+
{"...a", "a"},
56+
{"....a", "a"},
57+
{".a.b", "b"},
58+
{"...a.b", "b"},
59+
{"...a.b.", ""},
60+
{"..foo", "foo"},
61+
{"foo.", ""},
62+
{"test.", ""},
63+
{"test..", ""},
64+
{"test...", ""},
65+
{"test.rb", "rb"},
66+
{"a/b/d/test.rb" , "rb"},
67+
{".a/b/d/test.rb", "rb"},
68+
{"test", null},
69+
{".profile", null},
70+
{".profile.sh", "sh"},
71+
{"foo.tar.gz", "gz"},
72+
{"foo.bar.", ""},
73+
{"archive.zip", "zip"},
74+
{"compress.gzip", "gzip"},
75+
{"waitwhat.&$!#%", "&$!#%"},
76+
{"6.283185307", "283185307"}
77+
};
78+
return pairs;
79+
}
80+
81+
@Test(dataProvider = "getProvider")
82+
public static void get(String pathname, String extension) {
83+
Assert.assertEquals(Path.of(pathname).getExtension(), extension);
84+
}
85+
}

0 commit comments

Comments
 (0)
Please sign in to comment.