-
Notifications
You must be signed in to change notification settings - Fork 17.9k
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
runtime: change in Caller prevents detecting program name #23128
Comments
Can you just look at |
I'm looking for the full import path of the program. When adding instrumentation to https://godoc.org/golang.org/x/tools/cmd/present, the path I'd look for is "golang.org/x/tools/cmd/present". We'd end up with a bunch of different programs claiming to be "server", "api", or "worker". I guess it could be done via |
Something like this? |
Why does it need to be called in init? Calling it from |
Calling from @cznic I'm looking for the import path of the program's Reading the symbol table has worked so far on linux and darwin. I haven't tackled windows support yet. Looking at the call stack relied on implementation details, but it was cross-platform. The runtime package provides access to GOOS/GOARCH, the compiler name, the compiler version, and GOROOT. Maybe it could also give access to the import path ("name") of the program? |
Correct, but then why not just call it from within the |
The results are used in several metrics/logging packages. They are initialized in a variety of places across a large number of programs (each with a different main package) which are maintained by different teams. Migrating all of our programs to use a new API which only works when called from the |
Hi @rhysh. Sorry that this broke for you. If you're willing to try more implementation-dependent tricks, I believe it should still work to call
My concern with doing something like this is that package/import paths aren't actually part of the Go language. They're part of the build system, and there are build systems that define them differently (e.g., Blaze [Bazel used to, but it seems that was fixed, thankfully]). And this may get more complicated even in the go tool with changes for package versioning. |
I would argue that this is actually the "right" way to do this, even though I understand it requires a little platform-dependent code (I think only a few lines per platform though?). It doesn't depend on any implementation details and it goes through well-defined interfaces. Even the runtime doesn't really have this information handy; it obviously has a map from PC to function, but there's no map the other way around. |
This is unfortunate but it sounds like there are workarounds. Closing. Please comment if you disagree. |
Right @ianlancetaylor , the workarounds are sufficient for me. Thanks @aclements for the advice on |
I used to have a way for packages to determine the name of the program that included them. This is very helpful for providing easy-to-use metrics and logging packages: the package determines the name of the program, and everyone knows how to map between the name of the metrics/logs and the location of the relevant source code. The implementation details I was using to do that have changed during the Go 1.10 cycle, and it's not clear how I can move forward.
What version of Go are you using (
go version
)?go1.10beta1:
Does this issue reproduce with the latest release?
This is a regression since Go 1.9, present in go1.10beta1.
What operating system and processor architecture are you using (
go env
)?darwin/amd64, linux/amd64
What did you do?
Since Go 1.4, I've maintained a (closed-source) package that allows programs to determine their name: the import path of the "main" package. This was based on using
runtime.Caller
in aninit
function until encounteringmain.init
. The filename defining that function would allow my package to determine the import path of the program (assuming $GOPATH didn't include a bonus "/src/" element).This worked well until Go 1.9, when the filenames of all init functions changed to be
<autogenerated>
. I updated my package to rely on even more arcane implementation details, usingruntime.FuncForPC
to do a binary search of the executable between the PC formain.init
and the PC for its own init function to find other code within package main that included a real filename, and returning the import path it calculated from that.Then, I upgraded to go1.10beta1 and found that the package was once again broken, with
git bisect
pointing to the removal of<autogenerated>
methods from stack traces (e972095). /cc @aclementsWhat did you expect to see?
I expected that the caller of an
init
function would continue to be the code that caused it to run—that is, the package that caused it to be part of the program by importing it.I expect there to be a way for programs to determine their identity. This package is deployed in a large number of (closed-source) applications, where it is used when naming logs and metric streams for a clear bidirectional mapping between metrics and the programs that generated them.
I expected to be able to work around the change, with some other way of getting ahold of a program counter for something in package main, without having to dig even deeper into unsupported behavior.
From #22231 (comment), "Using Caller(1) in an init function is definitely depending on an implementation detail that could easily change for any number of reasons." makes it clear that this behavior should not be trusted .. but what is the alternative?
What did you see instead?
Today https://play.golang.org/p/-RRrePXZMo prints three frames, and would print more if it were in a non-main package:
But with go1.10beta1, it prints only two:
Critically, the final frame is in the function that calls
runtime.Stack
/runtime.Caller
and no longer leads all the way up the import tree to the "main" package.The text was updated successfully, but these errors were encountered: