123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618 |
- package caddy_test
- import (
- "bytes"
- "fmt"
- "net/http"
- "os"
- "path/filepath"
- "strings"
- "sync"
- "testing"
- "github.com/dunglas/frankenphp"
- "github.com/prometheus/client_golang/prometheus"
- "github.com/prometheus/client_golang/prometheus/testutil"
- "github.com/stretchr/testify/require"
- "github.com/caddyserver/caddy/v2/caddytest"
- )
- var testPort = "9080"
- func TestPHP(t *testing.T) {
- var wg sync.WaitGroup
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- for i := 0; i < 100; i++ {
- wg.Add(1)
- go func(i int) {
- tester.AssertGetResponse(fmt.Sprintf("http://localhost:"+testPort+"/index.php?i=%d", i), http.StatusOK, fmt.Sprintf("I am by birth a Genevese (%d)", i))
- wg.Done()
- }(i)
- }
- wg.Wait()
- }
- func TestLargeRequest(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- tester.AssertPostResponseBody(
- "http://localhost:"+testPort+"/large-request.php",
- []string{},
- bytes.NewBufferString(strings.Repeat("f", 1_048_576)),
- http.StatusOK,
- "Request body size: 1048576 (unknown)",
- )
- }
- func TestWorker(t *testing.T) {
- var wg sync.WaitGroup
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp {
- worker ../testdata/index.php 2
- }
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- for i := 0; i < 100; i++ {
- wg.Add(1)
- go func(i int) {
- tester.AssertGetResponse(fmt.Sprintf("http://localhost:"+testPort+"/index.php?i=%d", i), http.StatusOK, fmt.Sprintf("I am by birth a Genevese (%d)", i))
- wg.Done()
- }(i)
- }
- wg.Wait()
- }
- func TestEnv(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp {
- worker {
- file ../testdata/worker-env.php
- num 1
- env FOO bar
- }
- }
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- env FOO baz
- }
- }
- }
- `, "caddyfile")
- tester.AssertGetResponse("http://localhost:"+testPort+"/worker-env.php", http.StatusOK, "bazbar")
- }
- func TestJsonEnv(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- "admin": {
- "listen": "localhost:2999"
- },
- "apps": {
- "frankenphp": {
- "workers": [
- {
- "env": {
- "FOO": "bar"
- },
- "file_name": "../testdata/worker-env.php",
- "num": 1
- }
- ]
- },
- "http": {
- "http_port": `+testPort+`,
- "https_port": 9443,
- "servers": {
- "srv0": {
- "listen": [
- ":`+testPort+`"
- ],
- "routes": [
- {
- "handle": [
- {
- "handler": "subroute",
- "routes": [
- {
- "handle": [
- {
- "handler": "subroute",
- "routes": [
- {
- "handle": [
- {
- "env": {
- "FOO": "baz"
- },
- "handler": "php",
- "root": "../testdata"
- }
- ]
- }
- ]
- }
- ]
- }
- ]
- }
- ],
- "match": [
- {
- "host": [
- "localhost"
- ]
- }
- ],
- "terminal": true
- }
- ]
- }
- }
- },
- "pki": {
- "certificate_authorities": {
- "local": {
- "install_trust": false
- }
- }
- }
- }
- }
- `, "json")
- tester.AssertGetResponse("http://localhost:"+testPort+"/worker-env.php", http.StatusOK, "bazbar")
- }
- func TestCustomCaddyVariablesInEnv(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp {
- worker {
- file ../testdata/worker-env.php
- num 1
- env FOO world
- }
- }
- }
- localhost:`+testPort+` {
- route {
- map 1 {my_customvar} {
- default "hello "
- }
- php {
- root ../testdata
- env FOO {my_customvar}
- }
- }
- }
- `, "caddyfile")
- tester.AssertGetResponse("http://localhost:"+testPort+"/worker-env.php", http.StatusOK, "hello world")
- }
- func TestPHPServerDirective(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp
- }
- localhost:`+testPort+` {
- root * ../testdata
- php_server
- }
- `, "caddyfile")
- tester.AssertGetResponse("http://localhost:"+testPort, http.StatusOK, "I am by birth a Genevese (i not set)")
- tester.AssertGetResponse("http://localhost:"+testPort+"/hello.txt", http.StatusOK, "Hello")
- tester.AssertGetResponse("http://localhost:"+testPort+"/not-found.txt", http.StatusOK, "I am by birth a Genevese (i not set)")
- }
- func TestPHPServerDirectiveDisableFileServer(t *testing.T) {
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp
- order php_server before respond
- }
- localhost:`+testPort+` {
- root * ../testdata
- php_server {
- file_server off
- }
- respond "Not found" 404
- }
- `, "caddyfile")
- tester.AssertGetResponse("http://localhost:"+testPort, http.StatusOK, "I am by birth a Genevese (i not set)")
- tester.AssertGetResponse("http://localhost:"+testPort+"/not-found.txt", http.StatusOK, "I am by birth a Genevese (i not set)")
- }
- func TestMetrics(t *testing.T) {
- var wg sync.WaitGroup
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- // Make some requests
- for i := 0; i < 10; i++ {
- wg.Add(1)
- go func(i int) {
- tester.AssertGetResponse(fmt.Sprintf("http://localhost:"+testPort+"/index.php?i=%d", i), http.StatusOK, fmt.Sprintf("I am by birth a Genevese (%d)", i))
- wg.Done()
- }(i)
- }
- wg.Wait()
- // Fetch metrics
- resp, err := http.Get("http://localhost:2999/metrics")
- if err != nil {
- t.Fatalf("failed to fetch metrics: %v", err)
- }
- defer resp.Body.Close()
- // Read and parse metrics
- metrics := new(bytes.Buffer)
- _, err = metrics.ReadFrom(resp.Body)
- if err != nil {
- t.Fatalf("failed to read metrics: %v", err)
- }
- cpus := fmt.Sprintf("%d", frankenphp.MaxThreads)
- // Check metrics
- expectedMetrics := `
- # HELP frankenphp_total_threads Total number of PHP threads
- # TYPE frankenphp_total_threads counter
- frankenphp_total_threads ` + cpus + `
- # HELP frankenphp_busy_threads Number of busy PHP threads
- # TYPE frankenphp_busy_threads gauge
- frankenphp_busy_threads 0
- `
- require.NoError(t, testutil.GatherAndCompare(prometheus.DefaultGatherer, strings.NewReader(expectedMetrics), "frankenphp_total_threads", "frankenphp_busy_threads"))
- }
- func TestWorkerMetrics(t *testing.T) {
- var wg sync.WaitGroup
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp {
- worker ../testdata/index.php 2
- }
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- // Make some requests
- for i := 0; i < 10; i++ {
- wg.Add(1)
- go func(i int) {
- tester.AssertGetResponse(fmt.Sprintf("http://localhost:"+testPort+"/index.php?i=%d", i), http.StatusOK, fmt.Sprintf("I am by birth a Genevese (%d)", i))
- wg.Done()
- }(i)
- }
- wg.Wait()
- // Fetch metrics
- resp, err := http.Get("http://localhost:2999/metrics")
- if err != nil {
- t.Fatalf("failed to fetch metrics: %v", err)
- }
- defer resp.Body.Close()
- // Read and parse metrics
- metrics := new(bytes.Buffer)
- _, err = metrics.ReadFrom(resp.Body)
- if err != nil {
- t.Fatalf("failed to read metrics: %v", err)
- }
- cpus := fmt.Sprintf("%d", frankenphp.MaxThreads)
- // Check metrics
- expectedMetrics := `
- # HELP frankenphp_total_threads Total number of PHP threads
- # TYPE frankenphp_total_threads counter
- frankenphp_total_threads ` + cpus + `
- # HELP frankenphp_busy_threads Number of busy PHP threads
- # TYPE frankenphp_busy_threads gauge
- frankenphp_busy_threads 2
- # HELP frankenphp_testdata_index_php_busy_workers Number of busy PHP workers for this worker
- # TYPE frankenphp_testdata_index_php_busy_workers gauge
- frankenphp_testdata_index_php_busy_workers 0
- # HELP frankenphp_testdata_index_php_total_workers Total number of PHP workers for this worker
- # TYPE frankenphp_testdata_index_php_total_workers gauge
- frankenphp_testdata_index_php_total_workers 2
- # HELP frankenphp_testdata_index_php_worker_request_count
- # TYPE frankenphp_testdata_index_php_worker_request_count counter
- frankenphp_testdata_index_php_worker_request_count 10
- # HELP frankenphp_testdata_index_php_ready_workers Running workers that have successfully called frankenphp_handle_request at least once
- # TYPE frankenphp_testdata_index_php_ready_workers gauge
- frankenphp_testdata_index_php_ready_workers 2
- # HELP frankenphp_testdata_index_php_worker_crashes Number of PHP worker crashes for this worker
- # TYPE frankenphp_testdata_index_php_worker_crashes counter
- frankenphp_testdata_index_php_worker_crashes 0
- # HELP frankenphp_testdata_index_php_worker_restarts Number of PHP worker restarts for this worker
- # TYPE frankenphp_testdata_index_php_worker_restarts counter
- frankenphp_testdata_index_php_worker_restarts 0
- `
- require.NoError(t,
- testutil.GatherAndCompare(
- prometheus.DefaultGatherer,
- strings.NewReader(expectedMetrics),
- "frankenphp_total_threads",
- "frankenphp_busy_threads",
- "frankenphp_testdata_index_php_busy_workers",
- "frankenphp_testdata_index_php_total_workers",
- "frankenphp_testdata_index_php_worker_request_count",
- "frankenphp_testdata_index_php_worker_crashes",
- "frankenphp_testdata_index_php_worker_restarts",
- "frankenphp_testdata_index_php_ready_workers",
- ))
- }
- func TestAutoWorkerConfig(t *testing.T) {
- var wg sync.WaitGroup
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- https_port 9443
- frankenphp {
- worker ../testdata/index.php
- }
- }
- localhost:`+testPort+` {
- route {
- php {
- root ../testdata
- }
- }
- }
- `, "caddyfile")
- // Make some requests
- for i := 0; i < 10; i++ {
- wg.Add(1)
- go func(i int) {
- tester.AssertGetResponse(fmt.Sprintf("http://localhost:"+testPort+"/index.php?i=%d", i), http.StatusOK, fmt.Sprintf("I am by birth a Genevese (%d)", i))
- wg.Done()
- }(i)
- }
- wg.Wait()
- // Fetch metrics
- resp, err := http.Get("http://localhost:2999/metrics")
- if err != nil {
- t.Fatalf("failed to fetch metrics: %v", err)
- }
- defer resp.Body.Close()
- // Read and parse metrics
- metrics := new(bytes.Buffer)
- _, err = metrics.ReadFrom(resp.Body)
- if err != nil {
- t.Fatalf("failed to read metrics: %v", err)
- }
- cpus := fmt.Sprintf("%d", frankenphp.MaxThreads)
- workers := fmt.Sprintf("%d", frankenphp.MaxThreads-1)
- // Check metrics
- expectedMetrics := `
- # HELP frankenphp_total_threads Total number of PHP threads
- # TYPE frankenphp_total_threads counter
- frankenphp_total_threads ` + cpus + `
- # HELP frankenphp_busy_threads Number of busy PHP threads
- # TYPE frankenphp_busy_threads gauge
- frankenphp_busy_threads ` + workers + `
- # HELP frankenphp_testdata_index_php_busy_workers Number of busy PHP workers for this worker
- # TYPE frankenphp_testdata_index_php_busy_workers gauge
- frankenphp_testdata_index_php_busy_workers 0
- # HELP frankenphp_testdata_index_php_total_workers Total number of PHP workers for this worker
- # TYPE frankenphp_testdata_index_php_total_workers gauge
- frankenphp_testdata_index_php_total_workers ` + workers + `
- # HELP frankenphp_testdata_index_php_worker_request_count
- # TYPE frankenphp_testdata_index_php_worker_request_count counter
- frankenphp_testdata_index_php_worker_request_count 10
- # HELP frankenphp_testdata_index_php_ready_workers Running workers that have successfully called frankenphp_handle_request at least once
- # TYPE frankenphp_testdata_index_php_ready_workers gauge
- frankenphp_testdata_index_php_ready_workers ` + workers + `
- # HELP frankenphp_testdata_index_php_worker_crashes Number of PHP worker crashes for this worker
- # TYPE frankenphp_testdata_index_php_worker_crashes counter
- frankenphp_testdata_index_php_worker_crashes 0
- # HELP frankenphp_testdata_index_php_worker_restarts Number of PHP worker restarts for this worker
- # TYPE frankenphp_testdata_index_php_worker_restarts counter
- frankenphp_testdata_index_php_worker_restarts 0
- `
- require.NoError(t,
- testutil.GatherAndCompare(
- prometheus.DefaultGatherer,
- strings.NewReader(expectedMetrics),
- "frankenphp_total_threads",
- "frankenphp_busy_threads",
- "frankenphp_testdata_index_php_busy_workers",
- "frankenphp_testdata_index_php_total_workers",
- "frankenphp_testdata_index_php_worker_request_count",
- "frankenphp_testdata_index_php_worker_crashes",
- "frankenphp_testdata_index_php_worker_restarts",
- "frankenphp_testdata_index_php_ready_workers",
- ))
- }
- func TestAllDefinedServerVars(t *testing.T) {
- documentRoot, _ := filepath.Abs("../testdata/")
- expectedBodyFile, _ := os.ReadFile("../testdata/server-all-vars-ordered.txt")
- expectedBody := string(expectedBodyFile)
- expectedBody = strings.ReplaceAll(expectedBody, "{documentRoot}", documentRoot)
- expectedBody = strings.ReplaceAll(expectedBody, "\r\n", "\n")
- expectedBody = strings.ReplaceAll(expectedBody, "{testPort}", testPort)
- tester := caddytest.NewTester(t)
- tester.InitServer(`
- {
- skip_install_trust
- admin localhost:2999
- http_port `+testPort+`
- frankenphp
- }
- localhost:`+testPort+` {
- route {
- root ../testdata
- # rewrite to test that the original path is passed as $REQUEST_URI
- rewrite /server-all-vars-ordered.php/path
- php
- }
- }
- `, "caddyfile")
- tester.AssertPostResponseBody(
- "http://user@localhost:"+testPort+"/original-path?specialChars=%3E\\x00%00</>",
- []string{
- "Content-Type: application/x-www-form-urlencoded",
- "Content-Length: 14", // maliciously set to 14
- "Special-Chars: <%00>",
- "Host: Malicious Host",
- "X-Empty-Header:",
- },
- bytes.NewBufferString("foo=bar"),
- http.StatusOK,
- expectedBody,
- )
- }
|