Daniel, I would characterize the use case you describe as a very indirect way of setting these options on transactions. That's can be a good thing when contexts are already passed around, and coming up with new ways of passing such options down through a few layers of code is some amount of work. It's certainly convenient that they are part of the context in that case.
That said, I believe there are large downsides to this indirect API design. Even when you benefit from this convenience, it encourages a form of "spooky action at a distance" programming where it's not clear at the BeginContext call site what the properties of the transaction will be. Using Go has made me very appreciative of other language features that directly remove this "spooky action at a distance" from other programming languages. Not having exceptions is a great example of this. I would say that Go trades convenience in favor of clarity in those circumstances, and I'm concerned that we're making the opposite tradeoff here. One can then easily argue that nothing is stopping me or anybody else from forgoing the indirect nature and finding a way of propagating these options down to the code that actually calls BeginContext. In fact, that's probably what I would do. The problem then is that the API is needlessly awkward; it would be much more straightforward to expose an API with functional options if that's the encouraged way of using it. Another worry of mine is that contexts are passed on to other functions. Even though it's rare that a single request consists of multiple transactions, it still happens and I wouldn't be too surprised to find people accidentally setting the wrong transaction properties simply because it was accidentally inherited from the context that was used to begin another transaction. Again, being more explicit in the API design would cause people to stop and think before passing along a set of transaction options from an unrelated transaction, similar to how Go's explicit error handling forces you to think about errors, even if it's less convenient. To end with a concrete proposal, I think changing the signature to be "func (*DB) BeginContext(ctx context.Context, opts ...TxOption)" or similar (and then changing the transaction properties to be TxOptions) would solve most of the issues I highlighted above. André On Friday, December 9, 2016 at 9:14:08 PM UTC+1, Daniel Theophanes wrote: > > Hi Chandra, > > In my view using context to store value for these uses (Read-Only and > Isolation Level) is somewhat of a gray area. They are not a classical > request only scope, like a tracing ID, but they also aren't strictly > limited to a single request either. Say you wanted to kick off a request, > but ensure you don't modify the database in anyway. you could pass in a > context with ReadOnly set and any further request function will follow that > directive. Similarly, by allowing you to set the Isolation at a higher > level, you can effectively influence the behavior of the queries for any > subsequent request action. > > Should we do it this way? Maybe, maybe not. I think there are benefits to > this approach, and I think they can be considered request scoped, even if > in the degenerate case they are set at the query level. However, I think it > will be really common to set these when dispatching requests in the router > and using them for the subsequent requests. > > Has /report/ prefix? Set to ReadOnly and Isolation X > Has /api/ prefix? Set Isolation to Y > > At least that is how I would envision using them. I have been known to be > wrong before. > What do you think? -Daniel > > > On Thursday, December 8, 2016 at 11:27:36 PM UTC-8, Chandra Sekar S wrote: >> >> Bump >> >> -- >> Chandra Sekar.S >> >> On Wed, Dec 7, 2016 at 10:24 AM, Chandru <chand...@gmail.com> wrote: >> >>> I can understand db/sql using Context for cancellation. It is the >>> optional arguments to BeginContext like IsolationLevel and read-only flag, >>> which are not request-specific, that seem to contradict context's >>> documentation. >>> >>> -- >>> Chandra Sekar.S >>> >>> On Tue, Dec 6, 2016 at 9:50 PM, <parais...@gmail.com> wrote: >>> >>>> Either the doc should be changed or the std lib should follow the >>>> spirit of the doc. But my understanding is that context.Context is not >>>> just >>>> about HTTP request/response cycle but deals with anything that needs >>>> cancellation or a resource clean up signal , some kind of DIY RAII . In >>>> any >>>> case the documentation should be clarified. >>>> >>>> Le mardi 6 décembre 2016 16:48:48 UTC+1, Chandra Sekar S a écrit : >>>>> >>>>> Documentation of the context package says, >>>>> >>>>> "Use context Values only for request-scoped data that transits >>>>> processes and APIs, not for passing optional parameters to functions." >>>>> >>>>> sql.BeginContext introduced in 1.8, uses Context to receive options >>>>> like IsolationLevel and read-only flag. These are neither >>>>> request-specific >>>>> nor cross-cutting. They are options that are typically specific to a type >>>>> of operation, but common to all requests. >>>>> >>>>> Isn't this use in db/sql contradicting the recommendation in context's >>>>> doc? >>>>> >>>>> -- >>>>> Chandra Sekar.S >>>>> >>>> -- >>>> You received this message because you are subscribed to the Google >>>> Groups "golang-nuts" group. >>>> To unsubscribe from this group and stop receiving emails from it, send >>>> an email to golang-nuts...@googlegroups.com. >>>> For more options, visit https://groups.google.com/d/optout. >>>> >>> >>> >> -- You received this message because you are subscribed to the Google Groups "golang-nuts" group. To unsubscribe from this group and stop receiving emails from it, send an email to golang-nuts+unsubscr...@googlegroups.com. For more options, visit https://groups.google.com/d/optout.