* bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module @ 2023-04-10 21:02 Dmitry Gutov 2023-04-11 5:51 ` Eli Zaretskii 0 siblings, 1 reply; 5+ messages in thread From: Dmitry Gutov @ 2023-04-10 21:02 UTC (permalink / raw) To: 62761 [-- Attachment #1: Type: text/plain, Size: 545 bytes --] Example: module M module N module C class D def C.foo _ end end end end end (ruby-add-log-current-method) currently returns "M::C.foo" While it should return "M::N::C.foo". Patch attached. This discovery stems from Mattias Engdegård's report (in private) about an ignored return value from `nreverse`. Is this good for emacs-29? It seems pretty safe (with decent test coverage, including the new test), but probably low urgency. Not a regression either. [-- Attachment #2: ruby-add-log-reference-containing.diff --] [-- Type: text/x-patch, Size: 1837 bytes --] diff --git a/lisp/progmodes/ruby-mode.el b/lisp/progmodes/ruby-mode.el index beccb8182a7..1199af64821 100644 --- a/lisp/progmodes/ruby-mode.el +++ b/lisp/progmodes/ruby-mode.el @@ -1911,7 +1911,7 @@ ruby-add-log-current-method (while ml (if (string-equal (car ml) (car mn)) (setq mlist (nreverse (cdr ml)) ml nil)) - (or (setq ml (cdr ml)) (nreverse mlist)))) + (setq ml (cdr ml)))) (if mlist (setcdr (last mlist) (butlast mn)) (setq mlist (butlast mn)))) diff --git a/test/lisp/progmodes/ruby-mode-tests.el b/test/lisp/progmodes/ruby-mode-tests.el index 8a75c83d2c3..117385ea3e8 100644 --- a/test/lisp/progmodes/ruby-mode-tests.el +++ b/test/lisp/progmodes/ruby-mode-tests.el @@ -567,6 +567,22 @@ ruby-add-log-current-method-namespace-shorthand (search-backward "_") (should (string= (ruby-add-log-current-method) "C::D#foo")))) +(ert-deftest ruby-add-log-current-method-singleton-referencing-outer () + (ruby-with-temp-buffer (ruby-test-string + "module M + | module N + | module C + | class D + | def C.foo + | _ + | end + | end + | end + | end + |end") + (search-backward "_") + (should (string= (ruby-add-log-current-method) "M::N::C.foo")))) + (ert-deftest ruby-add-log-current-method-after-inner-class () (ruby-with-temp-buffer (ruby-test-string "module M ^ permalink raw reply related [flat|nested] 5+ messages in thread
* bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module 2023-04-10 21:02 bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module Dmitry Gutov @ 2023-04-11 5:51 ` Eli Zaretskii 2023-04-12 0:17 ` Dmitry Gutov 0 siblings, 1 reply; 5+ messages in thread From: Eli Zaretskii @ 2023-04-11 5:51 UTC (permalink / raw) To: Dmitry Gutov; +Cc: 62761 > Date: Tue, 11 Apr 2023 00:02:03 +0300 > From: Dmitry Gutov <dmitry@gutov.dev> > > module M > module N > module C > class D > def C.foo > _ > end > end > end > end > end > > (ruby-add-log-current-method) currently returns "M::C.foo" > > While it should return "M::N::C.foo". Patch attached. > > This discovery stems from Mattias Engdegård's report (in private) about > an ignored return value from `nreverse`. > > Is this good for emacs-29? I guess so. But I wonder: this code was there since ruby-mode.el was added to Emacs in 2008, so are you saying that (cdr ml) can never be nil at this place, or if it is, it's okay to leave ml at the nil value? IOW, the return value of nreverse has been ignored, but what about the nreverse call before that: > --- a/lisp/progmodes/ruby-mode.el > +++ b/lisp/progmodes/ruby-mode.el > @@ -1911,7 +1911,7 @@ ruby-add-log-current-method > (while ml > (if (string-equal (car ml) (car mn)) > (setq mlist (nreverse (cdr ml)) ml nil)) > - (or (setq ml (cdr ml)) (nreverse mlist)))) > + (setq ml (cdr ml)))) > (if mlist > (setcdr (last mlist) (butlast mn)) > (setq mlist (butlast mn)))) Isn't the second nreverse there to undo the first? I guess I'm asking for slightly more detailed rationale for the change you are proposing, to include the analysis of the code involved and where that code is mistaken. For example, why not say (setq mlist (nreverse mlist)) or (setq ml (nreverse mlist)) instead of just removing the 2nd nreverse call? Thanks. ^ permalink raw reply [flat|nested] 5+ messages in thread
* bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module 2023-04-11 5:51 ` Eli Zaretskii @ 2023-04-12 0:17 ` Dmitry Gutov 2023-04-12 6:17 ` Eli Zaretskii 0 siblings, 1 reply; 5+ messages in thread From: Dmitry Gutov @ 2023-04-12 0:17 UTC (permalink / raw) To: Eli Zaretskii; +Cc: 62761 On 11/04/2023 08:51, Eli Zaretskii wrote: >> Date: Tue, 11 Apr 2023 00:02:03 +0300 >> From: Dmitry Gutov <dmitry@gutov.dev> >> >> module M >> module N >> module C >> class D >> def C.foo >> _ >> end >> end >> end >> end >> end >> >> (ruby-add-log-current-method) currently returns "M::C.foo" >> >> While it should return "M::N::C.foo". Patch attached. >> >> This discovery stems from Mattias Engdegård's report (in private) about >> an ignored return value from `nreverse`. >> >> Is this good for emacs-29? > > I guess so. But I wonder: this code was there since ruby-mode.el was > added to Emacs in 2008, Right. But the condition (if (string-equal (car ml) (car mn)) succeeds in a very rare case to begin with: when the definition uses a baroque, redundant kind of syntax (repeating the name of the constant instead of just using "def self.xxx" inside the appropriate module). And 'nreverse' is only harmful when there are >1 element in the list, meaning the bug also needs there to be at least 2 levels of nesting above said constant (both M and N, in the above example). And ruby-add-log-current-method isn't used often these days anyway, I guess (except from a little package of mine called robe). All this is to say, it's not surprising that this bug was never reported in the recent years. > so are you saying that (cdr ml) can never be > nil at this place, or if it is, it's okay to leave ml at the nil > value? It definitely can be nil (that's when the loop terminates -- as soon as ml reaches that value). > IOW, the return value of nreverse has been ignored, but what about the > nreverse call before that: > >> --- a/lisp/progmodes/ruby-mode.el >> +++ b/lisp/progmodes/ruby-mode.el >> @@ -1911,7 +1911,7 @@ ruby-add-log-current-method >> (while ml >> (if (string-equal (car ml) (car mn)) >> (setq mlist (nreverse (cdr ml)) ml nil)) >> - (or (setq ml (cdr ml)) (nreverse mlist)))) >> + (setq ml (cdr ml)))) >> (if mlist >> (setcdr (last mlist) (butlast mn)) >> (setq mlist (butlast mn)))) > > Isn't the second nreverse there to undo the first? > > I guess I'm asking for slightly more detailed rationale for the change > you are proposing, to include the analysis of the code involved and > where that code is mistaken. For example, why not say > > (setq mlist (nreverse mlist)) > or > (setq ml (nreverse mlist)) > > instead of just removing the 2nd nreverse call? There are, in total, 3 'nreverse' calls inside the surrounding 'let'. In the normal case (when the constant name is "resolved", that is, the search has a hit), the second 'nreverse', visible in the patch context above, is executed. The first one is not visible, and it's unconditional. Your point about "counteracting" is not without merit: one case I didn't test is when the scope matching the name is not found (meaning the constant on which the method is defined fails to "resolve"). Then the middle 'nreverse' is never called, and the list structure stays reversed. But that's basically undecidable code, and even one could write something like this using runtime constant redefinition, our parser cannot decide the full name of the constant the method is being defined at. Here's a slight modification of the patch which leaves just one (optional) 'nreverse' call, which acts on a fully temporary structure, and thus is unable to alter the values held inside mlist unless really intended. It keeps the behavior the same in the previous cases and slightly improves the one above (the "undecidable" one): diff --git a/lisp/progmodes/ruby-mode.el b/lisp/progmodes/ruby-mode.el index beccb8182a7..b252826680c 100644 --- a/lisp/progmodes/ruby-mode.el +++ b/lisp/progmodes/ruby-mode.el @@ -1905,13 +1905,13 @@ ruby-add-log-current-method (progn (unless (string-equal "self" (car mn)) ; def self.foo ;; def C.foo - (let ((ml (nreverse mlist))) + (let ((ml (reverse mlist))) ;; If the method name references one of the ;; containing modules, drop the more nested ones. (while ml (if (string-equal (car ml) (car mn)) (setq mlist (nreverse (cdr ml)) ml nil)) - (or (setq ml (cdr ml)) (nreverse mlist)))) + (setq ml (cdr ml)))) (if mlist (setcdr (last mlist) (butlast mn)) (setq mlist (butlast mn)))) ^ permalink raw reply related [flat|nested] 5+ messages in thread
* bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module 2023-04-12 0:17 ` Dmitry Gutov @ 2023-04-12 6:17 ` Eli Zaretskii 2023-04-12 21:47 ` Dmitry Gutov 0 siblings, 1 reply; 5+ messages in thread From: Eli Zaretskii @ 2023-04-12 6:17 UTC (permalink / raw) To: Dmitry Gutov; +Cc: 62761 > Date: Wed, 12 Apr 2023 03:17:55 +0300 > Cc: 62761@debbugs.gnu.org > From: Dmitry Gutov <dmitry@gutov.dev> > > Here's a slight modification of the patch which leaves just one > (optional) 'nreverse' call, which acts on a fully temporary structure, > and thus is unable to alter the values held inside mlist unless really > intended. It keeps the behavior the same in the previous cases and > slightly improves the one above (the "undecidable" one): LGTM, thanks. ^ permalink raw reply [flat|nested] 5+ messages in thread
* bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module 2023-04-12 6:17 ` Eli Zaretskii @ 2023-04-12 21:47 ` Dmitry Gutov 0 siblings, 0 replies; 5+ messages in thread From: Dmitry Gutov @ 2023-04-12 21:47 UTC (permalink / raw) To: Eli Zaretskii; +Cc: 62761-done Version: 29.1 On 12/04/2023 09:17, Eli Zaretskii wrote: >> Date: Wed, 12 Apr 2023 03:17:55 +0300 >> Cc:62761@debbugs.gnu.org >> From: Dmitry Gutov<dmitry@gutov.dev> >> >> Here's a slight modification of the patch which leaves just one >> (optional) 'nreverse' call, which acts on a fully temporary structure, >> and thus is unable to alter the values held inside mlist unless really >> intended. It keeps the behavior the same in the previous cases and >> slightly improves the one above (the "undecidable" one): > LGTM, thanks. Installed, marking as done. ^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2023-04-12 21:47 UTC | newest] Thread overview: 5+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2023-04-10 21:02 bug#62761: ruby-add-log-current-method drops some segments when singleton definition references outer module Dmitry Gutov 2023-04-11 5:51 ` Eli Zaretskii 2023-04-12 0:17 ` Dmitry Gutov 2023-04-12 6:17 ` Eli Zaretskii 2023-04-12 21:47 ` Dmitry Gutov
Code repositories for project(s) associated with this external index https://git.savannah.gnu.org/cgit/emacs.git https://git.savannah.gnu.org/cgit/emacs/org-mode.git This is an external index of several public inboxes, see mirroring instructions on how to clone and mirror all data and code used by this external index.