package http import ( "context" "encoding/json" "errors" "fmt" "io" "io/ioutil" "net/http" "net/url" "reflect" "strconv" "strings" cmds "github.com/ipfs/go-ipfs/commands" config "github.com/ipfs/go-ipfs/repo/config" "gx/ipfs/QmUyfy4QSr3NXym4etEiRyxBLqqAeKHJuRdi8AACxg63fZ/go-ipfs-cmdkit" ) const ( ApiUrlFormat = "http://%s%s/%s?%s" ApiPath = "/api/v0" // TODO: make configurable ) var OptionSkipMap = map[string]bool{ "api": true, } // Client is the commands HTTP client interface. type Client interface { Send(req cmds.Request) (cmds.Response, error) } type client struct { serverAddress string httpClient *http.Client } func NewClient(address string) Client { return &client{ serverAddress: address, httpClient: http.DefaultClient, } } func (c *client) Send(req cmds.Request) (cmds.Response, error) { if req.Context() == nil { log.Warningf("no context set in request") if err := req.SetRootContext(context.TODO()); err != nil { return nil, err } } // save user-provided encoding previousUserProvidedEncoding, found, err := req.Option(cmdkit.EncShort).String() if err != nil { return nil, err } // override with json to send to server req.SetOption(cmdkit.EncShort, cmds.JSON) // stream channel output req.SetOption(cmdkit.ChanOpt, "true") query, err := getQuery(req) if err != nil { return nil, err } var fileReader *MultiFileReader var reader io.Reader if req.Files() != nil { fileReader = NewMultiFileReader(req.Files(), true) reader = fileReader } path := strings.Join(req.Path(), "/") url := fmt.Sprintf(ApiUrlFormat, c.serverAddress, ApiPath, path, query) httpReq, err := http.NewRequest("POST", url, reader) if err != nil { return nil, err } // TODO extract string consts? if fileReader != nil { httpReq.Header.Set(contentTypeHeader, "multipart/form-data; boundary="+fileReader.Boundary()) } else { httpReq.Header.Set(contentTypeHeader, applicationOctetStream) } httpReq.Header.Set(uaHeader, config.ApiVersion) httpReq.Cancel = req.Context().Done() httpReq.Close = true httpRes, err := c.httpClient.Do(httpReq) if err != nil { return nil, err } // using the overridden JSON encoding in request res, err := getResponse(httpRes, req) if err != nil { return nil, err } if found && len(previousUserProvidedEncoding) > 0 { // reset to user provided encoding after sending request // NB: if user has provided an encoding but it is the empty string, // still leave it as JSON. req.SetOption(cmdkit.EncShort, previousUserProvidedEncoding) } return res, nil } func getQuery(req cmds.Request) (string, error) { query := url.Values{} for k, v := range req.Options() { if OptionSkipMap[k] { continue } str := fmt.Sprintf("%v", v) query.Set(k, str) } args := req.StringArguments() argDefs := req.Command().Arguments argDefIndex := 0 for _, arg := range args { argDef := argDefs[argDefIndex] // skip ArgFiles for argDef.Type == cmdkit.ArgFile { argDefIndex++ argDef = argDefs[argDefIndex] } query.Add("arg", arg) if len(argDefs) > argDefIndex+1 { argDefIndex++ } } return query.Encode(), nil } // getResponse decodes a http.Response to create a cmds.Response func getResponse(httpRes *http.Response, req cmds.Request) (cmds.Response, error) { var err error res := cmds.NewResponse(req) contentType := httpRes.Header.Get(contentTypeHeader) contentType = strings.Split(contentType, ";")[0] lengthHeader := httpRes.Header.Get(extraContentLengthHeader) if len(lengthHeader) > 0 { length, err := strconv.ParseUint(lengthHeader, 10, 64) if err != nil { return nil, err } res.SetLength(length) } rr := &httpResponseReader{httpRes} res.SetCloser(rr) if contentType != applicationJson { // for all non json output types, just stream back the output res.SetOutput(rr) return res, nil } else if len(httpRes.Header.Get(channelHeader)) > 0 { // if output is coming from a channel, decode each chunk outChan := make(chan interface{}) go readStreamedJson(req, rr, outChan, res) res.SetOutput((<-chan interface{})(outChan)) return res, nil } dec := json.NewDecoder(rr) // If we ran into an error if httpRes.StatusCode >= http.StatusBadRequest { var e *cmdkit.Error switch { case httpRes.StatusCode == http.StatusNotFound: // handle 404s e = &cmdkit.Error{Message: "Command not found.", Code: cmdkit.ErrClient} case contentType == plainText: // handle non-marshalled errors mes, err := ioutil.ReadAll(rr) if err != nil { return nil, err } e = &cmdkit.Error{Message: string(mes), Code: cmdkit.ErrNormal} default: // handle marshalled errors var rxErr cmdkit.Error err = dec.Decode(&rxErr) if err != nil { return nil, err } e = &rxErr } res.SetError(e, e.Code) return res, nil } outputType := reflect.TypeOf(req.Command().Type) v, err := decodeTypedVal(outputType, dec) if err != nil && err != io.EOF { return nil, err } res.SetOutput(v) return res, nil } // read json objects off of the given stream, and write the objects out to // the 'out' channel func readStreamedJson(req cmds.Request, rr io.Reader, out chan<- interface{}, resp cmds.Response) { defer close(out) dec := json.NewDecoder(rr) outputType := reflect.TypeOf(req.Command().Type) ctx := req.Context() for { v, err := decodeTypedVal(outputType, dec) if err != nil { if err != io.EOF { log.Error(err) resp.SetError(err, cmdkit.ErrNormal) } return } select { case <-ctx.Done(): return case out <- v: } } } // decode a value of the given type, if the type is nil, attempt to decode into // an interface{} anyways func decodeTypedVal(t reflect.Type, dec *json.Decoder) (interface{}, error) { var v interface{} var err error if t != nil { v = reflect.New(t).Interface() err = dec.Decode(v) } else { err = dec.Decode(&v) } return v, err } // httpResponseReader reads from the response body, and checks for an error // in the http trailer upon EOF, this error if present is returned instead // of the EOF. type httpResponseReader struct { resp *http.Response } func (r *httpResponseReader) Read(b []byte) (int, error) { n, err := r.resp.Body.Read(b) // reading on a closed response body is as good as an io.EOF here if err != nil && strings.Contains(err.Error(), "read on closed response body") { err = io.EOF } if err == io.EOF { _ = r.resp.Body.Close() trailerErr := r.checkError() if trailerErr != nil { return n, trailerErr } } return n, err } func (r *httpResponseReader) checkError() error { if e := r.resp.Trailer.Get(StreamErrHeader); e != "" { return errors.New(e) } return nil } func (r *httpResponseReader) Close() error { return r.resp.Body.Close() }