-
Notifications
You must be signed in to change notification settings - Fork 17.8k
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
encoding/gob: encoder should ignore embedded structs with no exported fields #5819
Comments
This is a tough one. The reason to leave things alone is that it's a very common user error to attempt to marshal a struct with no exported fields, and the error message helps diagnose the case with a clear error. However, diagnosing it in this more complicated case might be possible. Labels changed: added priority-later, removed priority-triage. Status changed to Accepted. |
Labels changed: added packagechange. Owner changed to @robpike. |
@robpike is there any indication when this is going to be implemented? |
Would it be possible to make the encoder configurable with a handful of flags? Have an interface that structs with unexported fields can meet to serialize / deserialize themselves? Running into problems with a library's configuration struct that includes an x509.CertPool not having any exported fields... |
First, surely it's wiser to change the x509 library (which can be done as a strict add-on by providing a GobEncoder interface) than to change the transport layer. Second, if you have configurations in the transport layer you introduce the likelihood that both sides of the transport may need to be configured the same, and that introduces a whole new set of problems. |
What configuration struct are you having problems with? If you wrote it
yourself, I was hoping you could add `gob:"-"` to exclude the field of type
x509.CertPool, like you can in the json and xml encoders. That doesn't work
today, but maybe we should do that for Go 1.7.
|
tl;dr my expectation is that gob should be able to cleanly serialize the total visible serializable state of any instance of a Go type. I was trying to serialize and deserialize cluster configurations within the gocql library: https://godoc.org/github.com/gocql/gocql#ClusterConfig (Also, embarrassed to say I missed the encoding.Binary{Marshaler,Unmarshaler} interfaces) Unfortunately I didn't write gocql, so for now I'm using my own struct and copying values over. Either way I find the behavior of gob in this circumstance to be somewhat inconsistent with the rest of the library. Given that gob does nice things around skipping function pointers, channels, unexported fields, etc. cleanly and making a best effort around type matching when the tags and field names don't match, throwing an error if a struct with no exported fields is visible anywhere in your object hierarchy is surprising. The spirit of gob otherwise seems to be very much do the sane thing with best effort around all exposed state and while I would understand the condition of no exported fields anywhere in the hierarchy being an error, it would be convenient to be able to override the behavior with respect to non-root nodes in the object tree so to speak. Not asking for a change in default behavior, but perhaps increased configurability. Also a change would have no effect on the binary format or the unmarshal / deserialize operation. Happy to propose a change and put up a PR if you are amenable to it? |
👍 this can be annoying if you're trying to serialize a type from a dependency and there's a struct with no exported fields embedded in something. In that case the answer shouldn't have to be "change the downstream library just to deal with this". |
There's a pretty common pattern in the code I'm working on where a type foo struct {
sync.RWMutex
Foo string
Bar int
} When trying to encode structs like this I get an error with |
Have another way to solve locks for struct fields and convert them to gob? |
@happierall I know this is quite an old issue, but here is how we solved it: If we have the following structure foo: type foo struct {
sync.RWMutex
Bar string
Baz int
} We create an Encode struct: type fooEncode struct {
Bar *string
Baz *int
} We then do this to encode: fooInstance.RLock()
defer fooInstance.RUnlock()
encodable := fooEncode {
Bar: &fooInstance.Bar,
Baz: &fooInstance.Baz,
}
err := gobEncoder.Encode(encodable)
... This is a lot of work for just embedding |
Any word on this? My use case is having structs that embed log15.Logger. For the one struct that I encode with gob I've had to work-around this by storing it on unexported field, giving the ugly mystruct.logger.Debug("foo"), instead of mystruct.Debug("foo") used everywhere else in my code. |
Change https://golang.org/cl/117875 mentions this issue: |
Unexported fields were correctly handled except structs without exported field for which an error message is returned. The check returning the error in the Encode is removed in the submitted fix as documentation is not excluding the case. The Decode part is also fixed for idem-potency. Tests have been amended. |
What is the rationale for the "limitation"? The fields of my structs are intentionally unexported because I do not want any other parts of the program to have direct access to it, but inside this file I would like to encode and decode this struct because I have visibility and it is OK or should be OK to do that. What should I use instead then? Would Nevermind, I got "invalid type". |
The Encoder you are using isn't part of your code. It is another package -
one that might be in the standard library, but still separate from yours.
Within your package could either create a temporary struct to Encode/Decode or you
can write your own Encoder/Decoder within your package.
|
I see. Thanks. :) |
There has been a breaking change in "crypto/elliptic" as mentioned in #57422 (comment) when go1.19 was released. But this code works fine in go1.18. This code is a very critical part of cryptocurrency wallets and I prefer to use the latest version of go. Code: https://go.dev/play/p/c6HKfG47CG4 There mentioned a workaround using the Marshal function in crypto/elliptic. But I am not able to achieve it so far. So, I think solving this issue has been a higher priority since go1.19 was released. I am looking forward to it. SuggestionAs it is mentioned by @robpike in the comment, It is helpful for diagnosis. But this error handling condition should be ignored/overridden if the |
Actually this does not work. I had the same usecase. It still threw error.
But instead what worked for me was to create a new struct without RWMutex and encode/decode that. |
by alan.strohm:
The text was updated successfully, but these errors were encountered: