frankenphp/frankenphp_test.go

326 lines
10 KiB
Go
Raw Normal View History

2021-09-24 16:52:20 +00:00
package frankenphp_test
import (
"fmt"
2021-10-31 23:18:30 +00:00
"io"
2022-05-19 15:42:12 +00:00
"log"
2021-10-31 23:18:30 +00:00
"net/http"
"net/http/cookiejar"
2021-10-31 23:18:30 +00:00
"net/http/httptest"
"net/url"
"os"
"strings"
"sync"
2021-09-24 16:52:20 +00:00
"testing"
"github.com/dunglas/frankenphp"
"github.com/stretchr/testify/assert"
)
type testOptions struct {
workerScript string
nbWorkers int
nbParrallelRequests int
realServer bool
2021-11-01 14:38:12 +00:00
}
func runTest(t *testing.T, test func(func(http.ResponseWriter, *http.Request), *httptest.Server, int), opts *testOptions) {
if opts == nil {
opts = &testOptions{}
}
if opts.nbWorkers == 0 {
opts.nbWorkers = 2
}
if opts.nbParrallelRequests == 0 {
opts.nbParrallelRequests = 5
}
2021-09-24 16:52:20 +00:00
cwd, _ := os.Getwd()
testDataDir := cwd + "/testdata/"
2021-10-31 23:18:30 +00:00
assert.Nil(t, frankenphp.Startup())
defer frankenphp.Shutdown()
2021-10-31 23:18:30 +00:00
if opts.workerScript != "" {
frankenphp.StartWorkers(testDataDir+opts.workerScript, opts.nbWorkers)
defer frankenphp.StopWorkers()
}
2021-10-31 23:18:30 +00:00
handler := func(w http.ResponseWriter, r *http.Request) {
var err error
2021-10-31 23:18:30 +00:00
req := frankenphp.NewRequestWithContext(r, testDataDir)
if opts.workerScript == "" {
err = frankenphp.ExecuteScript(w, req)
} else {
err = frankenphp.WorkerHandleRequest(w, req)
}
2021-10-31 23:18:30 +00:00
assert.Nil(t, err)
2021-10-31 23:18:30 +00:00
}
var ts *httptest.Server
if opts.realServer {
ts = httptest.NewServer(http.HandlerFunc(handler))
defer ts.Close()
}
var wg sync.WaitGroup
wg.Add(opts.nbParrallelRequests)
for i := 0; i < opts.nbParrallelRequests; i++ {
go func(i int) {
test(handler, ts, i)
wg.Done()
}(i)
}
wg.Wait()
2021-10-31 23:18:30 +00:00
}
func TestStartup(t *testing.T) {
2021-10-31 23:18:30 +00:00
defer frankenphp.Shutdown()
assert.Nil(t, frankenphp.Startup())
frankenphp.Shutdown()
2021-10-31 23:18:30 +00:00
assert.Nil(t, frankenphp.Startup())
}
2021-10-31 23:18:30 +00:00
func TestHelloWorld_module(t *testing.T) { testHelloWorld(t, nil) }
func TestHelloWorld_worker(t *testing.T) { testHelloWorld(t, &testOptions{workerScript: "index.php"}) }
func testHelloWorld(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("GET", fmt.Sprintf("http://example.com/index.php?i=%d", i), nil)
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
2021-10-31 23:18:30 +00:00
assert.Equal(t, fmt.Sprintf("I am by birth a Genevese (%d)", i), string(body))
}, opts)
2021-10-31 23:18:30 +00:00
}
func TestServerVariable_module(t *testing.T) { testServerVariable(t, nil) }
func TestServerVariable_worker(t *testing.T) {
testServerVariable(t, &testOptions{workerScript: "server-variable.php"})
}
func testServerVariable(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("GET", fmt.Sprintf("http://example.com/server-variable.php?foo=a&bar=b&i=%d#hash", i), nil)
req.SetBasicAuth("kevin", "password")
w := httptest.NewRecorder()
handler(w, req)
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
strBody := string(body)
assert.Contains(t, strBody, "[REMOTE_HOST]")
assert.Contains(t, strBody, "[REMOTE_USER] => kevin")
assert.Contains(t, strBody, "[PHP_AUTH_USER] => kevin")
assert.Contains(t, strBody, "[PHP_AUTH_PW] => password")
assert.Contains(t, strBody, "[HTTP_AUTHORIZATION] => Basic a2V2aW46cGFzc3dvcmQ=")
assert.Contains(t, strBody, "[DOCUMENT_ROOT]")
assert.Contains(t, strBody, "[CONTENT_TYPE]")
assert.Contains(t, strBody, fmt.Sprintf("[QUERY_STRING] => foo=a&bar=b&i=%d#hash", i))
assert.Contains(t, strBody, fmt.Sprintf("[REQUEST_URI] => /server-variable.php?foo=a&bar=b&i=%d#hash", i))
assert.Contains(t, strBody, "[SCRIPT_NAME]")
assert.Contains(t, strBody, "[CONTENT_LENGTH]")
assert.Contains(t, strBody, "[REMOTE_ADDR]")
assert.Contains(t, strBody, "[REMOTE_PORT]")
assert.Contains(t, strBody, "[REQUEST_SCHEME] => http")
assert.Contains(t, strBody, "[DOCUMENT_URI]")
assert.Contains(t, strBody, "[AUTH_TYPE]")
assert.Contains(t, strBody, "[REMOTE_IDENT]")
assert.Contains(t, strBody, "[REQUEST_METHOD] => GET")
assert.Contains(t, strBody, "[SERVER_NAME] => example.com")
assert.Contains(t, strBody, "[SERVER_PROTOCOL] => HTTP/1.1")
assert.Contains(t, strBody, "[SCRIPT_FILENAME]")
assert.Contains(t, strBody, "[SERVER_SOFTWARE] => FrankenPHP")
assert.Contains(t, strBody, "[REQUEST_TIME_FLOAT]")
assert.Contains(t, strBody, "[REQUEST_TIME]")
assert.Contains(t, strBody, "[REQUEST_TIME]")
}, opts)
}
2021-10-31 23:18:30 +00:00
func TestPathInfo_module(t *testing.T) { testPathInfo(t, nil) }
func TestPathInfo_worker(t *testing.T) {
testPathInfo(t, &testOptions{workerScript: "server-variable.php"})
}
func testPathInfo(t *testing.T, opts *testOptions) {
runTest(t, func(_ func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
handler := func(w http.ResponseWriter, r *http.Request) {
cwd, _ := os.Getwd()
testDataDir := cwd + "/testdata/"
rewriteRequest := frankenphp.NewRequestWithContext(r, testDataDir)
rewriteRequest.URL.Path = "/server-variable.php/pathinfo"
fc, _ := frankenphp.FromContext(rewriteRequest.Context())
fc.Env["REQUEST_URI"] = r.URL.RequestURI()
if opts == nil {
assert.Nil(t, frankenphp.ExecuteScript(w, rewriteRequest))
} else {
assert.Nil(t, frankenphp.WorkerHandleRequest(w, rewriteRequest))
}
}
req := httptest.NewRequest("GET", fmt.Sprintf("http://example.com/pathinfo/%d", i), nil)
w := httptest.NewRecorder()
handler(w, req)
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
strBody := string(body)
assert.Contains(t, strBody, "[PATH_INFO] => /pathinfo")
assert.Contains(t, strBody, fmt.Sprintf("[REQUEST_URI] => /pathinfo/%d", i))
assert.Contains(t, strBody, "[PATH_TRANSLATED] =>")
assert.Contains(t, strBody, "[SCRIPT_NAME] => /server-variable.php")
}, opts)
}
2021-10-31 23:18:30 +00:00
func TestHeaders_module(t *testing.T) { testHeaders(t, nil) }
func TestHeaders_worker(t *testing.T) { testHeaders(t, &testOptions{workerScript: "headers.php"}) }
func testHeaders(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("GET", fmt.Sprintf("http://example.com/headers.php?i=%d", i), nil)
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
2021-10-31 23:18:30 +00:00
assert.Equal(t, "Hello", string(body))
assert.Equal(t, 201, resp.StatusCode)
assert.Equal(t, "bar", resp.Header.Get("Foo"))
assert.Equal(t, "bar2", resp.Header.Get("Foo2"))
assert.Equal(t, fmt.Sprintf("%d", i), resp.Header.Get("I"))
}, opts)
2021-10-31 23:18:30 +00:00
}
func TestInput_module(t *testing.T) { testInput(t, nil) }
func TestInput_worker(t *testing.T) { testInput(t, &testOptions{workerScript: "input.php"}) }
func testInput(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("POST", "http://example.com/input.php", strings.NewReader(fmt.Sprintf("post data %d", i)))
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
assert.Equal(t, fmt.Sprintf("post data %d", i), string(body))
assert.Equal(t, "bar", resp.Header.Get("Foo"))
}, opts)
}
func TestPostSuperGlobals_module(t *testing.T) { testPostSuperGlobals(t, nil) }
func TestPostSuperGlobals_worker(t *testing.T) {
testPostSuperGlobals(t, &testOptions{workerScript: "super-globals.php"})
}
func testPostSuperGlobals(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
formData := url.Values{"baz": {"bat"}, "i": {fmt.Sprintf("%d", i)}}
req := httptest.NewRequest("POST", fmt.Sprintf("http://example.com/super-globals.php?foo=bar&iG=%d", i), strings.NewReader(formData.Encode()))
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
2021-10-31 23:18:30 +00:00
assert.Contains(t, string(body), "'foo' => 'bar'")
assert.Contains(t, string(body), fmt.Sprintf("'i' => '%d'", i))
assert.Contains(t, string(body), "'baz' => 'bat'")
assert.Contains(t, string(body), fmt.Sprintf("'iG' => '%d'", i))
}, opts)
2021-10-31 23:18:30 +00:00
}
func TestCookies_module(t *testing.T) { testCookies(t, nil) }
func TestCookies_worker(t *testing.T) { testCookies(t, &testOptions{workerScript: "cookies.php"}) }
func testCookies(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("GET", "http://example.com/cookies.php", nil)
req.AddCookie(&http.Cookie{Name: "foo", Value: "bar"})
req.AddCookie(&http.Cookie{Name: "i", Value: fmt.Sprintf("%d", i)})
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
2021-10-31 23:18:30 +00:00
assert.Contains(t, string(body), "'foo' => 'bar'")
assert.Contains(t, string(body), fmt.Sprintf("'i' => '%d'", i))
}, opts)
}
2021-10-31 23:18:30 +00:00
func TestSession_module(t *testing.T) { testSession(t, nil) }
func TestSession_worker(t *testing.T) {
testSession(t, &testOptions{workerScript: "session.php"})
2021-10-31 23:18:30 +00:00
}
func testSession(t *testing.T, opts *testOptions) {
if opts == nil {
opts = &testOptions{}
}
opts.realServer = true
runTest(t, func(_ func(http.ResponseWriter, *http.Request), ts *httptest.Server, i int) {
jar, err := cookiejar.New(&cookiejar.Options{})
if err != nil {
panic(err)
}
client := &http.Client{Jar: jar}
resp1, err := client.Get(ts.URL + "/session.php")
if err != nil {
panic(err)
}
2021-10-31 23:18:30 +00:00
body1, _ := io.ReadAll(resp1.Body)
assert.Equal(t, "Count: 0\n", string(body1))
resp2, err := client.Get(ts.URL + "/session.php")
if err != nil {
panic(err)
}
body2, _ := io.ReadAll(resp2.Body)
assert.Equal(t, "Count: 1\n", string(body2))
}, opts)
}
2021-10-31 23:18:30 +00:00
func TestPhpInfo_module(t *testing.T) { testPhpInfo(t, nil) }
func TestPhpInfo_worker(t *testing.T) { testPhpInfo(t, &testOptions{workerScript: "phpinfo.php"}) }
func testPhpInfo(t *testing.T, opts *testOptions) {
runTest(t, func(handler func(http.ResponseWriter, *http.Request), _ *httptest.Server, i int) {
req := httptest.NewRequest("GET", fmt.Sprintf("http://example.com/phpinfo.php?i=%d", i), nil)
w := httptest.NewRecorder()
handler(w, req)
2021-10-31 23:18:30 +00:00
resp := w.Result()
body, _ := io.ReadAll(resp.Body)
assert.Contains(t, string(body), "frankenphp")
assert.Contains(t, string(body), fmt.Sprintf("i=%d", i))
}, opts)
2021-09-24 16:52:20 +00:00
}
2022-05-19 15:42:12 +00:00
2022-05-20 07:42:30 +00:00
func ExampleExecuteScript() {
2022-05-19 15:42:12 +00:00
frankenphp.Startup()
defer frankenphp.Shutdown()
phpHandler := func(w http.ResponseWriter, req *http.Request) {
if err := frankenphp.ExecuteScript(w, req); err != nil {
log.Print(fmt.Errorf("error executing PHP script: %w", err))
}
}
http.HandleFunc("/", phpHandler)
log.Fatal(http.ListenAndServe(":8080", nil))
}