kubo/commands/http/client.go
2014-11-05 10:15:17 -08:00

178 lines
3.4 KiB
Go

package http
import (
"bytes"
"encoding/json"
"errors"
"fmt"
"io"
"net/http"
"net/url"
"strings"
cmds "github.com/jbenet/go-ipfs/commands"
)
var castError = errors.New("cast error")
const (
ApiUrlFormat = "http://%s%s/%s?%s"
ApiPath = "/api/v0" // TODO: make configurable
)
// Client is the commands HTTP client interface.
type Client interface {
Send(req cmds.Request) (cmds.Response, error)
}
type client struct {
serverAddress string
}
func NewClient(address string) Client {
return &client{address}
}
func (c *client) Send(req cmds.Request) (cmds.Response, error) {
var userEncoding string
if enc, found := req.Option(cmds.EncShort); found {
var ok bool
userEncoding, ok = enc.(string)
if !ok {
return nil, castError
}
req.SetOption(cmds.EncShort, cmds.JSON)
} else {
var ok bool
enc, _ := req.Option(cmds.EncLong)
userEncoding, ok = enc.(string)
if !ok {
return nil, castError
}
req.SetOption(cmds.EncLong, cmds.JSON)
}
query, inputStream, err := getQuery(req)
if err != nil {
return nil, err
}
path := strings.Join(req.Path(), "/")
url := fmt.Sprintf(ApiUrlFormat, c.serverAddress, ApiPath, path, query)
httpRes, err := http.Post(url, "application/octet-stream", inputStream)
if err != nil {
return nil, err
}
res, err := getResponse(httpRes, req)
if err != nil {
return nil, err
}
if len(userEncoding) > 0 {
req.SetOption(cmds.EncShort, userEncoding)
req.SetOption(cmds.EncLong, userEncoding)
}
return res, nil
}
func getQuery(req cmds.Request) (string, io.Reader, error) {
// TODO: handle multiple files with multipart
var inputStream io.Reader
query := url.Values{}
for k, v := range req.Options() {
str, ok := v.(string)
if !ok {
return "", nil, castError
}
query.Set(k, str)
}
args := req.Arguments()
argDefs := req.Command().Arguments
var argDef cmds.Argument
for i, arg := range args {
if i < len(argDefs) {
argDef = argDefs[i]
}
if argDef.Type == cmds.ArgString {
str, ok := arg.(string)
if !ok {
return "", nil, castError
}
query.Add("arg", str)
} else {
// TODO: multipart
if inputStream != nil {
return "", nil, fmt.Errorf("Currently, only one file stream is possible per request")
}
var ok bool
inputStream, ok = arg.(io.Reader)
if !ok {
return "", nil, castError
}
}
}
return query.Encode(), inputStream, 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["Content-Type"][0]
contentType = strings.Split(contentType, ";")[0]
if contentType == "application/octet-stream" {
res.SetOutput(httpRes.Body)
return res, nil
}
dec := json.NewDecoder(httpRes.Body)
if httpRes.StatusCode >= http.StatusBadRequest {
e := cmds.Error{}
if httpRes.StatusCode == http.StatusNotFound {
// handle 404s
e.Message = "Command not found."
e.Code = cmds.ErrClient
} else if contentType == "text/plain" {
// handle non-marshalled errors
buf := bytes.NewBuffer(nil)
io.Copy(buf, httpRes.Body)
e.Message = string(buf.Bytes())
e.Code = cmds.ErrNormal
} else {
// handle marshalled errors
err = dec.Decode(&e)
if err != nil {
return nil, err
}
}
res.SetError(e, e.Code)
} else {
v := req.Command().Type
err = dec.Decode(&v)
if err != nil {
return nil, err
}
res.SetOutput(v)
}
return res, nil
}