node-postgres icon indicating copy to clipboard operation
node-postgres copied to clipboard

Mismatch in Date precision causes odd behavior

Open Carbonyte opened this issue 8 years ago • 10 comments

The PostgreSQL TIMESTAMP type has a maximum precision of 1 microsecond, but the JavaScript Date type only has a maximum precision of 1 millisecond. This causes information to be silently discarded, leading to bizarre behavior:

client.query("SELECT '2017-01-01 00:00:00.00001'::timestamp as d", (err, res) => {
let date = res.rows[0].d;

client.query("SELECT '2017-01-01 00:00:00.00001'::timestamp = $1 as eq,\
'2017-01-01 00:00:00.00001'::timestamp > $1 as gt", [date], (err, res) => {
console.log(res.rows[0].eq); //false
console.log(res.rows[0].gt); //true
});
});

I'm not sure this a bug that could be fixed without using a non-standard Date object, but even if it isn't fixed it should be at least noted in the documentation.

Carbonyte avatar Jan 19 '17 23:01 Carbonyte

JavaScript doesn't support microseconds, so wouldn't this be self-implying for a JavaScript library?

No point documenting the obvious, imo.

And there is nothing bizarre in your example, because:

let date = res.rows[0].d;
//=> 2017-01-01T00:00:00.000Z

which makes the rest logically apparent:

2017-01-01 00:00:00.00001 = 2017-01-01T00:00:00.000Z => false
2017-01-01 00:00:00.00001 > 2017-01-01T00:00:00.000Z => true

vitaly-t avatar Jan 21 '17 12:01 vitaly-t

The behavior isn't bizarre once you know the cause, but it isn't a particularly easy thing to debug if you don't. How the various Postgres datatypes map to JS types is itself not well documented, which is part of the problem.

Carbonyte avatar Jan 22 '17 22:01 Carbonyte

For what it's worth, I probably wasted about 60 hours last week, and cause a lot of pain to users, because there wasn't a sentence about this in the main README.md page. I did systematically read through the fine README before starting to use the driver, and if there was a warning about timestamps silently rounding by "textually deleting the last 3 digits" instead of rounding, it would have helped. (And other than this, I really like this driver!)

I think the behavior of the driver is perhaps more surprising than your examples above suggest. What happens with the driver is completely different than what happens using postgresql's ::timestamp(3) explicit cast, which rounds to the nearest timestamp, rather than just textually removing the last 3 digits.

williamstein avatar Feb 07 '17 21:02 williamstein

A workaround is to override the type parser. That doesn't prevent confusion or fix the documentation, but at least you could handle microseconds with custom code.

var pg = require('pg');

function pgToString(value) {
    return value.toString();
}

pg.types.setTypeParser(1082, pgToString); // date
pg.types.setTypeParser(1083, pgToString); // time
pg.types.setTypeParser(1114, pgToString); // timestamp
pg.types.setTypeParser(1184, pgToString); // timestamptz
pg.types.setTypeParser(1266, pgToString); // timetz

rpedela avatar Feb 21 '17 21:02 rpedela

@rpedela THANKS!

williamstein avatar Feb 21 '17 23:02 williamstein

Added this to the docs milestone - I'll include a note around this when I discuss the type parsing there.

brianc avatar May 24 '17 15:05 brianc

Other fun information: the binary parser exposes microseconds, but the text parser doesn’t, and the microseconds are ignored when serializing back. (I don’t think the binary parser should expose microseconds in this way, because the methods behave differently compared to the rest of the date API.)

Time is overall just really uncomfortable, but it might be okay to start producing frozen or carefully subclassed dates with a microsecond field for timestamptzs that can be round-tripped in a future major version? (For all other datetime-related types, parsing to Date as a default will be deprecated is the current idea.)

charmander avatar Dec 06 '19 01:12 charmander

(And this did get added to the docs, at the bottom of https://node-postgres.com/features/types/.)

charmander avatar Dec 06 '19 01:12 charmander

The timestamp rounding has been fixed in https://github.com/bendrucker/postgres-date/releases/tag/v1.0.5. It will allow to consistently query for someDate::timestamp(3) for example.

salper avatar Apr 14 '20 10:04 salper

JFYI for future readers: ts::timestamptz(3) does rounding, and date_trunc('milliseconds', ts) does truncation (the truncation works the same way as postgres-date module used by node-postgres).

# select
  '2023-08-12 19:35:53.973605-07'::timestamptz, 
  '2023-08-12 19:35:53.973605-07'::timestamptz::timestamptz(3),
  date_trunc('milliseconds', '2023-08-12 19:35:53.973605-07'::timestamptz) \gx
-[ RECORD 1 ]------------------------------
timestamptz | 2023-08-12 19:35:53.973605-07
timestamptz | 2023-08-12 19:35:53.974-07
date_trunc  | 2023-08-12 19:35:53.973-07

dko-slapdash avatar Aug 13 '23 02:08 dko-slapdash