Skip to content

Commit aa9c213

Browse files
committed
http: ServeFile to handle Range header for partial requests
and send Content-Length. Also includes some testing of the server code. R=rsc CC=golang-dev https://golang.org/cl/2831041
1 parent 8984fa8 commit aa9c213

File tree

3 files changed

+264
-4
lines changed

3 files changed

+264
-4
lines changed

src/pkg/http/fs.go

Lines changed: 91 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import (
1212
"mime"
1313
"os"
1414
"path"
15+
"strconv"
1516
"strings"
1617
"time"
1718
"utf8"
@@ -130,23 +131,50 @@ func serveFile(w ResponseWriter, r *Request, name string, redirect bool) {
130131
}
131132

132133
// serve file
134+
size := d.Size
135+
code := StatusOK
136+
133137
// use extension to find content type.
134138
ext := path.Ext(name)
135139
if ctype := mime.TypeByExtension(ext); ctype != "" {
136140
w.SetHeader("Content-Type", ctype)
137141
} else {
138142
// read first chunk to decide between utf-8 text and binary
139143
var buf [1024]byte
140-
n, _ := io.ReadFull(f, buf[0:])
141-
b := buf[0:n]
144+
n, _ := io.ReadFull(f, buf[:])
145+
b := buf[:n]
142146
if isText(b) {
143147
w.SetHeader("Content-Type", "text-plain; charset=utf-8")
144148
} else {
145149
w.SetHeader("Content-Type", "application/octet-stream") // generic binary
146150
}
147-
w.Write(b)
151+
f.Seek(0, 0) // rewind to output whole file
152+
}
153+
154+
// handle Content-Range header.
155+
// TODO(adg): handle multiple ranges
156+
ranges, err := parseRange(r.Header["Range"], size)
157+
if err != nil || len(ranges) > 1 {
158+
Error(w, err.String(), StatusRequestedRangeNotSatisfiable)
159+
return
160+
}
161+
if len(ranges) == 1 {
162+
ra := ranges[0]
163+
if _, err := f.Seek(ra.start, 0); err != nil {
164+
Error(w, err.String(), StatusRequestedRangeNotSatisfiable)
165+
return
166+
}
167+
size = ra.length
168+
code = StatusPartialContent
169+
w.SetHeader("Content-Range", fmt.Sprintf("%d-%d/%d", ra.start, ra.start+ra.length, d.Size))
148170
}
149-
io.Copy(w, f)
171+
172+
w.SetHeader("Accept-Ranges", "bytes")
173+
w.SetHeader("Content-Length", strconv.Itoa64(size))
174+
175+
w.WriteHeader(code)
176+
177+
io.Copyn(w, f, size)
150178
}
151179

152180
// ServeFile replies to the request with the contents of the named file or directory.
@@ -174,3 +202,62 @@ func (f *fileHandler) ServeHTTP(w ResponseWriter, r *Request) {
174202
path = path[len(f.prefix):]
175203
serveFile(w, r, f.root+"/"+path, true)
176204
}
205+
206+
// httpRange specifies the byte range to be sent to the client.
207+
type httpRange struct {
208+
start, length int64
209+
}
210+
211+
// parseRange parses a Range header string as per RFC 2616.
212+
func parseRange(s string, size int64) ([]httpRange, os.Error) {
213+
if s == "" {
214+
return nil, nil // header not present
215+
}
216+
const b = "bytes="
217+
if !strings.HasPrefix(s, b) {
218+
return nil, os.NewError("invalid range")
219+
}
220+
var ranges []httpRange
221+
for _, ra := range strings.Split(s[len(b):], ",", -1) {
222+
i := strings.Index(ra, "-")
223+
if i < 0 {
224+
return nil, os.NewError("invalid range")
225+
}
226+
start, end := ra[:i], ra[i+1:]
227+
var r httpRange
228+
if start == "" {
229+
// If no start is specified, end specifies the
230+
// range start relative to the end of the file.
231+
i, err := strconv.Atoi64(end)
232+
if err != nil {
233+
return nil, os.NewError("invalid range")
234+
}
235+
if i > size {
236+
i = size
237+
}
238+
r.start = size - i
239+
r.length = size - r.start
240+
} else {
241+
i, err := strconv.Atoi64(start)
242+
if err != nil || i > size || i < 0 {
243+
return nil, os.NewError("invalid range")
244+
}
245+
r.start = i
246+
if end == "" {
247+
// If no end is specified, range extends to end of the file.
248+
r.length = size - r.start
249+
} else {
250+
i, err := strconv.Atoi64(end)
251+
if err != nil || r.start > i {
252+
return nil, os.NewError("invalid range")
253+
}
254+
if i >= size {
255+
i = size - 1
256+
}
257+
r.length = i - r.start + 1
258+
}
259+
}
260+
ranges = append(ranges, r)
261+
}
262+
return ranges, nil
263+
}

src/pkg/http/fs_test.go

Lines changed: 172 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,172 @@
1+
// Copyright 2010 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package http
6+
7+
import (
8+
"fmt"
9+
"io/ioutil"
10+
"net"
11+
"os"
12+
"sync"
13+
"testing"
14+
)
15+
16+
var ParseRangeTests = []struct {
17+
s string
18+
length int64
19+
r []httpRange
20+
}{
21+
{"", 0, nil},
22+
{"foo", 0, nil},
23+
{"bytes=", 0, nil},
24+
{"bytes=5-4", 10, nil},
25+
{"bytes=0-2,5-4", 10, nil},
26+
{"bytes=0-9", 10, []httpRange{{0, 10}}},
27+
{"bytes=0-", 10, []httpRange{{0, 10}}},
28+
{"bytes=5-", 10, []httpRange{{5, 5}}},
29+
{"bytes=0-20", 10, []httpRange{{0, 10}}},
30+
{"bytes=15-,0-5", 10, nil},
31+
{"bytes=-5", 10, []httpRange{{5, 5}}},
32+
{"bytes=-15", 10, []httpRange{{0, 10}}},
33+
{"bytes=0-499", 10000, []httpRange{{0, 500}}},
34+
{"bytes=500-999", 10000, []httpRange{{500, 500}}},
35+
{"bytes=-500", 10000, []httpRange{{9500, 500}}},
36+
{"bytes=9500-", 10000, []httpRange{{9500, 500}}},
37+
{"bytes=0-0,-1", 10000, []httpRange{{0, 1}, {9999, 1}}},
38+
{"bytes=500-600,601-999", 10000, []httpRange{{500, 101}, {601, 399}}},
39+
{"bytes=500-700,601-999", 10000, []httpRange{{500, 201}, {601, 399}}},
40+
}
41+
42+
func TestParseRange(t *testing.T) {
43+
for _, test := range ParseRangeTests {
44+
r := test.r
45+
ranges, err := parseRange(test.s, test.length)
46+
if err != nil && r != nil {
47+
t.Errorf("parseRange(%q) returned error %q", test.s, err)
48+
}
49+
if len(ranges) != len(r) {
50+
t.Errorf("len(parseRange(%q)) = %d, want %d", test.s, len(ranges), len(r))
51+
continue
52+
}
53+
for i := range r {
54+
if ranges[i].start != r[i].start {
55+
t.Errorf("parseRange(%q)[%d].start = %d, want %d", test.s, i, ranges[i].start, r[i].start)
56+
}
57+
if ranges[i].length != r[i].length {
58+
t.Errorf("parseRange(%q)[%d].length = %d, want %d", test.s, i, ranges[i].length, r[i].length)
59+
}
60+
}
61+
}
62+
}
63+
64+
const (
65+
testFile = "testdata/file"
66+
testFileLength = 11
67+
)
68+
69+
var (
70+
serverOnce sync.Once
71+
serverAddr string
72+
)
73+
74+
func startServer(t *testing.T) {
75+
serverOnce.Do(func() {
76+
HandleFunc("/ServeFile", func(w ResponseWriter, r *Request) {
77+
ServeFile(w, r, "testdata/file")
78+
})
79+
l, err := net.Listen("tcp", "127.0.0.1:0")
80+
if err != nil {
81+
t.Fatal("listen:", err)
82+
}
83+
serverAddr = l.Addr().String()
84+
go Serve(l, nil)
85+
})
86+
}
87+
88+
var ServeFileRangeTests = []struct {
89+
start, end int
90+
r string
91+
code int
92+
}{
93+
{0, testFileLength, "", StatusOK},
94+
{0, 5, "0-4", StatusPartialContent},
95+
{2, testFileLength, "2-", StatusPartialContent},
96+
{testFileLength - 5, testFileLength, "-5", StatusPartialContent},
97+
{3, 8, "3-7", StatusPartialContent},
98+
{0, 0, "20-", StatusRequestedRangeNotSatisfiable},
99+
}
100+
101+
func TestServeFile(t *testing.T) {
102+
startServer(t)
103+
var err os.Error
104+
105+
file, err := ioutil.ReadFile(testFile)
106+
if err != nil {
107+
t.Fatal("reading file:", err)
108+
}
109+
110+
// set up the Request (re-used for all tests)
111+
var req Request
112+
req.Header = make(map[string]string)
113+
if req.URL, err = ParseURL("http://" + serverAddr + "/ServeFile"); err != nil {
114+
t.Fatal("ParseURL:", err)
115+
}
116+
req.Method = "GET"
117+
118+
// straight GET
119+
_, body := getBody(t, req)
120+
if !equal(body, file) {
121+
t.Fatalf("body mismatch: got %q, want %q", body, file)
122+
}
123+
124+
// Range tests
125+
for _, rt := range ServeFileRangeTests {
126+
req.Header["Range"] = "bytes=" + rt.r
127+
if rt.r == "" {
128+
req.Header["Range"] = ""
129+
}
130+
r, body := getBody(t, req)
131+
if r.StatusCode != rt.code {
132+
t.Errorf("range=%q: StatusCode=%d, want %d", rt.r, r.StatusCode, rt.code)
133+
}
134+
if rt.code == StatusRequestedRangeNotSatisfiable {
135+
continue
136+
}
137+
h := fmt.Sprintf("%d-%d/%d", rt.start, rt.end, testFileLength)
138+
if rt.r == "" {
139+
h = ""
140+
}
141+
if r.Header["Content-Range"] != h {
142+
t.Errorf("header mismatch: range=%q: got %q, want %q", rt.r, r.Header["Content-Range"], h)
143+
}
144+
if !equal(body, file[rt.start:rt.end]) {
145+
t.Errorf("body mismatch: range=%q: got %q, want %q", rt.r, body, file[rt.start:rt.end])
146+
}
147+
}
148+
}
149+
150+
func getBody(t *testing.T, req Request) (*Response, []byte) {
151+
r, err := send(&req)
152+
if err != nil {
153+
t.Fatal(req.URL.String(), "send:", err)
154+
}
155+
b, err := ioutil.ReadAll(r.Body)
156+
if err != nil {
157+
t.Fatal("reading Body:", err)
158+
}
159+
return r, b
160+
}
161+
162+
func equal(a, b []byte) bool {
163+
if len(a) != len(b) {
164+
return false
165+
}
166+
for i := range a {
167+
if a[i] != b[i] {
168+
return false
169+
}
170+
}
171+
return true
172+
}

src/pkg/http/testdata/file

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
0123456789

0 commit comments

Comments
 (0)