| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- // Copyright 2015 The go-ethereum Authors
- // This file is part of go-ethereum.
- //
- // go-ethereum is free software: you can redistribute it and/or modify
- // it under the terms of the GNU Lesser General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // go-ethereum is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU Lesser General Public License for more details.
- //
- // You should have received a copy of the GNU Lesser General Public License
- // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
- package docserver
- import (
- "fmt"
- "io/ioutil"
- "net/http"
- "path/filepath"
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/crypto"
- )
- type DocServer struct {
- *http.Transport
- DocRoot string
- schemes []string
- }
- func New(docRoot string) (self *DocServer) {
- self = &DocServer{
- Transport: &http.Transport{},
- DocRoot: docRoot,
- schemes: []string{"file"},
- }
- self.DocRoot = "/tmp/"
- self.RegisterProtocol("file", http.NewFileTransport(http.Dir(self.DocRoot)))
- return
- }
- // Clients should be reused instead of created as needed. Clients are safe for concurrent use by multiple goroutines.
- // A Client is higher-level than a RoundTripper (such as Transport) and additionally handles HTTP details such as cookies and redirects.
- func (self *DocServer) Client() *http.Client {
- return &http.Client{
- Transport: self,
- }
- }
- func (self *DocServer) RegisterScheme(scheme string, rt http.RoundTripper) {
- self.schemes = append(self.schemes, scheme)
- self.RegisterProtocol(scheme, rt)
- }
- func (self *DocServer) HasScheme(scheme string) bool {
- for _, s := range self.schemes {
- if s == scheme {
- return true
- }
- }
- return false
- }
- func (self *DocServer) GetAuthContent(uri string, hash common.Hash) (content []byte, err error) {
- // retrieve content
- content, err = self.Get(uri, "")
- if err != nil {
- return
- }
- // check hash to authenticate content
- chash := crypto.Sha3Hash(content)
- if chash != hash {
- content = nil
- err = fmt.Errorf("content hash mismatch %x != %x (exp)", hash[:], chash[:])
- }
- return
- }
- // Get(uri, path) downloads the document at uri, if path is non-empty it
- // is interpreted as a filepath to which the contents are saved
- func (self *DocServer) Get(uri, path string) (content []byte, err error) {
- // retrieve content
- resp, err := self.Client().Get(uri)
- defer func() {
- if resp != nil {
- resp.Body.Close()
- }
- }()
- if err != nil {
- return
- }
- content, err = ioutil.ReadAll(resp.Body)
- if err != nil {
- return
- }
- if path != "" {
- var abspath string
- abspath, err = filepath.Abs(path)
- ioutil.WriteFile(abspath, content, 0700)
- }
- return
- }
|