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 }