mirror of
				https://github.com/fatedier/frp
				synced 2025-10-20 10:03:07 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			467 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			467 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package basic
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"strconv"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/gorilla/websocket"
 | |
| 	"github.com/onsi/ginkgo/v2"
 | |
| 
 | |
| 	"github.com/fatedier/frp/test/e2e/framework"
 | |
| 	"github.com/fatedier/frp/test/e2e/framework/consts"
 | |
| 	"github.com/fatedier/frp/test/e2e/mock/server/httpserver"
 | |
| 	"github.com/fatedier/frp/test/e2e/pkg/request"
 | |
| )
 | |
| 
 | |
| var _ = ginkgo.Describe("[Feature: HTTP]", func() {
 | |
| 	f := framework.NewDefaultFramework()
 | |
| 
 | |
| 	getDefaultServerConf := func(vhostHTTPPort int) string {
 | |
| 		conf := consts.DefaultServerConfig + `
 | |
| 		vhostHTTPPort = %d
 | |
| 		`
 | |
| 		return fmt.Sprintf(conf, vhostHTTPPort)
 | |
| 	}
 | |
| 	newHTTPServer := func(port int, respContent string) *httpserver.Server {
 | |
| 		return httpserver.New(
 | |
| 			httpserver.WithBindPort(port),
 | |
| 			httpserver.WithHandler(framework.SpecifiedHTTPBodyHandler([]byte(respContent))),
 | |
| 		)
 | |
| 	}
 | |
| 
 | |
| 	ginkgo.It("HTTP route by locations", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		fooPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(fooPort, "foo"))
 | |
| 
 | |
| 		barPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(barPort, "bar"))
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "foo"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			locations = ["/","/foo"]
 | |
| 
 | |
| 			[[proxies]]
 | |
| 			name = "bar"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			locations = ["/bar"]
 | |
| 			`, fooPort, barPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		tests := []struct {
 | |
| 			path       string
 | |
| 			expectResp string
 | |
| 			desc       string
 | |
| 		}{
 | |
| 			{path: "/foo", expectResp: "foo", desc: "foo path"},
 | |
| 			{path: "/bar", expectResp: "bar", desc: "bar path"},
 | |
| 			{path: "/other", expectResp: "foo", desc: "other path"},
 | |
| 		}
 | |
| 
 | |
| 		for _, test := range tests {
 | |
| 			framework.NewRequestExpect(f).Explain(test.desc).Port(vhostHTTPPort).
 | |
| 				RequestModify(func(r *request.Request) {
 | |
| 					r.HTTP().HTTPHost("normal.example.com").HTTPPath(test.path)
 | |
| 				}).
 | |
| 				ExpectResp([]byte(test.expectResp)).
 | |
| 				Ensure()
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("HTTP route by HTTP user", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		fooPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(fooPort, "foo"))
 | |
| 
 | |
| 		barPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(barPort, "bar"))
 | |
| 
 | |
| 		otherPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(otherPort, "other"))
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "foo"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			routeByHTTPUser = "user1"
 | |
| 
 | |
| 			[[proxies]]
 | |
| 			name = "bar"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			routeByHTTPUser = "user2"
 | |
| 
 | |
| 			[[proxies]]
 | |
| 			name = "catchAll"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			`, fooPort, barPort, otherPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		// user1
 | |
| 		framework.NewRequestExpect(f).Explain("user1").Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTPAuth("user1", "")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("foo")).
 | |
| 			Ensure()
 | |
| 
 | |
| 		// user2
 | |
| 		framework.NewRequestExpect(f).Explain("user2").Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTPAuth("user2", "")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("bar")).
 | |
| 			Ensure()
 | |
| 
 | |
| 		// other user
 | |
| 		framework.NewRequestExpect(f).Explain("other user").Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTPAuth("user3", "")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("other")).
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("HTTP Basic Auth", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = {{ .%s }}
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			httpUser = "test"
 | |
| 			httpPassword = "test"
 | |
| 			`, framework.HTTPSimpleServerPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		// not set auth header
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com")
 | |
| 			}).
 | |
| 			Ensure(framework.ExpectResponseCode(401))
 | |
| 
 | |
| 		// set incorrect auth header
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTPAuth("test", "invalid")
 | |
| 			}).
 | |
| 			Ensure(framework.ExpectResponseCode(401))
 | |
| 
 | |
| 		// set correct auth header
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTPAuth("test", "test")
 | |
| 			}).
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Wildcard domain", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = {{ .%s }}
 | |
| 			customDomains = ["*.example.com"]
 | |
| 			`, framework.HTTPSimpleServerPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		// not match host
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("not-match.test.com")
 | |
| 			}).
 | |
| 			Ensure(framework.ExpectResponseCode(404))
 | |
| 
 | |
| 		// test.example.com match *.example.com
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("test.example.com")
 | |
| 			}).
 | |
| 			Ensure()
 | |
| 
 | |
| 		// sub.test.example.com match *.example.com
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("sub.test.example.com")
 | |
| 			}).
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Subdomain", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 		serverConf += `
 | |
| 		subdomainHost = "example.com"
 | |
| 		`
 | |
| 
 | |
| 		fooPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(fooPort, "foo"))
 | |
| 
 | |
| 		barPort := f.AllocPort()
 | |
| 		f.RunServer("", newHTTPServer(barPort, "bar"))
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "foo"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			subdomain = "foo"
 | |
| 
 | |
| 			[[proxies]]
 | |
| 			name = "bar"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			subdomain = "bar"
 | |
| 			`, fooPort, barPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		// foo
 | |
| 		framework.NewRequestExpect(f).Explain("foo subdomain").Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("foo.example.com")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("foo")).
 | |
| 			Ensure()
 | |
| 
 | |
| 		// bar
 | |
| 		framework.NewRequestExpect(f).Explain("bar subdomain").Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("bar.example.com")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("bar")).
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Modify request headers", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		localPort := f.AllocPort()
 | |
| 		localServer := httpserver.New(
 | |
| 			httpserver.WithBindPort(localPort),
 | |
| 			httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
 | |
| 				_, _ = w.Write([]byte(req.Header.Get("X-From-Where")))
 | |
| 			})),
 | |
| 		)
 | |
| 		f.RunServer("", localServer)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			requestHeaders.set.x-from-where = "frp"
 | |
| 			`, localPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("frp")). // local http server will write this X-From-Where header to response body
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Modify response headers", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		localPort := f.AllocPort()
 | |
| 		localServer := httpserver.New(
 | |
| 			httpserver.WithBindPort(localPort),
 | |
| 			httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
 | |
| 				w.WriteHeader(200)
 | |
| 			})),
 | |
| 		)
 | |
| 		f.RunServer("", localServer)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			responseHeaders.set.x-from-where = "frp"
 | |
| 			`, localPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com")
 | |
| 			}).
 | |
| 			Ensure(func(res *request.Response) bool {
 | |
| 				return res.Header.Get("X-From-Where") == "frp"
 | |
| 			})
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Host Header Rewrite", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		localPort := f.AllocPort()
 | |
| 		localServer := httpserver.New(
 | |
| 			httpserver.WithBindPort(localPort),
 | |
| 			httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
 | |
| 				_, _ = w.Write([]byte(req.Host))
 | |
| 			})),
 | |
| 		)
 | |
| 		f.RunServer("", localServer)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			hostHeaderRewrite = "rewrite.example.com"
 | |
| 			`, localPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com")
 | |
| 			}).
 | |
| 			ExpectResp([]byte("rewrite.example.com")). // local http server will write host header to response body
 | |
| 			Ensure()
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("Websocket protocol", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 
 | |
| 		upgrader := websocket.Upgrader{}
 | |
| 
 | |
| 		localPort := f.AllocPort()
 | |
| 		localServer := httpserver.New(
 | |
| 			httpserver.WithBindPort(localPort),
 | |
| 			httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
 | |
| 				c, err := upgrader.Upgrade(w, req, nil)
 | |
| 				if err != nil {
 | |
| 					return
 | |
| 				}
 | |
| 				defer c.Close()
 | |
| 				for {
 | |
| 					mt, message, err := c.ReadMessage()
 | |
| 					if err != nil {
 | |
| 						break
 | |
| 					}
 | |
| 					err = c.WriteMessage(mt, message)
 | |
| 					if err != nil {
 | |
| 						break
 | |
| 					}
 | |
| 				}
 | |
| 			})),
 | |
| 		)
 | |
| 
 | |
| 		f.RunServer("", localServer)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["127.0.0.1"]
 | |
| 			`, localPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		u := url.URL{Scheme: "ws", Host: "127.0.0.1:" + strconv.Itoa(vhostHTTPPort)}
 | |
| 		c, _, err := websocket.DefaultDialer.Dial(u.String(), nil)
 | |
| 		framework.ExpectNoError(err)
 | |
| 
 | |
| 		err = c.WriteMessage(websocket.TextMessage, []byte(consts.TestString))
 | |
| 		framework.ExpectNoError(err)
 | |
| 
 | |
| 		_, msg, err := c.ReadMessage()
 | |
| 		framework.ExpectNoError(err)
 | |
| 		framework.ExpectEqualValues(consts.TestString, string(msg))
 | |
| 	})
 | |
| 
 | |
| 	ginkgo.It("vhostHTTPTimeout", func() {
 | |
| 		vhostHTTPPort := f.AllocPort()
 | |
| 		serverConf := getDefaultServerConf(vhostHTTPPort)
 | |
| 		serverConf += `
 | |
| 		vhostHTTPTimeout = 2
 | |
| 		`
 | |
| 
 | |
| 		delayDuration := 0 * time.Second
 | |
| 		localPort := f.AllocPort()
 | |
| 		localServer := httpserver.New(
 | |
| 			httpserver.WithBindPort(localPort),
 | |
| 			httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
 | |
| 				time.Sleep(delayDuration)
 | |
| 				_, _ = w.Write([]byte(req.Host))
 | |
| 			})),
 | |
| 		)
 | |
| 		f.RunServer("", localServer)
 | |
| 
 | |
| 		clientConf := consts.DefaultClientConfig
 | |
| 		clientConf += fmt.Sprintf(`
 | |
| 			[[proxies]]
 | |
| 			name = "test"
 | |
| 			type = "http"
 | |
| 			localPort = %d
 | |
| 			customDomains = ["normal.example.com"]
 | |
| 			`, localPort)
 | |
| 
 | |
| 		f.RunProcesses([]string{serverConf}, []string{clientConf})
 | |
| 
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTP().Timeout(time.Second)
 | |
| 			}).
 | |
| 			ExpectResp([]byte("normal.example.com")).
 | |
| 			Ensure()
 | |
| 
 | |
| 		delayDuration = 3 * time.Second
 | |
| 		framework.NewRequestExpect(f).Port(vhostHTTPPort).
 | |
| 			RequestModify(func(r *request.Request) {
 | |
| 				r.HTTP().HTTPHost("normal.example.com").HTTP().Timeout(5 * time.Second)
 | |
| 			}).
 | |
| 			Ensure(framework.ExpectResponseCode(504))
 | |
| 	})
 | |
| })
 | 
