1
0
mirror of https://github.com/ipfs/kubo.git synced 2025-09-11 23:46:39 +08:00
Files
kubo/commands/http/handler.go

172 lines
3.9 KiB
Go

package http
import (
"errors"
"fmt"
"io"
"net/http"
cmds "github.com/jbenet/go-ipfs/commands"
u "github.com/jbenet/go-ipfs/util"
)
var log = u.Logger("commands/http")
type Handler struct {
ctx cmds.Context
root *cmds.Command
origin string
}
var ErrNotFound = errors.New("404 page not found")
const (
streamHeader = "X-Stream-Output"
channelHeader = "X-Chunked-Output"
contentTypeHeader = "Content-Type"
contentLengthHeader = "Content-Length"
transferEncodingHeader = "Transfer-Encoding"
)
var mimeTypes = map[string]string{
cmds.JSON: "application/json",
cmds.XML: "application/xml",
cmds.Text: "text/plain",
}
func NewHandler(ctx cmds.Context, root *cmds.Command, origin string) *Handler {
// allow whitelisted origins (so we can make API requests from the browser)
if len(origin) > 0 {
log.Info("Allowing API requests from origin: " + origin)
}
return &Handler{ctx, root, origin}
}
func (i Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
log.Debug("Incoming API request: ", r.URL)
if len(i.origin) > 0 {
w.Header().Set("Access-Control-Allow-Origin", i.origin)
}
w.Header().Set("Access-Control-Allow-Headers", "Content-Type")
req, err := Parse(r, i.root)
if err != nil {
if err == ErrNotFound {
w.WriteHeader(http.StatusNotFound)
} else {
w.WriteHeader(http.StatusBadRequest)
}
w.Write([]byte(err.Error()))
return
}
req.SetContext(i.ctx)
// call the command
res := i.root.Call(req)
// set the Content-Type based on res output
if _, ok := res.Output().(io.Reader); ok {
// we don't set the Content-Type for streams, so that browsers can MIME-sniff the type themselves
// we set this header so clients have a way to know this is an output stream
// (not marshalled command output)
// TODO: set a specific Content-Type if the command response needs it to be a certain type
w.Header().Set(streamHeader, "1")
} else {
enc, found, err := req.Option(cmds.EncShort).String()
if err != nil || !found {
w.WriteHeader(http.StatusInternalServerError)
return
}
mime := mimeTypes[enc]
w.Header().Set(contentTypeHeader, mime)
}
// if the res output is a channel, set a custom header for it
isChan := false
if _, ok := res.Output().(chan interface{}); ok {
w.Header().Set(channelHeader, "1")
isChan = true
}
// if response contains an error, write an HTTP error status code
if e := res.Error(); e != nil {
if e.Code == cmds.ErrClient {
w.WriteHeader(http.StatusBadRequest)
} else {
w.WriteHeader(http.StatusInternalServerError)
}
}
out, err := res.Reader()
if err != nil {
w.WriteHeader(http.StatusInternalServerError)
w.Header().Set(contentTypeHeader, "text/plain")
w.Write([]byte(err.Error()))
return
}
if isChan {
err = copyChunks(w, out)
if err != nil {
log.Error(err)
fmt.Println(err)
}
return
}
io.Copy(w, out)
}
// Copies from an io.Reader to a http.ResponseWriter.
// Flushes chunks over HTTP stream as they are read (if supported by transport).
func copyChunks(w http.ResponseWriter, out io.Reader) error {
hijacker, ok := w.(http.Hijacker)
if !ok {
return errors.New("Could not create hijacker")
}
conn, writer, err := hijacker.Hijack()
if err != nil {
return err
}
defer conn.Close()
writer.WriteString("HTTP/1.1 200 OK\r\n")
writer.WriteString(contentTypeHeader + ": application/json\r\n")
writer.WriteString(transferEncodingHeader + ": chunked\r\n")
writer.WriteString(channelHeader + ": 1\r\n\r\n")
buf := make([]byte, 32*1024)
for {
n, err := out.Read(buf)
if n > 0 {
length := fmt.Sprintf("%x\r\n", n)
writer.WriteString(length)
_, err := writer.Write(buf[0:n])
if err != nil {
return err
}
writer.WriteString("\r\n")
writer.Flush()
}
if err != nil && err != io.EOF {
return err
}
if err == io.EOF {
break
}
}
writer.WriteString("0\r\n\r\n")
writer.Flush()
return nil
}