1 /* 2 * Copyright (c) 2009, 2013, 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 /* @test 25 * @bug 6844313 8011647 26 * @summary Unit test for java.nio.file.FileTime 27 * @key randomness 28 */ 29 30 31 import java.nio.file.attribute.FileTime; 32 import java.time.Instant; 33 import java.util.concurrent.TimeUnit; 34 import static java.util.concurrent.TimeUnit.*; 35 import java.util.Random; 36 import java.util.EnumSet; 37 38 public class Basic { 39 40 static final Random rand = new Random(); 41 42 public static void main(String[] args) { 43 long now = System.currentTimeMillis(); 44 long tomorrowInDays = TimeUnit.DAYS.convert(now, MILLISECONDS) + 1; 45 long yesterdayInDays = TimeUnit.DAYS.convert(now, MILLISECONDS) - 1; 46 47 Instant nowInstant = Instant.ofEpochMilli(now); 48 49 // equals 50 eq(now, MILLISECONDS, now, MILLISECONDS); 51 eq(now, MILLISECONDS, now*1000L, MICROSECONDS); 52 neq(now, MILLISECONDS, 0, MILLISECONDS); 53 neq(now, MILLISECONDS, 0, MICROSECONDS); 54 55 eq(nowInstant, now, MILLISECONDS); 56 eq(nowInstant, now*1000L, MICROSECONDS); 57 neq(nowInstant, 0, MILLISECONDS); 58 neq(nowInstant, 0, MICROSECONDS); 59 60 // compareTo 61 cmp(now, MILLISECONDS, now, MILLISECONDS, 0); 62 cmp(now, MILLISECONDS, now*1000L, MICROSECONDS, 0); 63 cmp(now, MILLISECONDS, now-1234, MILLISECONDS, 1); 64 cmp(now, MILLISECONDS, now+1234, MILLISECONDS, -1); 65 66 cmp(tomorrowInDays, DAYS, now, MILLISECONDS, 1); 67 cmp(now, MILLISECONDS, tomorrowInDays, DAYS, -1); 68 cmp(yesterdayInDays, DAYS, now, MILLISECONDS, -1); 69 cmp(now, MILLISECONDS, yesterdayInDays, DAYS, 1); 70 cmp(yesterdayInDays, DAYS, now, MILLISECONDS, -1); 71 72 cmp(Long.MAX_VALUE, DAYS, Long.MAX_VALUE, NANOSECONDS, 1); 73 cmp(Long.MAX_VALUE, DAYS, Long.MIN_VALUE, NANOSECONDS, 1); 74 cmp(Long.MIN_VALUE, DAYS, Long.MIN_VALUE, NANOSECONDS, -1); 75 cmp(Long.MIN_VALUE, DAYS, Long.MAX_VALUE, NANOSECONDS, -1); 76 77 cmp(Instant.MIN, Long.MIN_VALUE, DAYS, 1); 78 cmp(Instant.MIN, Long.MIN_VALUE, HOURS, 1); 79 cmp(Instant.MIN, Long.MIN_VALUE, MINUTES, 1); 80 cmp(Instant.MIN, Long.MIN_VALUE, SECONDS, 1); 81 cmp(Instant.MIN, Instant.MIN.getEpochSecond() - 1, SECONDS, 1); 82 cmp(Instant.MIN, Instant.MIN.getEpochSecond() - 100, SECONDS, 1); 83 cmp(Instant.MIN, Instant.MIN.getEpochSecond(), SECONDS, 0); 84 85 cmp(Instant.MAX, Long.MAX_VALUE, DAYS, -1); 86 cmp(Instant.MAX, Long.MAX_VALUE, HOURS, -1); 87 cmp(Instant.MAX, Long.MAX_VALUE, MINUTES, -1); 88 cmp(Instant.MAX, Long.MAX_VALUE, SECONDS, -1); 89 cmp(Instant.MAX, Instant.MAX.getEpochSecond() + 1, SECONDS, -1); 90 cmp(Instant.MAX, Instant.MAX.getEpochSecond() + 100, SECONDS, -1); 91 cmp(Instant.MAX, Instant.MAX.getEpochSecond(), SECONDS, 0); 92 93 cmp(nowInstant, now, MILLISECONDS, 0); 94 cmp(nowInstant, now*1000L, MICROSECONDS, 0); 95 cmp(nowInstant, now-1234, MILLISECONDS, 1); 96 cmp(nowInstant, now+1234, MILLISECONDS, -1); 97 cmp(nowInstant, tomorrowInDays, DAYS, -1); 98 cmp(nowInstant, yesterdayInDays, DAYS, 1); 99 100 // to(TimeUnit) 101 to(MILLISECONDS.convert(1, DAYS) - 1, MILLISECONDS); 102 to(MILLISECONDS.convert(1, DAYS) + 0, MILLISECONDS); 103 to(MILLISECONDS.convert(1, DAYS) + 1, MILLISECONDS); 104 to(1, MILLISECONDS); 105 to(0, MILLISECONDS); 106 to(1, MILLISECONDS); 107 to(MILLISECONDS.convert(-1, DAYS) - 1, MILLISECONDS); 108 to(MILLISECONDS.convert(-1, DAYS) + 0, MILLISECONDS); 109 to(MILLISECONDS.convert(-1, DAYS) + 1, MILLISECONDS); 110 for (TimeUnit unit: TimeUnit.values()) { 111 for (int i=0; i<100; i++) { to(rand.nextLong(), unit); } 112 to(Long.MIN_VALUE, unit); 113 to(Long.MAX_VALUE, unit); 114 } 115 116 // toInstant() 117 int N = 1000; 118 for (TimeUnit unit : EnumSet.allOf(TimeUnit.class)) { 119 for (int i = 0; i < N; i++) { 120 long value = rand.nextLong(); 121 FileTime ft = FileTime.from(value, unit); 122 Instant instant = ft.toInstant(); 123 if (instant != Instant.MIN && instant != Instant.MAX) { 124 eqTime(value, unit, instant); 125 } 126 } 127 } 128 for (TimeUnit unit : EnumSet.allOf(TimeUnit.class)) { 129 long value = Long.MIN_VALUE; 130 FileTime ft = FileTime.from(value, unit); 131 Instant instant = ft.toInstant(); 132 if (unit.compareTo(TimeUnit.SECONDS) < 0) { 133 eqTime(value, unit, instant); 134 } else if (!instant.equals(Instant.MIN)) { 135 throw new RuntimeException("should overflow to MIN"); 136 } 137 value = Long.MAX_VALUE; 138 ft = FileTime.from(value, unit); 139 instant = ft.toInstant(); 140 if (unit.compareTo(TimeUnit.SECONDS) < 0) { 141 eqTime(value, unit, instant); 142 } else if (!instant.equals(Instant.MAX)) { 143 throw new RuntimeException("should overflow to MAX"); 144 } 145 } 146 147 // from(Instant) 148 final long MAX_SECOND = 31556889864403199L; 149 for (int i = 0; i < N; i++) { 150 long v = rand.nextLong(); 151 long secs = v % MAX_SECOND; 152 Instant instant = Instant.ofEpochSecond(secs, rand.nextInt(1000_000_000)); 153 FileTime ft = FileTime.from(instant); 154 if (!ft.toInstant().equals(instant) || ft.to(SECONDS) != secs) { 155 throw new RuntimeException("from(Instant) failed"); 156 } 157 long millis = v; 158 instant = Instant.ofEpochMilli(millis); 159 ft = FileTime.from(instant); 160 if (!ft.toInstant().equals(instant) || 161 ft.toMillis() != instant.toEpochMilli()) { 162 throw new RuntimeException("from(Instant) failed"); 163 } 164 long nanos = v; 165 ft = FileTime.from(nanos, NANOSECONDS); 166 secs = nanos / 1000_000_000; 167 nanos = nanos % 1000_000_000; 168 instant = Instant.ofEpochSecond(secs, nanos); 169 if (!ft.equals(FileTime.from(instant))) { 170 throw new RuntimeException("from(Instant) failed"); 171 } 172 } 173 174 // toString 175 ts(1L, DAYS, "1970-01-02T00:00:00Z"); 176 ts(1L, HOURS, "1970-01-01T01:00:00Z"); 177 ts(1L, MINUTES, "1970-01-01T00:01:00Z"); 178 ts(1L, SECONDS, "1970-01-01T00:00:01Z"); 179 ts(1L, MILLISECONDS, "1970-01-01T00:00:00.001Z"); 180 ts(1L, MICROSECONDS, "1970-01-01T00:00:00.000001Z"); 181 ts(1L, NANOSECONDS, "1970-01-01T00:00:00.000000001Z"); 182 ts(999999999L, NANOSECONDS, "1970-01-01T00:00:00.999999999Z"); 183 ts(9999999999L, NANOSECONDS, "1970-01-01T00:00:09.999999999Z"); 184 185 ts(-1L, DAYS, "1969-12-31T00:00:00Z"); 186 ts(-1L, HOURS, "1969-12-31T23:00:00Z"); 187 ts(-1L, MINUTES, "1969-12-31T23:59:00Z"); 188 ts(-1L, SECONDS, "1969-12-31T23:59:59Z"); 189 ts(-1L, MILLISECONDS, "1969-12-31T23:59:59.999Z"); 190 ts(-1L, MICROSECONDS, "1969-12-31T23:59:59.999999Z"); 191 ts(-1L, NANOSECONDS, "1969-12-31T23:59:59.999999999Z"); 192 ts(-999999999L, NANOSECONDS, "1969-12-31T23:59:59.000000001Z"); 193 ts(-9999999999L, NANOSECONDS, "1969-12-31T23:59:50.000000001Z"); 194 195 ts(-62135596799999L, MILLISECONDS, "0001-01-01T00:00:00.001Z"); 196 ts(-62135596800000L, MILLISECONDS, "0001-01-01T00:00:00Z"); 197 ts(-62135596800001L, MILLISECONDS, "-0001-12-31T23:59:59.999Z"); 198 199 ts(253402300799999L, MILLISECONDS, "9999-12-31T23:59:59.999Z"); 200 ts(-377642044800001L, MILLISECONDS, "-9999-12-31T23:59:59.999Z"); 201 202 // NTFS epoch in usec. 203 ts(-11644473600000000L, MICROSECONDS, "1601-01-01T00:00:00Z"); 204 205 ts(Instant.MIN, "-1000000001-01-01T00:00:00Z"); 206 ts(Instant.MAX, "1000000000-12-31T23:59:59.999999999Z"); 207 208 try { 209 FileTime.from(0L, null); 210 throw new RuntimeException("NullPointerException expected"); 211 } catch (NullPointerException npe) { } 212 try { 213 FileTime.from(null); 214 throw new RuntimeException("NullPointerException expected"); 215 } catch (NullPointerException npe) { } 216 217 FileTime time = FileTime.fromMillis(now); 218 if (time.equals(null)) 219 throw new RuntimeException("should not be equal to null"); 220 try { 221 time.compareTo(null); 222 throw new RuntimeException("NullPointerException expected"); 223 } catch (NullPointerException npe) { } 224 225 // Instant + toMilli() overflow 226 overflow(Long.MAX_VALUE, 227 FileTime.from(Instant.MAX).toMillis()); 228 overflow(Long.MAX_VALUE, 229 FileTime.from(Instant.ofEpochSecond(Long.MAX_VALUE / 1000 + 1)) 230 .toMillis()); 231 overflow(Long.MIN_VALUE, 232 FileTime.from(Instant.MIN).toMillis()); 233 overflow(Long.MIN_VALUE, 234 FileTime.from(Instant.ofEpochSecond(Long.MIN_VALUE / 1000 - 1)) 235 .toMillis()); 236 237 // Instant + to(TimeUnit) overflow 238 overflow(Long.MAX_VALUE, 239 FileTime.from(Instant.ofEpochSecond(Long.MAX_VALUE / 1000 + 1)) 240 .to(MILLISECONDS)); 241 overflow(Long.MAX_VALUE, 242 FileTime.from(Instant.ofEpochSecond(Long.MAX_VALUE / 1000, 243 MILLISECONDS.toNanos(1000))) 244 .to(MILLISECONDS)); 245 overflow(Long.MIN_VALUE, 246 FileTime.from(Instant.ofEpochSecond(Long.MIN_VALUE / 1000 - 1)) 247 .to(MILLISECONDS)); 248 overflow(Long.MIN_VALUE, 249 FileTime.from(Instant.ofEpochSecond(Long.MIN_VALUE / 1000, 250 -MILLISECONDS.toNanos(1))) 251 .to(MILLISECONDS)); 252 } 253 254 static void overflow(long minmax, long v) { 255 if (v != minmax) 256 throw new RuntimeException("saturates to Long.MIN/MAX_VALUE expected"); 257 } 258 259 static void cmp(long v1, TimeUnit u1, long v2, TimeUnit u2, int expected) { 260 int result = FileTime.from(v1, u1).compareTo(FileTime.from(v2, u2)); 261 if (result != expected) 262 throw new RuntimeException("unexpected order"); 263 } 264 265 static void cmp(Instant ins, long v2, TimeUnit u2, int expected) { 266 int result = FileTime.from(ins).compareTo(FileTime.from(v2, u2)); 267 if (result != expected) 268 throw new RuntimeException("unexpected order"); 269 } 270 271 static void eq(long v1, TimeUnit u1, long v2, TimeUnit u2) { 272 FileTime t1 = FileTime.from(v1, u1); 273 FileTime t2 = FileTime.from(v2, u2); 274 if (!t1.equals(t2)) 275 throw new RuntimeException("not equal"); 276 if (t1.hashCode() != t2.hashCode()) 277 throw new RuntimeException("hashCodes should be equal"); 278 } 279 280 static void eq(Instant ins, long v2, TimeUnit u2) { 281 FileTime t1 = FileTime.from(ins); 282 FileTime t2 = FileTime.from(v2, u2); 283 if (!t1.equals(t2)) 284 throw new RuntimeException("not equal"); 285 if (t1.hashCode() != t2.hashCode()) 286 throw new RuntimeException("hashCodes should be equal"); 287 } 288 289 static void eqTime(long value, TimeUnit unit, Instant instant) { 290 long secs = SECONDS.convert(value, unit); 291 long nanos = NANOSECONDS.convert(value - unit.convert(secs, SECONDS), unit); 292 if (nanos < 0) { // normalize nanoOfSecond to positive 293 secs -= 1; 294 nanos += 1000_000_000; 295 } 296 if (secs != instant.getEpochSecond() || (int)nanos != instant.getNano()) { 297 System.err.println(" ins=" + instant); 298 throw new RuntimeException("ft and instant are not the same time point"); 299 } 300 } 301 302 static void neq(long v1, TimeUnit u1, long v2, TimeUnit u2) { 303 FileTime t1 = FileTime.from(v1, u1); 304 FileTime t2 = FileTime.from(v2, u2); 305 if (t1.equals(t2)) 306 throw new RuntimeException("should not be equal"); 307 } 308 309 static void neq(Instant ins, long v2, TimeUnit u2) { 310 FileTime t1 = FileTime.from(ins); 311 FileTime t2 = FileTime.from(v2, u2); 312 if (t1.equals(t2)) 313 throw new RuntimeException("should not be equal"); 314 } 315 316 static void to(long v, TimeUnit unit) { 317 FileTime t = FileTime.from(v, unit); 318 for (TimeUnit u: TimeUnit.values()) { 319 long result = t.to(u); 320 long expected = u.convert(v, unit); 321 if (result != expected) { 322 throw new RuntimeException("unexpected result"); 323 } 324 } 325 } 326 327 static void ts(long v, TimeUnit unit, String expected) { 328 String result = FileTime.from(v, unit).toString(); 329 if (!result.equals(expected)) { 330 System.err.format("FileTime.from(%d, %s).toString() failed\n", v, unit); 331 System.err.format("Expected: %s\n", expected); 332 System.err.format(" Got: %s\n", result); 333 throw new RuntimeException(); 334 } 335 } 336 337 static void ts(Instant instant, String expected) { 338 String result = FileTime.from(instant).toString(); 339 if (!result.equals(expected)) { 340 System.err.format("FileTime.from(%s).toString() failed\n", instant); 341 System.err.format("Expected: %s\n", expected); 342 System.err.format(" Got: %s\n", result); 343 throw new RuntimeException(); 344 } 345 } 346 }