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

repl: fix .load infinite loop caused by shared use of lineEnding RegExp #46742

Merged
merged 12 commits into from
Mar 1, 2023

Conversation

Theo-Steiner
Copy link
Contributor

@Theo-Steiner Theo-Steiner commented Feb 20, 2023

repl: fix .load infinite loop caused by shared use of lineEnding RegExp

Since the lineEnding is declared on the module scope of lib/internal/readline/interface.js,
recursive invocations of its [kTtyWrite] method share one instance of this Regular Expression.
Since the state of a RegExp is managed by instance, alternately calling RegExpPrototypeExec
with the same RegExp on different strings can lead to the state changing unexpectedly.
This was the root cause of this infinite loop bug when calling .load on javascript files of certain shapes.

Fixes: #46731

I attempted to write a test for this, mimicking test/parallel/test-repl-load-multiline.js.
However, I failed reproducing the infinite loop bug from the programatic REPL, as the problematic indentation preservation is not executed for some reason I have yet to understand. I thought setting terminal: true might produce the intended result, but I had no luck with it for now.
If anyone has ideas for how to test for this, please let me know!

Edit 4: I found a solution to the core issue and updated this PR accordingly
Edit 5: also found a way to test this

@nodejs-github-bot nodejs-github-bot added needs-ci PRs that need a full CI run. repl Issues and PRs related to the REPL subsystem. labels Feb 20, 2023
@Theo-Steiner Theo-Steiner force-pushed the fix/repl-infinite-loop branch from f919b51 to 831d193 Compare February 20, 2023 12:06
@Theo-Steiner
Copy link
Contributor Author

To be honest, I am not quite sure, why calling .load puts the REPL into editorMode in the first place.
If that could be safely removed, that might be an even better option.

@tniessen tniessen requested a review from aduh95 February 21, 2023 03:48
@tniessen
Copy link
Member

cc @nodejs/repl

@Theo-Steiner
Copy link
Contributor Author

Theo-Steiner commented Feb 21, 2023

#46731 (comment)
I found that the actual problem that triggers this is with reusing the lineEnding RegExp.
Do you think that using String.prototype.matchAll with this RegExp might be a better solution than disabling the editor mode?

EDIT: Turns out avoiding using methods that depend on the RegExp prototype was what introduced this bug in the first place. #36593 -> #45614

@Theo-Steiner Theo-Steiner force-pushed the fix/repl-infinite-loop branch from 7a8ce5f to 578c040 Compare February 22, 2023 10:14
@Theo-Steiner Theo-Steiner changed the title repl: fix infinite loop caused by indentation preservation repl: fix .load infinite loop caused by shared use of lineEnding RegExp Feb 22, 2023
@cola119 cola119 added the request-ci Add this label to start a Jenkins CI on a PR. label Feb 22, 2023
@github-actions github-actions bot removed the request-ci Add this label to start a Jenkins CI on a PR. label Feb 22, 2023
@nodejs-github-bot
Copy link
Collaborator

@nodejs-github-bot
Copy link
Collaborator

Copy link
Contributor

@aduh95 aduh95 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi, thanks for investigating that and sending a PR. It seems that the added test already passes on main, which makes me think that it's not able to reproduce the issue. Could you have a look?

lib/repl.js Outdated Show resolved Hide resolved
@Theo-Steiner
Copy link
Contributor Author

Theo-Steiner commented Feb 23, 2023

It seems that the added test already passes on main, which makes me think that it's not able to reproduce the issue. Could you have a look?

Yes, I did not manage to reproduce the infinite loop in a test. The problem is that the infinite loop only occurs if the code block starting on this line

if (self.editorMode) {
is entered. For some reason the test does not enter this block. Probably there is some difference between the programmatic REPL and the interactive one at play here?
I've tried passing {terminal: true} to the REPL, but that did not do it either.
Do you have an idea for what could be the difference between programmatic and interactive REPL?
I won't be able to further investigate this until Monday, I hope that is alright.

@aduh95
Copy link
Contributor

aduh95 commented Feb 23, 2023

I won't be able to further investigate this until Monday, I hope that is alright.

Of course it's alright, take all the time you need! Node.js is striving from free contributions from contributors like you and me, the only expectations we set is to comply to the Code of Conduct, we don't have any expectations regarding the quantity or the rapidity of work :)

@aduh95
Copy link
Contributor

aduh95 commented Feb 24, 2023

I'm not sure if it's going to perform better, but the bug can be resolved with a more minimal diff:

diff --git a/lib/internal/readline/interface.js b/lib/internal/readline/interface.js
index 45524612ae..f69d6b14b0 100644
--- a/lib/internal/readline/interface.js
+++ b/lib/internal/readline/interface.js
@@ -1327,9 +1327,11 @@ class Interface extends InterfaceConstructor {
               this[kInsertString](StringPrototypeSlice(s, 0, nextMatch.index));
               let { lastIndex } = lineEnding;
               while ((nextMatch = RegExpPrototypeExec(lineEnding, s)) !== null) {
+                const previousLastIndex = lineEnding.lastIndex;
                 this[kLine]();
                 this[kInsertString](StringPrototypeSlice(s, lastIndex, nextMatch.index));
-                ({ lastIndex } = lineEnding);
+                lastIndex = previousLastIndex;
+                lineEnding.lastIndex = previousLastIndex;
               }
               if (lastIndex === s.length) this[kLine]();
             } else {

When it comes to the repro, you may have more luck with trying to reproduce the bug in node:readline/promises rather than node:repl, REPL adds quite a bit of complexity that you can avoid by hitting directly the source of the bug. Let me know if you need more help with the issue.

Since the lineEnding Regular Expression is declared on the module scope,
recursive invocations of its `[kTtyWrite]` method share one instance of
this Regular Expression.
Since the state of a RegExp is managed by instance,
alternately calling RegExpPrototypeExec with the same RegExp on
different strings can lead to the state changing unexpectedly.
This is the root cause of this infinite loop bug when calling .load on
javascript files of certain shapes.
Theo-Steiner added a commit to Theo-Steiner/node that referenced this pull request Feb 27, 2023
@Theo-Steiner Theo-Steiner force-pushed the fix/repl-infinite-loop branch from 0a4fcbf to 3cc2379 Compare February 27, 2023 08:28
@Theo-Steiner Theo-Steiner force-pushed the fix/repl-infinite-loop branch from 3cc2379 to a9010dc Compare February 27, 2023 08:30
@Theo-Steiner
Copy link
Contributor Author

@aduh95 thank you for your advice.
Using it I managed to write a test using readline's interface that shows the underlying problem!
It fails for everything after 18.12.0 and is fixed by this PR.

I intentionally changed the implementation to use a separate instance of the RegExp, since I thought a similar problem could occur in the future if we fix it by just shuffling the variable declarations around.
The side effects of RegExpPrototypeExec are very counterintuitive to me, so I thought other developers might overlook them too. That's why I refactored the code to be (1) more obvious in what it does and (2) prevent errors caused by alternately calling RegExpPrototypeExec with the same RegExp and a different string.

If keeping diffs small is desirable however, I totally understand and will make the changes as suggested by you. Just let me know how to proceed 🥳

@nodejs-github-bot
Copy link
Collaborator

@cola119 cola119 added the readline Issues and PRs related to the built-in readline module. label Feb 27, 2023
@Theo-Steiner
Copy link
Contributor Author

Theo-Steiner commented Mar 1, 2023

I added a line to reset the RegExp's lastIndex property to 0 before commencing the search.
I think now all tests should pass!

@cola119 cola119 added the request-ci Add this label to start a Jenkins CI on a PR. label Mar 1, 2023
@github-actions github-actions bot removed the request-ci Add this label to start a Jenkins CI on a PR. label Mar 1, 2023
@nodejs-github-bot
Copy link
Collaborator

@cola119 cola119 added the commit-queue Add this label to land a pull request using GitHub Actions. label Mar 1, 2023
@nodejs-github-bot nodejs-github-bot added commit-queue-failed An error occurred while landing this pull request using GitHub Actions. and removed commit-queue Add this label to land a pull request using GitHub Actions. labels Mar 1, 2023
@nodejs-github-bot
Copy link
Collaborator

Commit Queue failed
- Loading data for nodejs/node/pull/46742
✔  Done loading data for nodejs/node/pull/46742
----------------------------------- PR info ------------------------------------
Title      repl: fix .load infinite loop caused by shared use of lineEnding RegExp (#46742)
   ⚠  Could not retrieve the email or name of the PR author's from user's GitHub profile!
Branch     Theo-Steiner:fix/repl-infinite-loop -> nodejs:main
Labels     readline, repl, author ready, needs-ci
Commits    12
 - repl: fix .load infinite loop caused by shared use of lineEnding RegExp
 - test: add test that fails without the changes of #46742
 - chore: remove unrelated comment
 - chore: fix linting errors and rename test
 - remove magic number from test
 - fix: refactor to use module scope instance of RegExp safely for perfo…
 - chore: fix linting errors
 - docs: update comments
 - chore: make finishing touches on comments
 - test: fix test failing due to difference in terminal codes
 - Revert "test: fix test failing due to difference in terminal codes"
 - fix: erase lastIndex state for previous searches
Committers 2
 - Theodor Steiner 
 - GitHub 
PR-URL: /~https://github.com/nodejs/node/pull/46742
Fixes: /~https://github.com/nodejs/node/issues/46731
Reviewed-By: Kohei Ueno 
Reviewed-By: Antoine du Hamel 
------------------------------ Generated metadata ------------------------------
PR-URL: /~https://github.com/nodejs/node/pull/46742
Fixes: /~https://github.com/nodejs/node/issues/46731
Reviewed-By: Kohei Ueno 
Reviewed-By: Antoine du Hamel 
--------------------------------------------------------------------------------
   ℹ  This PR was created on Mon, 20 Feb 2023 11:54:47 GMT
   ✔  Approvals: 2
   ✔  - Kohei Ueno (@cola119): /~https://github.com/nodejs/node/pull/46742#pullrequestreview-1319782434
   ✔  - Antoine du Hamel (@aduh95) (TSC): /~https://github.com/nodejs/node/pull/46742#pullrequestreview-1319512073
   ✔  Last GitHub CI successful
   ℹ  Last Full PR CI on 2023-03-01T08:33:26Z: https://ci.nodejs.org/job/node-test-pull-request/50125/
   ℹ  Last Benchmark CI on 2023-02-27T19:56:39Z: https://ci.nodejs.org/view/Node.js%20benchmark/job/benchmark-node-micro-benchmarks/1299/
- Querying data for job/node-test-pull-request/50125/
   ✔  Last Jenkins CI successful
--------------------------------------------------------------------------------
   ✔  No git cherry-pick in progress
   ✔  No git am in progress
   ✔  No git rebase in progress
--------------------------------------------------------------------------------
- Bringing origin/main up to date...
From /~https://github.com/nodejs/node
 * branch                  main       -> FETCH_HEAD
✔  origin/main is now up-to-date
- Downloading patch for 46742
From /~https://github.com/nodejs/node
 * branch                  refs/pull/46742/merge -> FETCH_HEAD
✔  Fetched commits as da0bc6db98ce..395054215dc3
--------------------------------------------------------------------------------
[main b230fffbcf] repl: fix .load infinite loop caused by shared use of lineEnding RegExp
 Author: Theodor Steiner 
 Date: Mon Feb 20 20:47:33 2023 +0900
 3 files changed, 66 insertions(+), 10 deletions(-)
 create mode 100644 test/parallel/test-repl-load-multiline-function.js
[main 645a7b54da] test: add test that fails without the changes of #46742
 Author: Theodor Steiner 
 Date: Mon Feb 27 17:23:13 2023 +0900
 2 files changed, 31 insertions(+), 47 deletions(-)
 create mode 100644 test/parallel/test-readline-recursive-writes.js
 delete mode 100644 test/parallel/test-repl-load-multiline-function.js
[main 12cec6217c] chore: remove unrelated comment
 Author: Theodor Steiner 
 Date: Mon Feb 27 17:29:53 2023 +0900
 1 file changed, 1 deletion(-)
[main ac24d34df7] chore: fix linting errors and rename test
 Author: Theodor Steiner 
 Date: Mon Feb 27 17:33:58 2023 +0900
 1 file changed, 4 insertions(+), 5 deletions(-)
 rename test/parallel/{test-readline-recursive-writes.js => test-readline-interface-recursive-writes.js} (86%)
[main 8029f97d09] remove magic number from test
 Author: Theodor Steiner <40017636+Theo-Steiner@users.noreply.github.com>
 Date: Tue Feb 28 09:39:32 2023 +0900
 1 file changed, 8 insertions(+), 5 deletions(-)
[main 8ec5286f41] fix: refactor to use module scope instance of RegExp safely for performance considerations
 Author: Theodor Steiner 
 Date: Tue Feb 28 15:16:02 2023 +0900
 1 file changed, 3 insertions(+), 4 deletions(-)
[main 7871c71f83] chore: fix linting errors
 Author: Theodor Steiner 
 Date: Tue Feb 28 15:17:54 2023 +0900
 1 file changed, 1 insertion(+), 2 deletions(-)
[main ace780692a] docs: update comments
 Author: Theodor Steiner 
 Date: Tue Feb 28 15:52:02 2023 +0900
 1 file changed, 1 insertion(+), 5 deletions(-)
[main fdbc00369a] chore: make finishing touches on comments
 Author: Theodor Steiner <40017636+Theo-Steiner@users.noreply.github.com>
 Date: Tue Feb 28 22:55:35 2023 +0900
 1 file changed, 3 insertions(+), 3 deletions(-)
[main e8d71d9d6a] test: fix test failing due to difference in terminal codes
 Author: Theodor Steiner 
 Date: Wed Mar 1 11:49:30 2023 +0900
 1 file changed, 5 insertions(+), 2 deletions(-)
[main 7e6509411b] Revert "test: fix test failing due to difference in terminal codes"
 Author: Theodor Steiner 
 Date: Wed Mar 1 15:54:34 2023 +0900
 1 file changed, 2 insertions(+), 5 deletions(-)
[main 5c29f7d92d] fix: erase lastIndex state for previous searches
 Author: Theodor Steiner 
 Date: Wed Mar 1 15:56:00 2023 +0900
 1 file changed, 2 insertions(+)
   ✔  Patches applied
There are 12 commits in the PR. Attempting autorebase.
Rebasing (2/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
repl: fix .load infinite loop caused by shared use of lineEnding RegExp

Since the lineEnding Regular Expression is declared on the module scope,
recursive invocations of its [kTtyWrite] method share one instance of
this Regular Expression.
Since the state of a RegExp is managed by instance,
alternately calling RegExpPrototypeExec with the same RegExp on
different strings can lead to the state changing unexpectedly.
This is the root cause of this infinite loop bug when calling .load on
javascript files of certain shapes.

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 74d5e0d47b] repl: fix .load infinite loop caused by shared use of lineEnding RegExp
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Mon Feb 20 20:47:33 2023 +0900
3 files changed, 66 insertions(+), 10 deletions(-)
create mode 100644 test/parallel/test-repl-load-multiline-function.js
Rebasing (3/24)
Rebasing (4/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
test: add test that fails without the changes of #46742

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD bb55df9366] test: add test that fails without the changes of #46742
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Mon Feb 27 17:23:13 2023 +0900
2 files changed, 31 insertions(+), 47 deletions(-)
create mode 100644 test/parallel/test-readline-recursive-writes.js
delete mode 100644 test/parallel/test-repl-load-multiline-function.js
Rebasing (5/24)
Rebasing (6/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
chore: remove unrelated comment

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 34391c9429] chore: remove unrelated comment
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Mon Feb 27 17:29:53 2023 +0900
1 file changed, 1 deletion(-)
Rebasing (7/24)
Rebasing (8/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
chore: fix linting errors and rename test

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 06ecce247c] chore: fix linting errors and rename test
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Mon Feb 27 17:33:58 2023 +0900
1 file changed, 4 insertions(+), 5 deletions(-)
rename test/parallel/{test-readline-recursive-writes.js => test-readline-interface-recursive-writes.js} (86%)
Rebasing (9/24)
Rebasing (10/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
remove magic number from test

Co-authored-by: Antoine du Hamel duhamelantoine1995@gmail.com
PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 85c32ca998] remove magic number from test
Author: Theodor Steiner 40017636+Theo-Steiner@users.noreply.github.com
Date: Tue Feb 28 09:39:32 2023 +0900
1 file changed, 8 insertions(+), 5 deletions(-)
Rebasing (11/24)
Rebasing (12/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
fix: refactor to use module scope instance of RegExp safely for performance considerations

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 191f742311] fix: refactor to use module scope instance of RegExp safely for performance considerations
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Tue Feb 28 15:16:02 2023 +0900
1 file changed, 3 insertions(+), 4 deletions(-)
Rebasing (13/24)
Rebasing (14/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
chore: fix linting errors

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD abd94f85b4] chore: fix linting errors
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Tue Feb 28 15:17:54 2023 +0900
1 file changed, 1 insertion(+), 2 deletions(-)
Rebasing (15/24)
Rebasing (16/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
docs: update comments

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD f68814e734] docs: update comments
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Tue Feb 28 15:52:02 2023 +0900
1 file changed, 1 insertion(+), 5 deletions(-)
Rebasing (17/24)
Rebasing (18/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
chore: make finishing touches on comments

Co-authored-by: Antoine du Hamel duhamelantoine1995@gmail.com
PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 27b4146827] chore: make finishing touches on comments
Author: Theodor Steiner 40017636+Theo-Steiner@users.noreply.github.com
Date: Tue Feb 28 22:55:35 2023 +0900
1 file changed, 3 insertions(+), 3 deletions(-)
Rebasing (19/24)
Rebasing (20/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
test: fix test failing due to difference in terminal codes

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD d6950dbf57] test: fix test failing due to difference in terminal codes
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Wed Mar 1 11:49:30 2023 +0900
1 file changed, 5 insertions(+), 2 deletions(-)
Rebasing (21/24)
Rebasing (22/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
Revert "test: fix test failing due to difference in terminal codes"

This reverts commit 6699359c7c15b92256c1c3c9813bd50f05a5606f.

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD e80778e35c] Revert "test: fix test failing due to difference in terminal codes"
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Wed Mar 1 15:54:34 2023 +0900
1 file changed, 2 insertions(+), 5 deletions(-)
Rebasing (23/24)
Rebasing (24/24)

Executing: git node land --amend --yes
--------------------------------- New Message ----------------------------------
fix: erase lastIndex state for previous searches

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno kohei.ueno119@gmail.com
Reviewed-By: Antoine du Hamel duhamelantoine1995@gmail.com

[detached HEAD 2f274c0c73] fix: erase lastIndex state for previous searches
Author: Theodor Steiner theodor.steiner@linecorp.com
Date: Wed Mar 1 15:56:00 2023 +0900
1 file changed, 2 insertions(+)

Successfully rebased and updated refs/heads/main.

ℹ Add commit-queue-squash label to land the PR as one commit, or commit-queue-rebase to land as separate commits.

/~https://github.com/nodejs/node/actions/runs/4303874198

@cola119 cola119 added commit-queue-squash Add this label to instruct the Commit Queue to squash all the PR commits into the first one. and removed commit-queue-failed An error occurred while landing this pull request using GitHub Actions. labels Mar 1, 2023
@aduh95 aduh95 added the commit-queue Add this label to land a pull request using GitHub Actions. label Mar 1, 2023
@nodejs-github-bot nodejs-github-bot removed the commit-queue Add this label to land a pull request using GitHub Actions. label Mar 1, 2023
@nodejs-github-bot nodejs-github-bot merged commit 333aff0 into nodejs:main Mar 1, 2023
@nodejs-github-bot
Copy link
Collaborator

Landed in 333aff0

@Theo-Steiner Theo-Steiner deleted the fix/repl-infinite-loop branch March 1, 2023 14:19
@aduh95
Copy link
Contributor

aduh95 commented Mar 1, 2023

Thanks a lot for your work here, and congrats for your first merged PR for Node.js 🎉

targos pushed a commit that referenced this pull request Mar 13, 2023
Since the lineEnding Regular Expression is declared on the module scope,
recursive invocations of its `[kTtyWrite]` method share one instance of
this Regular Expression.
Since the state of a RegExp is managed by instance,
alternately calling RegExpPrototypeExec with the same RegExp on
different strings can lead to the state changing unexpectedly.
This is the root cause of this infinite loop bug when calling .load on
javascript files of certain shapes.

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno <kohei.ueno119@gmail.com>
Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
targos pushed a commit that referenced this pull request Mar 14, 2023
Since the lineEnding Regular Expression is declared on the module scope,
recursive invocations of its `[kTtyWrite]` method share one instance of
this Regular Expression.
Since the state of a RegExp is managed by instance,
alternately calling RegExpPrototypeExec with the same RegExp on
different strings can lead to the state changing unexpectedly.
This is the root cause of this infinite loop bug when calling .load on
javascript files of certain shapes.

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno <kohei.ueno119@gmail.com>
Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
@starball5
Copy link

starball5 commented Mar 18, 2023

I see that this fix is in the v19 changelog, but I don't see it in the v18 changelog. As mentioned earlier, this bug exists starting after 18.12.0.

Considering that v18 is currently in active LTS status, can we get the fix in v18 as well please?

Related Stack Overflow questions:

danielleadams pushed a commit that referenced this pull request Apr 11, 2023
Since the lineEnding Regular Expression is declared on the module scope,
recursive invocations of its `[kTtyWrite]` method share one instance of
this Regular Expression.
Since the state of a RegExp is managed by instance,
alternately calling RegExpPrototypeExec with the same RegExp on
different strings can lead to the state changing unexpectedly.
This is the root cause of this infinite loop bug when calling .load on
javascript files of certain shapes.

PR-URL: #46742
Fixes: #46731
Reviewed-By: Kohei Ueno <kohei.ueno119@gmail.com>
Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
author ready PRs that have at least one approval, no pending requests for changes, and a CI started. commit-queue-squash Add this label to instruct the Commit Queue to squash all the PR commits into the first one. needs-ci PRs that need a full CI run. readline Issues and PRs related to the built-in readline module. repl Issues and PRs related to the REPL subsystem.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

.load infinite loop in REPL
6 participants