client.go 2.65 KB
Newer Older
1 2 3
package http

import (
4
	"bytes"
5 6
	"encoding/json"
	"fmt"
7
	"io"
8 9 10 11 12 13
	"net/http"
	"strings"

	cmds "github.com/jbenet/go-ipfs/commands"
)

14 15 16 17
const (
	ApiUrlFormat = "http://%s%s/%s"
	ApiPath      = "/api/v0" // TODO: make configurable
)
18

19 20 21 22 23 24 25 26
// Client is the commands HTTP client interface.
type Client interface {
	Send(req cmds.Request) (cmds.Response, error)
}

type client struct {
	serverAddress string
}
27

28 29 30
func NewClient(address string) Client {
	return &client{address}
}
31

32
func (c *client) Send(req cmds.Request) (cmds.Response, error) {
33 34
	path := strings.Join(req.Path(), "/")
	url := fmt.Sprintf(ApiUrlFormat, c.serverAddress, ApiPath, path)
35

36 37 38 39 40 41 42 43 44 45
	var userEncoding string
	if enc, found := req.Option(cmds.EncShort); found {
		userEncoding = enc.(string)
		req.SetOption(cmds.EncShort, cmds.JSON)
	} else {
		enc, _ := req.Option(cmds.EncLong)
		userEncoding = enc.(string)
		req.SetOption(cmds.EncLong, cmds.JSON)
	}

46 47 48
	// TODO: handle multiple files with multipart
	var in io.Reader

49
	query := "?"
50
	for k, v := range req.Options() {
51 52
		query += "&" + k + "=" + v.(string)
	}
53 54

	args := req.Arguments()
55 56 57
	argDefs := req.Command().Arguments
	var argDef cmds.Argument

58
	for i, arg := range args {
59 60 61 62 63
		if i < len(argDefs) {
			argDef = argDefs[i]
		}

		if argDef.Type == cmds.ArgString {
64 65 66 67 68 69 70
			query += "&arg=" + arg.(string)

		} else {
			// TODO: multipart
			if in != nil {
				return nil, fmt.Errorf("Currently, only one file stream is possible per request")
			}
71
			in = arg.(io.Reader)
72
		}
73
	}
74

75
	httpRes, err := http.Post(url+query, "application/octet-stream", in)
76 77 78 79 80 81 82 83 84 85
	if err != nil {
		return nil, err
	}

	res := cmds.NewResponse(req)

	contentType := httpRes.Header["Content-Type"][0]
	contentType = strings.Split(contentType, ";")[0]

	if contentType == "application/octet-stream" {
86
		res.SetOutput(httpRes.Body)
87 88 89 90 91 92 93
		return res, nil
	}

	dec := json.NewDecoder(httpRes.Body)

	if httpRes.StatusCode >= http.StatusBadRequest {
		e := cmds.Error{}
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113

		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 {
				fmt.Println(err)
				return nil, err
			}
114 115 116 117 118
		}

		res.SetError(e, e.Code)

	} else {
119
		v := req.Command().Type
120 121 122 123 124 125
		err = dec.Decode(&v)
		if err != nil {
			fmt.Println(err)
			return nil, err
		}

126
		res.SetOutput(v)
127 128 129 130 131 132
	}

	if len(userEncoding) > 0 {
		req.SetOption(cmds.EncShort, userEncoding)
		req.SetOption(cmds.EncLong, userEncoding)
	}
133 134 135

	return res, nil
}