1 // Copyright 2011 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.
5 // Tests a Go CGI program running under a Go CGI host process.
6 // Further, the two programs are the same binary, just checking
7 // their environment to figure out what mode to run in.
24 // This test is a CGI host (testing host.go) that runs its own binary
25 // as a child process testing the other half of CGI (child.go).
26 func TestHostingOurselves(t
*testing
.T
) {
27 if runtime
.GOOS
== "nacl" {
28 t
.Skip("skipping on nacl")
34 Args
: []string{"-test.run=TestBeChildCGIProcess"},
36 expectedMap
:= map[string]string{
37 "test": "Hello CGI-in-CGI",
40 "env-GATEWAY_INTERFACE": "CGI/1.1",
41 "env-HTTP_HOST": "example.com",
43 "env-QUERY_STRING": "foo=bar&a=b",
44 "env-REMOTE_ADDR": "1.2.3.4",
45 "env-REMOTE_HOST": "1.2.3.4",
46 "env-REQUEST_METHOD": "GET",
47 "env-REQUEST_URI": "/test.go?foo=bar&a=b",
48 "env-SCRIPT_FILENAME": os
.Args
[0],
49 "env-SCRIPT_NAME": "/test.go",
50 "env-SERVER_NAME": "example.com",
51 "env-SERVER_PORT": "80",
52 "env-SERVER_SOFTWARE": "go",
54 replay
:= runCgiTest(t
, h
, "GET /test.go?foo=bar&a=b HTTP/1.0\nHost: example.com\n\n", expectedMap
)
56 if expected
, got
:= "text/html; charset=utf-8", replay
.Header().Get("Content-Type"); got
!= expected
{
57 t
.Errorf("got a Content-Type of %q; expected %q", got
, expected
)
59 if expected
, got
:= "X-Test-Value", replay
.Header().Get("X-Test-Header"); got
!= expected
{
60 t
.Errorf("got a X-Test-Header of %q; expected %q", got
, expected
)
64 type customWriterRecorder
struct {
66 *httptest
.ResponseRecorder
69 func (r
*customWriterRecorder
) Write(p
[]byte) (n
int, err error
) {
73 type limitWriter
struct {
78 func (w
*limitWriter
) Write(p
[]byte) (n
int, err error
) {
87 err
= errors
.New("past write limit")
92 // If there's an error copying the child's output to the parent, test
93 // that we kill the child.
94 func TestKillChildAfterCopyError(t
*testing
.T
) {
95 if runtime
.GOOS
== "nacl" {
96 t
.Skip("skipping on nacl")
99 defer func() { testHookStartProcess
= nil }()
100 proc
:= make(chan *os
.Process
, 1)
101 testHookStartProcess
= func(p
*os
.Process
) {
108 Args
: []string{"-test.run=TestBeChildCGIProcess"},
110 req
, _
:= http
.NewRequest("GET", "http://example.com/test.cgi?write-forever=1", nil)
111 rec
:= httptest
.NewRecorder()
113 const writeLen
= 50 << 10
114 rw
:= &customWriterRecorder
{&limitWriter
{&out
, writeLen
}, rec
}
116 donec
:= make(chan bool, 1)
124 if out
.Len() != writeLen || out
.Bytes()[0] != 'a' {
125 t
.Errorf("unexpected output: %q", out
.Bytes())
127 case <-time
.After(5 * time
.Second
):
128 t
.Errorf("timeout. ServeHTTP hung and didn't kill the child process?")
132 t
.Logf("killed process")
134 t
.Logf("didn't kill process")
139 // Test that a child handler writing only headers works.
140 // golang.org/issue/7196
141 func TestChildOnlyHeaders(t
*testing
.T
) {
142 if runtime
.GOOS
== "nacl" {
143 t
.Skip("skipping on nacl")
149 Args
: []string{"-test.run=TestBeChildCGIProcess"},
151 expectedMap
:= map[string]string{
154 replay
:= runCgiTest(t
, h
, "GET /test.go?no-body=1 HTTP/1.0\nHost: example.com\n\n", expectedMap
)
155 if expected
, got
:= "X-Test-Value", replay
.Header().Get("X-Test-Header"); got
!= expected
{
156 t
.Errorf("got a X-Test-Header of %q; expected %q", got
, expected
)
160 // golang.org/issue/7198
161 func Test500WithNoHeaders(t
*testing
.T
) { want500Test(t
, "/immediate-disconnect") }
162 func Test500WithNoContentType(t
*testing
.T
) { want500Test(t
, "/no-content-type") }
163 func Test500WithEmptyHeaders(t
*testing
.T
) { want500Test(t
, "/empty-headers") }
165 func want500Test(t
*testing
.T
, path
string) {
169 Args
: []string{"-test.run=TestBeChildCGIProcess"},
171 expectedMap
:= map[string]string{
174 replay
:= runCgiTest(t
, h
, "GET "+path
+" HTTP/1.0\nHost: example.com\n\n", expectedMap
)
175 if replay
.Code
!= 500 {
176 t
.Errorf("Got code %d; want 500", replay
.Code
)
180 type neverEnding
byte
182 func (b neverEnding
) Read(p
[]byte) (n
int, err error
) {
189 // Note: not actually a test.
190 func TestBeChildCGIProcess(t
*testing
.T
) {
191 if os
.Getenv("REQUEST_METHOD") == "" {
192 // Not in a CGI environment; skipping test.
195 switch os
.Getenv("REQUEST_URI") {
196 case "/immediate-disconnect":
198 case "/no-content-type":
199 fmt
.Printf("Content-Length: 6\n\nHello\n")
201 case "/empty-headers":
202 fmt
.Printf("\nHello")
205 Serve(http
.HandlerFunc(func(rw http
.ResponseWriter
, req
*http
.Request
) {
206 rw
.Header().Set("X-Test-Header", "X-Test-Value")
208 if req
.FormValue("no-body") == "1" {
211 if req
.FormValue("write-forever") == "1" {
212 io
.Copy(rw
, neverEnding('a'))
214 time
.Sleep(5 * time
.Second
) // hang forever, until killed
217 fmt
.Fprintf(rw
, "test=Hello CGI-in-CGI\n")
218 for k
, vv
:= range req
.Form
{
219 for _
, v
:= range vv
{
220 fmt
.Fprintf(rw
, "param-%s=%s\n", k
, v
)
223 for _
, kv
:= range os
.Environ() {
224 fmt
.Fprintf(rw
, "env-%s\n", kv
)