[metrics-bugs] #28859 [Metrics/ExoneraTor]: Use Java 8 date-time functionality in ExoneraTor

Tor Bug Tracker & Wiki blackhole at torproject.org
Sun Dec 16 10:39:08 UTC 2018


#28859: Use Java 8 date-time functionality in ExoneraTor
------------------------------------+----------------------
     Reporter:  karsten             |      Owner:  karsten
         Type:  enhancement         |     Status:  assigned
     Priority:  Medium              |  Milestone:
    Component:  Metrics/ExoneraTor  |    Version:
     Severity:  Normal              |   Keywords:
Actual Points:                      |  Parent ID:  #23752
       Points:                      |   Reviewer:
      Sponsor:                      |
------------------------------------+----------------------
 We had plans to use Java 8 date-time functionality in all our code bases
 for quite a while now. I finally picked this up by reading a few
 tutorials, mostly to figure out which of the three types we're going to
 use: `Instant`, `LocalDateTime`, or `ZonedDateTime`. I picked ExoneraTor
 for this discussion/review, because nothing else depends on it and we're
 currently not working on this code base in other tickets.

 So, I think I have a plan now, and I'm presenting it here using snippets
 from my ExoneraTor patch:

 {{{
 @@ -131,10 +132,11 @@ public class ExoneraTorDatabaseImporter {
      try {
        if (lockFile.exists()) {
          BufferedReader br = new BufferedReader(new FileReader(lockFile));
 -        long runStarted = Long.parseLong(br.readLine());
 +        Instant runStarted = Instant.ofEpochMilli(Long.parseLong(
 +            br.readLine()));
          br.close();
 -        if (System.currentTimeMillis() - runStarted
 -            < 6L * 60L * 60L * 1000L) {
 +        if (runStarted.plus(Duration.ofHours(6L))
 +            .compareTo(Instant.now()) >= 0) {
            logger.warn("File 'exonerator-lock' is less than 6 "
                + "hours old.  Exiting.");
            System.exit(1);
 }}}

 This code is used to write the current system time to a local lock file to
 determine when we need to obey that or can safely delete it. Let's ignore
 the design and just look at the date-time part for now.

 So, I think that we can replace code where we used
 `System.currentTimeMillis()` in the past with `Instant`. The concept is
 close enough to the "machine time" concept rather than "human time" that
 `Instant` will work for us. We care about hours here, not things like
 calendar months.

 {{{
 @@ -223,7 +225,8 @@ public class ExoneraTorDatabaseImporter {

    /* Parse a consensus. */
    private static void parseConsensus(RelayNetworkStatusConsensus
 consensus) {
 -    long validAfterMillis = consensus.getValidAfterMillis();
 +    LocalDateTime validAfter =
 LocalDateTime.ofInstant(Instant.ofEpochMilli(
 +        consensus.getValidAfterMillis()), ZoneOffset.UTC);
      for (NetworkStatusEntry entry :
 consensus.getStatusEntries().values()) {
        if (entry.getFlags().contains("Running")) {
          String fingerprintBase64 = null;
 }}}

 Things are a bit different here. The valid-after time in a consensus is
 not just a point on the timeline. It's an actual date and time that has a
 meaning for humans. That somewhat rules out the `Instant`.

 Now, why use `LocalDateTime` here und not `ZonedDateTime`, as suggested
 before? Well, I could imagine doing either, as long as we're consistent
 across our codebases.

 However, one reason that made me pick `LocalDateTime` in the end was the
 comparison with PostgreSQL's `TIMESTAMP [WITHOUT TIMEZONE]` vs. `TIMESTAMP
 WITH TIMEZONE` types. We're importing lots of timestamps into PostgreSQL,
 but we're always using `TIMESTAMP [WITHOUT TIMEZONE]`. Because we know
 that timestamps in Tor are always UTC and we simply don't need timezone
 information. The [https://jdbc.postgresql.org/documentation/head/8-date-
 time.html corresponding types] for `TIMESTAMP [WITHOUT TIMEZONE]` and
 `TIMESTAMP WITH TIMEZONE` are `LocalDateTime` and `OffsetDateTime`, by the
 way.

 The only use case I see where `LocalDateTime` could be problematic is when
 somebody wants to convert our timestamps to their local timezone. But
 that's not a common use case for us, and we're primarily building our
 tools for ourselves and the services we provide.

 {{{
 @@ -248,26 +251,21 @@ public class ExoneraTorDatabaseImporter {
 [...]
      try {
        for (String orAddress : orAddresses) {
          insertStatusentryStatement.clearParameters();
 -        insertStatusentryStatement.setTimestamp(1,
 -            new Timestamp(validAfterMillis), calendarUTC);
 +        insertStatusentryStatement.setObject(1, validAfter);
          insertStatusentryStatement.setString(2, fingerprintBase64);
          if (!orAddress.contains(":")) {
            insertStatusentryStatement.setString(3, orAddress);
 }}}

 This is what it looks like when we're passing a `LocalDateTime` to the
 database.

 {{{
 @@ -47,7 +50,9 @@ public class QueryServlet extends HttpServlet {

    private DataSource ds;

 [...]
 +  private static final DateTimeFormatter validAfterTimeFormatter
 +      = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")
 +      .withZone(ZoneOffset.UTC);

    @Override
    public void init() {
 }}}

 And this is how we're parsing/formatting a Tor date-time. It's not a
 standard format, but we can easily define it once per class or package or
 codebase. Yay, thread safety!

 Okay, these are the highlights. I'll post the branch once I have a ticket
 number.

--
Ticket URL: <https://trac.torproject.org/projects/tor/ticket/28859>
Tor Bug Tracker & Wiki <https://trac.torproject.org/>
The Tor Project: anonymity online


More information about the metrics-bugs mailing list