Sometimes users report "X is slow", where X is usually something that has
some debug output, like hg push. Having timestamps as part of the ui.debug()
output will help identify what they're seeing. The timestamp is customizable
using [ui]debugtimestampfmt, which accepts any timestamp that
datetime.strftime() can handle. I'm using datetime.strftime() since it provides
the %f format, at least on some platforms, and time.strftime() does not; having
sub-second precision is pretty useful for these messages.
Details
- Reviewers
- None
- Group Reviewers
hg-reviewers
Diff Detail
- Repository
- rHG Mercurial
- Lint
Lint Skipped - Unit
Unit Tests Skipped
Event Timeline
Someone pointed me at https://bitbucket.org/octobus/mercurial-devel/commits/668806cb43a0 and mentioned discussions on the devel mailing list that I haven't tracked down yet. Theirs is more complete and probably better than mine, so it would be nice to unify them if possible, at least in configuration.
Since this patch is only touching 'ui.debug()', it's easy to identify places where we write 'ui.debug(' and don't have a \n['"] in the line, and go and check them to see if they're actually attempting single-line output in multiple invocations. Most of the cases I found were my grep being very simplistic, some I sent patches for in this series because they *should* have had a \n or weren't doing anything, and in the end there was only this one in histedit.py:1189 that I wasn't sure about and didn't touch:
ui.debug('histedit: %s is replaced by %s\n' % ( node.short(prec), node.short(succs[0]))) if len(succs) > 1: m = 'histedit: %s' for n in succs[1:]: ui.debug(m % node.short(n))
I'll try to track down those mailing list conversations now
https://www.mercurial-scm.org/pipermail/mercurial-devel/2017-April/096617.html is the mailing list thread. Sounds like core is generally negative on the idea, which is disappointing, but OK. I can drop it easily :)
FWIW, if you don't have ts installed. That's 4 lines in Ruby:
#!/usr/bin/env ruby start = Time.now.to_f STDIN.each_line do |l| puts "#{'%.3f' % (Time.now.to_f - start)} #{l}" end
Besides, I believe --profile is the more correct way to do profiling.
I'm not going to recommend to users to "see if you can reproduce the slowness issues while running with PYTHONUNBUFFERED=1 /usr/bin/hg --debug | ruby -e'start = Time.now.to_f' -e'STDIN.each_line do |l|' -e" puts \"#{'%.3f' % (Time.now.to_f - start)} #{l}\"" -e'end'" :) Yes, we can make a wrapper for that I suppose.
Regarding profiling: --profile is CPU profiling, right? It doesn't actually tell you *when something happened*. If you're trying to match up things in the blackbox.log with what the user saw on the screen, it doesn't help much. If you're trying to identify when the user saw pauses, it's a more complicated mechanism than just looking at the output the user saw but having timestamps. :/
Maybe that's what I should be doing instead, though. A flag/option to make it so that if debugflag is set, the blackbox gets the output in addition to the screen...
--profile is wall time by default. If the callgraph is more complete (ex. traceprof's case), it does tell you when something happens.
While chg does print timestamps its debug output, I think in a Python world, using a proper profiling API is cleaner than abusing ui.debug. As for now, it seems most ui.debugs are used for revealing internal states for tests to verify, less about profiling.