mirror of https://github.com/perkeep/perkeep.git
138 lines
3.0 KiB
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)
|
|
}
|
|
}
|