I have heard about difficulties of using JavaScript date APIs, but it was not until recently when I eventually experienced it myself. I am going to describe one particular phenomenon that can lead to wrong date values sent from client’s browser to a server. When analysing these examples please keep in mind they were executed on machine with UTC+01:00 time zone unless I explicitly tell that an example refers to a different time zone.
Let’s try to parse a JavaScript Date object:
What draws my attention is the time value. It is 01:00 which may look strange. But it is not if we correlate this value with time zone information which is stored along with JavaScript object. The time zone information is an inherent part of the object and it comes from the browser, which obviously derives it from the operating system’s culture settings. It turns out these two pieces of information are essential when making AJAX calls, because then .toJSON()
method is called. I am making this assumption based on the example behaviour of ngResource library, but other frameworks or libraries probably do the same, because they must somehow convert JavaScript Date object to a universal text format to be able to send it via HTTP. By the way, .toJSON()
returns the same result as to .toISOString()
.
What we have got here is UTC-normalized date and time value. The time zone offset of actual time value used with time zone information allow us to normalize the date when sending it to a server. The most important thing here is that the values stored in Date object are expressed in local time zone i.e. the browser’s one. This implies some strange consequences like the one of being in UTC-xx:xx time zones. Let’s try the same example after setting time zone to UTC-01:00.
The problem here is that we have actually ended up with parsed values which are different from their original textual representation i.e. March 1st versus February 28th. But it is still OK providing that our date processing logic relies on normalized values:
However, it can be misleading when we try to get individual date components. Here we try to get day component.
But in general the object still can serve the purpose if we rely on normalized values and call appropriate methods.
The problem is that not all Date constructors behave in the same way. Let’s try this one:
Now the parsed date although it still contains time zone information derived from the operating system, but it does not contain time value modified by the corresponding offset. In the first example we have 1am time which corresponds to GMT+01:00, here we have just 00:00 time and, of course, we still do have GMT+01:00 time zone information. This time zone information without correctly shifted time value is actually catastrophic. Look what happens when .toJSON()
is called:
The result is wrong date sent to a server. This is not the end of the observation. The same phenomenon can also happen in the other way round, i.e. when we are transferring date values from a server to a client. Now let’s assume the server sent the following date and we are parsing it. Please keep in mind that the actual process of parsing may happen implicitly in some framework’s code, for instance when specifying data source for Kendo grid. So the one who parses it for us can be a framework’s code.
As we see, this constructor results in shifted time value just like the Date(“2015-03-01”) one. But when considering displaying these retrieved values we inevitably have to answer the question, whether we aim at showing local time or the server time. We have to remember that in case when the client’s browser is in GMT-xx:xx time zone and we try to show the parsed value (like in c.getDate()
example), not the normalized one, this may result in wrong date displayed in front of the user. I say `may`, because this can really be a desired behaviour depending on the requirements. For example, in Angularwe can enforce displaying normalized value by providing optional time zone parameter to $filter('date')
.
Here we do not worry about internal, actual component values of object c whose prototype is Date. It internally may store February 28th but it does not matter. $filter
is told to output values for UTC time. It is also worth mentioning that the Date constructor also assumes that its argument in specified in UTC time. So we populate the Date object with UTC value and return also UTC value not worrying about internal representation which is local. This approach results in the output date and time being equal to the intended input one.
As a conclusion I should write some recommendations on how to use Date object and what to avoid. But honestly I cannot say I gained enough knowledge in this area to make any kind of guidelines. I can only afford making some general statements. Just pay attention to what your library components like, for instance, a date picker control operates on. Is their output a Date object or string representation of the date? What do you do with this output? Is their input a Date object or a string representation and they do the parsing on their own? Just examine carefully and do not blindly trust your framework. I personally do not accept situation when something works and I do now know why it does. I tend to dig into details and try to find the nitty gritty. I have always believed deep research (even one which takes much time) and understanding of underlying technology is worthwhile and I often recall the post by Scott Hanselman who also appears to follow this principle.
> But honestly I cannot say I gained enough knowledge in this area to make any kind of guidelines. I can only afford making some general statements.
I think one of the things your post illustrates is that it’s usually better to work with normalized data. If you always think in normalized data, in this case UTC time, it’s easier to reason about the requirements. Your examples were really easy to follow because they presented the Dates as a UTC time *plus* a time zone.
A generalization I would give programmers working with Dates, would be to first think about the timestamp, then think about the presentation requirement as a deviation from that. For example, if you need to show localized time but you’re using an API that doesn’t give timezone information (using the server time implicitly), you should subtract the server’s timezone before doing anything else.
Thinking in timezones is difficult at the best of times. The fact that the Date object coerces in mysterious ways makes the problem liable to come up when you shouldn’t need to.
Alex is right though: the solution is to persist dates as either unix timestamps or ISO UTC strings and never mutate them; keep these records immutable (avoiding the Date object!) and only ever transform them to any other representation when it comes to user interface.
But the Date object API is itself a huge part of the problem and IMO should be avoided at all costs. In the previous paragraph I urge for immutability – and despite moment.js being object-oriented & mutative in nature it is IMO essential for sanity when working with date transformation, interpolation and comparison. If you combine the general practice of storing dates in the unambiguous Unix timestamp or ISO string (whichever is most appropriate to your business logic), never mutating instances, and using a decent wrapping API like Moment’s, most of these problems can be mitigated.
http://momentjs.com/