Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

RangeError: Invalid time value, on RN Android #17

Open
cyanzule opened this issue Sep 27, 2024 · 5 comments
Open

RangeError: Invalid time value, on RN Android #17

cyanzule opened this issue Sep 27, 2024 · 5 comments

Comments

@cyanzule
Copy link

cyanzule commented Sep 27, 2024

Hi! 👋

Firstly, thanks for your work on this project! 🙂

Today I used patch-package to patch @date-fns/[email protected] for the project I'm working on.

Environment

Base React Native project, as in, without Expo CLI, Android builds (compile SDK 34, tested on Android 14, both emulator and physical device), with dependencies:

    "@date-fns/tz": "^1.1.2",
    "date-fns": "^4.1.0",
    "intl-pluralrules": "^2.0.1",
    "react-native": "0.74.5",

Not sure on the impact of that polyfill, but adding it for reference and how a polyfill might be needed to begin with. Timezone doesn't seem to matter, but I've added mine in Replication.

Replication

All dates parsed with a time zone print Date { NaN } when logged, with the following code snippet being enough for a thrown exception:

new TZDateMini(new Date(), 'Atlantic/Madeira')

Further Investigation

So it seems this is the thereabout same issue as in #8, only the issue only focused on iOS builds. When running the suggested troubleshooting code there, it outputs the following:

new Intl.DateTimeFormat("en-GB", { timeZone: "Asia/Singapore", hour: "numeric", timeZoneName: "longOffset" }).format(new Date())

// GMT+08:00 (hour: 17)

which seems to break calcOffset, returning NaN. You can further verify this textual discrepancy in the Expo demo supplied on the original issue referenced above, when swapping to the Android build.

image

Further investigation lead to me finding out that, by default, Android only correctly formats dates for the en-US locale, unless you modify its compiling parameters, adding 6MB for the trouble. Hermes, the JS engine React Native runs on, simply outputs whatever the underlying operating system returns for Intlcalls, I think, but I haven't personally tested if using org.webkit:android-jsc-intl:+ does resolve the issue.

I don't think this patch below is the solution we should publish, changing the default locale and all, but it works as intended and doesn't break iOS builds.

Package patch

Here is the diff that solved my problem:

diff --git a/node_modules/@date-fns/tz/tzOffset/index.cjs b/node_modules/@date-fns/tz/tzOffset/index.cjs
index c6dae93..d78129d 100644
--- a/node_modules/@date-fns/tz/tzOffset/index.cjs
+++ b/node_modules/@date-fns/tz/tzOffset/index.cjs
@@ -19,7 +19,7 @@ const offsetCache = {};
  */
 function tzOffset(timeZone, date) {
   try {
-    const format = offsetFormatCache[timeZone] ||= new Intl.DateTimeFormat("en-GB", {
+    const format = offsetFormatCache[timeZone] ||= new Intl.DateTimeFormat("en-US", {
       timeZone,
       hour: "numeric",
       timeZoneName: "longOffset"
diff --git a/node_modules/@date-fns/tz/tzOffset/index.js b/node_modules/@date-fns/tz/tzOffset/index.js
index ee975c9..6bf9624 100644
--- a/node_modules/@date-fns/tz/tzOffset/index.js
+++ b/node_modules/@date-fns/tz/tzOffset/index.js
@@ -16,7 +16,7 @@ const offsetCache = {};
  */
 export function tzOffset(timeZone, date) {
   try {
-    const format = offsetFormatCache[timeZone] ||= new Intl.DateTimeFormat("en-GB", {
+    const format = offsetFormatCache[timeZone] ||= new Intl.DateTimeFormat("en-US", {
       timeZone,
       hour: "numeric",
       timeZoneName: "longOffset"

This issue body was partially generated by patch-package.

@hc-12
Copy link

hc-12 commented Oct 9, 2024

I also face the same issue in RN, where Android would crash but not iOS.

The patch has been tested and proven to work

@abo-rythm
Copy link

Hello,

Is it planned an official fix?

@goldmont
Copy link

Hi,

Same error here.

@callmehallo
Copy link

same error here

@henrikontio
Copy link

I also encountered this error on react-native app on Android (tested on android 13 and 15).

I first patched this by fixing the broken casting of minutes-string, which is something like 00 (hour that produces NaN when casting like this: +('00 (hour')

diff --git a/tzOffset/index.cjs b/tzOffset/index.cjs
index c6dae93843bb2231f64706f131f1803b58eb1cc2..b5222a0c3e0a931bb49cc3d8a3634fbcc8a118db 100644
--- a/tzOffset/index.cjs
+++ b/tzOffset/index.cjs
@@ -39,6 +39,6 @@ function tzOffset(timeZone, date) {
 const offsetRe = /([+-]\d\d):?(\d\d)?/;
 function calcOffset(cacheStr, values) {
   const hours = +values[0];
-  const minutes = +(values[1] || 0);
+  const minutes = +(values[1].slice(0, 2) || 0);
   return offsetCache[cacheStr] = hours > 0 ? hours * 60 + minutes : hours * 60 - minutes;
 }
diff --git a/tzOffset/index.js b/tzOffset/index.js
index ee975c99bcaeb34a003d77991f67598da0980076..d7a40b4e71ed7553a5a44afc8e4ad21b048e6ce3 100644
--- a/tzOffset/index.js
+++ b/tzOffset/index.js
@@ -36,6 +36,6 @@ export function tzOffset(timeZone, date) {
 const offsetRe = /([+-]\d\d):?(\d\d)?/;
 function calcOffset(cacheStr, values) {
   const hours = +values[0];
-  const minutes = +(values[1] || 0);
+  const minutes = +(values[1].slice(0, 2) || 0);
   return offsetCache[cacheStr] = hours > 0 ? hours * 60 + minutes : hours * 60 - minutes;
 }

But I think I'll change to the patch introduced here which changes the locale used as it is neater. Thanks @cyanzule

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

6 participants