perkeep/pkg/context/context.go

138 lines
3.0 KiB
Go

/*
Copyright 2013 The Camlistore Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
// Package context provides a Context type to propagate state and cancellation
// information.
package context
import (
"errors"
"net/http"
"sync"
)
// ErrCanceled may be returned by code when it receives from a Context.Done channel.
var ErrCanceled = errors.New("canceled")
// TODO returns a dummy context. It's a signal that the caller code is not yet correct,
// and needs its own context to propagate.
func TODO() *Context {
return nil
}
// A Context is carries state and cancellation information between calls.
// A nil Context pointer is valid, for now.
type Context struct {
cancelOnce sync.Once
done chan struct{}
parent *Context
httpClient *http.Client // nil means default
}
// New returns a new Context.
// Any provided params modify the returned context.
func New(params ...Param) *Context {
c := &Context{
done: make(chan struct{}),
}
for _, p := range params {
p.modify(c)
}
return c
}
// HTTPClient returns the HTTP Client to use for this context.
func (c *Context) HTTPClient() *http.Client {
if c != nil {
if cl := c.httpClient; cl != nil {
return cl
}
return c.parent.HTTPClient()
}
return http.DefaultClient
}
// WithHTTPClient sets the HTTP client as returned by HTTPClient.
func WithHTTPClient(cl *http.Client) Param {
return httpParam{cl}
}
// A Param modifies a Context as returned by New.
type Param interface {
modify(*Context)
}
type httpParam struct {
cl *http.Client
}
func (p httpParam) modify(c *Context) {
c.httpClient = p.cl
}
// New returns a child context attached to the receiver parent context c.
// The returned context is done when the parent is done, but the returned child
// context can be canceled independently without affecting the parent.
func (c *Context) New(params ...Param) *Context {
subc := New()
subc.parent = c
for _, p := range params {
p.modify(subc)
}
if c == nil {
return subc
}
go func() {
<-c.Done()
subc.Cancel()
}()
return subc
}
// Done returns a channel that is closed when the Context is canceled
// or finished.
func (c *Context) Done() <-chan struct{} {
if c == nil {
return nil
}
return c.done
}
// IsCanceled reports whether this context has been canceled.
func (c *Context) IsCanceled() bool {
select {
case <-c.Done():
return true
default:
return false
}
}
// Cancel cancels the context. It is idempotent.
func (c *Context) Cancel() {
if c == nil {
return
}
c.cancelOnce.Do(c.cancel)
}
func (c *Context) cancel() {
if c.done != nil {
close(c.done)
}
}