-
-
Notifications
You must be signed in to change notification settings - Fork 646
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
Expensive calls tocider-current-repl
in latency sensitive contexts
#3344
Comments
I'm not sure this statement is true.
Just like in today's other issue, I'm not trying to negate the problems - let's just simply make sure that the diagnostics are precise. We cannot come up with a correct solution otherwise. I also think it would be a good idea to create one issue per problem (I'd be OK with an issue in this repo for the agressive indentation, even if it's not "our" problem. Spoiler: maybe we should create a specially-tailored, lightweight indentation mode just for that context). Cheers - V |
Thanks for the feedback! These findings were prompted by my direct experience of stuttering and latency while typing, so the problem definitely exists to some extent. I can't prove a direct link to eg. Eldoc but the profiler shows a large percentage of CPU and memory being taken up by its idle timer, specifically by the call to My
Code is usually not typed in a single unbroken stream, so once you pause for half a second between keystrokes, Eldoc gets triggered, and then the next character you typed does not appear until it is done with its request. This happened frequently enough to feel stuttery and jarring, to the point that I ended up spending hours to track down these performance issues and file bug reports 😅
There are many ways to configure Company - I believe you can also have it trigger on idle delay so an overlay / childframe pops up when you pause typing. I recently tried out Corfu which has this option in
Agreed, which is why I did not mention it originally - essentially it calls
Yes, feel free to ignore this one - I didn't have performance issues with formatting, but added it for 'completeness' after grepping through the codebase to see where else (cider-current-repl) was being used. |
Ok. Anyway, had you considered increasing This doesn't close the door to optimizing this code path, however I'd find it wise to prioritize things very carefully.
I find this a delicate point, because I don't want to tell you "don't use that", however not all features are necessarily good ideas all the time. As I would imagine it, CIDER, while of course strives to be performant, is generally built on the principle of runtime-powered insights. This can easily involve hitting the JVM, clojure code, the filesystem, etc. That performs more than acceptably for the constraints it was built for, however Again, that's not to close the door to possible optimizations. Let's just perceive edge cases as such when, IMO, they are. That allows optimal task prioritization.
Nice. I guess that this is a forthcoming PR? Looking forward! |
I'll admit that a large part of my motivation here is a sort of Moral Indignation at poorly optimized code... ie. how many battery lifespans and CPU cycles are being spent churning away in the background at something so seemingly trivial as the current REPL 🫠. Like a Bitcoin miner, generating waste heat while doing no useful work. Tonsky has a great rant on the topic: https://tonsky.me/blog/disenchantment/ Not to get too philosophical, but Emacs is such a rare blessing in the modern software landscape, where an end-user like me can experience faults with a tool and be empowered enough to introspect the root causes and modify them directly. So suggestions to accommodate the lagginess by avoiding certain completion styles or having Cider tweak the buffer-local Eldoc configs feel somewhat beside the point, when there appear to be low-hanging performance gains to be had... the tradeoff between dumb static linting vs. runtime-powered smarts shouldn't have to be such a hard dichotomy. And speaking of delicate points, there's maybe a element of privilege involved when it comes to prioritizing these performance concerns. They disproportionately affect users on older hardware, and the fact is I'm already encountering them on a relatively new M1 Macbook Pro, even if they only crop up in somewhat extreme 'edge cases'. Putting some concrete numbers to it, I had 4-5 separate REPLs open and maybe 60+ clj files in the background (the majority from nosing around in clj-kondo's corpus directory). After pruning most of these with the help of ibuffer, things quickly got back to a usable state. Just trying to be aware that our 'barely noticeable edge cases' might be someone else's 'annoying daily occurrence', any chance to save resources and widen accessibility in general shouldn't be lightly overlooked. |
I hadn't replied to this one back then, in good part because I didn't feel like arguing on the internet... My tldr would be that PRs are most welcome - let's just try together to summarize things as they really are. OP appears to describe a series of acute problems, that later can be seen as I'll gladly review optimizations. I think it's also reasonable to weigh the value of PRs against any possible complexity or lack of test coverage they might, hypothetically, produce. I don't think that would be the case with you as we've seen nothing but impressive contributions! |
btw, Would you be OK if we close this issue and create one named That would seem focused and actionable. It would be probably one of the finishing touches (while also a design concern) in the sesman refactoring. |
Yeah absolutely - looking back I'm not sure why I was making such a big deal of this, some work related frustrations seeping through at the time, probably 😶 Sorry about that, and thanks for your work helping to maintain the project! @vemv Would you prefer to write up the new issue yourself? |
Cheers 🍻 Will create an issue. I think GH recently introduced "subtask" issue types. |
In case it's useful, today I ran the following experiment: (require 'benchmark)
(format "%.8f" (benchmark-elapse (cider-current-repl))) It returns (We have fixed/simplified a bunch of Sesman aspects this summer) Would love to know if anything has changed for you. Cheers - V |
I have a wip PR that further optimizes things #3463 |
Expected behavior
Several features in Cider run on short idle timers or hooks that run synchronously while the user is typing/editing.
Any latency or momentary freezes in these contexts make for a unpleasant editing experience, so they should be made as optimized as possible.
Actual behavior
Many of these features call the harmless-looking
(cider-current-repl)
under the hood, and profiling shows it is in fact an expensive operation involving various heuristics and filesystem lookups.The performance impact grows ~linearly with the number of open REPL connections, See #3343 for more details and a sample backtrace.
cider-eldoc
andcider-complete
via.(cider-connected-p)
clojure-indent-line
viacider--get-symbol-indent
->cider-resolve-var
.These are the main ones that showed up when profiling my own editing usage - eg. I don't use
cider-format
but that seems like it should be affected too. Searching the codebase also finds variouscider-nrepl-sync-request
functions taking an optionalconnection
arg which forwards to a call to(cider-current-repl)
.Steps to reproduce the problem
Set the eldoc timer to a short delay to magnify the issues:
(setq eldoc-idle-delay 0.01)
then jack-in to many projects simultaneously.
Navigating or typing in a connected clj source buffer should get increasingly laggy with the number of open connections.
Environment & Version information
CIDER version information
Lein / Clojure CLI version
Clojure CLI version 1.11.1.1273
Emacs version
GNU Emacs 29.0.90
Operating system
macOS 12.5.1
JDK distribution
Temurin jdk-20.0.1+9
The text was updated successfully, but these errors were encountered: