1
0
mirror of https://github.com/dutchcoders/transfer.sh.git synced 2020-11-18 19:53:40 -08:00
2019-03-17 20:19:56 +01:00

210 lines
5.6 KiB
Go

// Copyright 2015 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Package body allows for the replacement of message body on responses.
package body
import (
"bytes"
"crypto/rand"
"encoding/json"
"fmt"
"io"
"io/ioutil"
"mime/multipart"
"net/http"
"net/textproto"
"strconv"
"strings"
"github.com/google/martian/log"
"github.com/google/martian/parse"
)
func init() {
parse.Register("body.Modifier", modifierFromJSON)
}
// Modifier substitutes the body on an HTTP response.
type Modifier struct {
contentType string
body []byte
boundary string
}
type modifierJSON struct {
ContentType string `json:"contentType"`
Body []byte `json:"body"` // Body is expected to be a Base64 encoded string.
Scope []parse.ModifierType `json:"scope"`
}
// NewModifier constructs and returns a body.Modifier.
func NewModifier(b []byte, contentType string) *Modifier {
log.Debugf("body.NewModifier: len(b): %d, contentType %s", len(b), contentType)
return &Modifier{
contentType: contentType,
body: b,
boundary: randomBoundary(),
}
}
// modifierFromJSON takes a JSON message as a byte slice and returns a
// body.Modifier and an error.
//
// Example JSON Configuration message:
// {
// "scope": ["request", "response"],
// "contentType": "text/plain",
// "body": "c29tZSBkYXRhIHdpdGggACBhbmQg77u/" // Base64 encoded body
// }
func modifierFromJSON(b []byte) (*parse.Result, error) {
msg := &modifierJSON{}
if err := json.Unmarshal(b, msg); err != nil {
return nil, err
}
mod := NewModifier(msg.Body, msg.ContentType)
return parse.NewResult(mod, msg.Scope)
}
// ModifyRequest sets the Content-Type header and overrides the request body.
func (m *Modifier) ModifyRequest(req *http.Request) error {
log.Debugf("body.ModifyRequest: request: %s", req.URL)
req.Body.Close()
req.Header.Set("Content-Type", m.contentType)
// Reset the Content-Encoding since we know that the new body isn't encoded.
req.Header.Del("Content-Encoding")
req.ContentLength = int64(len(m.body))
req.Body = ioutil.NopCloser(bytes.NewReader(m.body))
return nil
}
// SetBoundary set the boundary string used for multipart range responses.
func (m *Modifier) SetBoundary(boundary string) {
m.boundary = boundary
}
// ModifyResponse sets the Content-Type header and overrides the response body.
func (m *Modifier) ModifyResponse(res *http.Response) error {
log.Debugf("body.ModifyResponse: request: %s", res.Request.URL)
// Replace the existing body, close it first.
res.Body.Close()
res.Header.Set("Content-Type", m.contentType)
// Reset the Content-Encoding since we know that the new body isn't encoded.
res.Header.Del("Content-Encoding")
// If no range request header is present, return the body as the response body.
if res.Request.Header.Get("Range") == "" {
res.ContentLength = int64(len(m.body))
res.Body = ioutil.NopCloser(bytes.NewReader(m.body))
return nil
}
rh := res.Request.Header.Get("Range")
rh = strings.ToLower(rh)
sranges := strings.Split(strings.TrimLeft(rh, "bytes="), ",")
var ranges [][]int
for _, rng := range sranges {
if strings.HasSuffix(rng, "-") {
rng = fmt.Sprintf("%s%d", rng, len(m.body)-1)
}
rs := strings.Split(rng, "-")
if len(rs) != 2 {
res.StatusCode = http.StatusRequestedRangeNotSatisfiable
return nil
}
start, err := strconv.Atoi(strings.TrimSpace(rs[0]))
if err != nil {
return err
}
end, err := strconv.Atoi(strings.TrimSpace(rs[1]))
if err != nil {
return err
}
if start > end {
res.StatusCode = http.StatusRequestedRangeNotSatisfiable
return nil
}
ranges = append(ranges, []int{start, end})
}
// Range request.
res.StatusCode = http.StatusPartialContent
// Single range request.
if len(ranges) == 1 {
start := ranges[0][0]
end := ranges[0][1]
seg := m.body[start : end+1]
res.ContentLength = int64(len(seg))
res.Body = ioutil.NopCloser(bytes.NewReader(seg))
res.Header.Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", start, end, len(m.body)))
return nil
}
// Multipart range request.
var mpbody bytes.Buffer
mpw := multipart.NewWriter(&mpbody)
mpw.SetBoundary(m.boundary)
for _, rng := range ranges {
start, end := rng[0], rng[1]
mimeh := make(textproto.MIMEHeader)
mimeh.Set("Content-Type", m.contentType)
mimeh.Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", start, end, len(m.body)))
seg := m.body[start : end+1]
pw, err := mpw.CreatePart(mimeh)
if err != nil {
return err
}
if _, err := pw.Write(seg); err != nil {
return err
}
}
mpw.Close()
res.ContentLength = int64(len(mpbody.Bytes()))
res.Body = ioutil.NopCloser(bytes.NewReader(mpbody.Bytes()))
res.Header.Set("Content-Type", fmt.Sprintf("multipart/byteranges; boundary=%s", m.boundary))
return nil
}
// randomBoundary generates a 30 character string for boundaries for mulipart range
// requests. This func panics if io.Readfull fails.
// Borrowed from: https://golang.org/src/mime/multipart/writer.go?#L73
func randomBoundary() string {
var buf [30]byte
_, err := io.ReadFull(rand.Reader, buf[:])
if err != nil {
panic(err)
}
return fmt.Sprintf("%x", buf[:])
}