123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- package features
- import (
- "fmt"
- "strconv"
- "sync"
- "time"
- "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/mock/server/streamserver"
- "github.com/fatedier/frp/test/e2e/pkg/request"
- )
- var _ = ginkgo.Describe("[Feature: Group]", func() {
- f := framework.NewDefaultFramework()
- newHTTPServer := func(port int, respContent string) *httpserver.Server {
- return httpserver.New(
- httpserver.WithBindPort(port),
- httpserver.WithHandler(framework.SpecifiedHTTPBodyHandler([]byte(respContent))),
- )
- }
- validateFooBarResponse := func(resp *request.Response) bool {
- if string(resp.Content) == "foo" || string(resp.Content) == "bar" {
- return true
- }
- return false
- }
- doFooBarHTTPRequest := func(vhostPort int, host string) []string {
- results := []string{}
- var wait sync.WaitGroup
- var mu sync.Mutex
- expectFn := func() {
- framework.NewRequestExpect(f).Port(vhostPort).
- RequestModify(func(r *request.Request) {
- r.HTTP().HTTPHost(host)
- }).
- Ensure(validateFooBarResponse, func(resp *request.Response) bool {
- mu.Lock()
- defer mu.Unlock()
- results = append(results, string(resp.Content))
- return true
- })
- }
- for i := 0; i < 10; i++ {
- wait.Add(1)
- go func() {
- defer wait.Done()
- expectFn()
- }()
- }
- wait.Wait()
- return results
- }
- ginkgo.Describe("Load Balancing", func() {
- ginkgo.It("TCP", func() {
- serverConf := consts.DefaultServerConfig
- clientConf := consts.DefaultClientConfig
- fooPort := f.AllocPort()
- fooServer := streamserver.New(streamserver.TCP, streamserver.WithBindPort(fooPort), streamserver.WithRespContent([]byte("foo")))
- f.RunServer("", fooServer)
- barPort := f.AllocPort()
- barServer := streamserver.New(streamserver.TCP, streamserver.WithBindPort(barPort), streamserver.WithRespContent([]byte("bar")))
- f.RunServer("", barServer)
- remotePort := f.AllocPort()
- clientConf += fmt.Sprintf(`
- [[proxies]]
- name = "foo"
- type = "tcp"
- localPort = %d
- remotePort = %d
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- [[proxies]]
- name = "bar"
- type = "tcp"
- localPort = %d
- remotePort = %d
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- `, fooPort, remotePort, barPort, remotePort)
- f.RunProcesses([]string{serverConf}, []string{clientConf})
- fooCount := 0
- barCount := 0
- for i := 0; i < 10; i++ {
- framework.NewRequestExpect(f).Explain("times " + strconv.Itoa(i)).Port(remotePort).Ensure(func(resp *request.Response) bool {
- switch string(resp.Content) {
- case "foo":
- fooCount++
- case "bar":
- barCount++
- default:
- return false
- }
- return true
- })
- }
- framework.ExpectTrue(fooCount > 1 && barCount > 1, "fooCount: %d, barCount: %d", fooCount, barCount)
- })
- })
- ginkgo.Describe("Health Check", func() {
- ginkgo.It("TCP", func() {
- serverConf := consts.DefaultServerConfig
- clientConf := consts.DefaultClientConfig
- fooPort := f.AllocPort()
- fooServer := streamserver.New(streamserver.TCP, streamserver.WithBindPort(fooPort), streamserver.WithRespContent([]byte("foo")))
- f.RunServer("", fooServer)
- barPort := f.AllocPort()
- barServer := streamserver.New(streamserver.TCP, streamserver.WithBindPort(barPort), streamserver.WithRespContent([]byte("bar")))
- f.RunServer("", barServer)
- remotePort := f.AllocPort()
- clientConf += fmt.Sprintf(`
- [[proxies]]
- name = "foo"
- type = "tcp"
- localPort = %d
- remotePort = %d
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- healthCheck.type = "tcp"
- healthCheck.intervalSeconds = 1
- [[proxies]]
- name = "bar"
- type = "tcp"
- localPort = %d
- remotePort = %d
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- healthCheck.type = "tcp"
- healthCheck.intervalSeconds = 1
- `, fooPort, remotePort, barPort, remotePort)
- f.RunProcesses([]string{serverConf}, []string{clientConf})
-
- results := []string{}
- for i := 0; i < 10; i++ {
- framework.NewRequestExpect(f).Port(remotePort).Ensure(validateFooBarResponse, func(resp *request.Response) bool {
- results = append(results, string(resp.Content))
- return true
- })
- }
- framework.ExpectContainElements(results, []string{"foo", "bar"})
-
- barServer.Close()
- time.Sleep(2 * time.Second)
- for i := 0; i < 10; i++ {
- framework.NewRequestExpect(f).Port(remotePort).ExpectResp([]byte("foo")).Ensure()
- }
-
- f.RunServer("", barServer)
- time.Sleep(2 * time.Second)
- results = []string{}
- for i := 0; i < 10; i++ {
- framework.NewRequestExpect(f).Port(remotePort).Ensure(validateFooBarResponse, func(resp *request.Response) bool {
- results = append(results, string(resp.Content))
- return true
- })
- }
- framework.ExpectContainElements(results, []string{"foo", "bar"})
- })
- ginkgo.It("HTTP", func() {
- vhostPort := f.AllocPort()
- serverConf := consts.DefaultServerConfig + fmt.Sprintf(`
- vhostHTTPPort = %d
- `, vhostPort)
- clientConf := consts.DefaultClientConfig
- fooPort := f.AllocPort()
- fooServer := newHTTPServer(fooPort, "foo")
- f.RunServer("", fooServer)
- barPort := f.AllocPort()
- barServer := newHTTPServer(barPort, "bar")
- f.RunServer("", barServer)
- clientConf += fmt.Sprintf(`
- [[proxies]]
- name = "foo"
- type = "http"
- localPort = %d
- customDomains = ["example.com"]
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- healthCheck.type = "http"
- healthCheck.intervalSeconds = 1
- healthCheck.path = "/healthz"
- [[proxies]]
- name = "bar"
- type = "http"
- localPort = %d
- customDomains = ["example.com"]
- loadBalancer.group = "test"
- loadBalancer.groupKey = "123"
- healthCheck.type = "http"
- healthCheck.intervalSeconds = 1
- healthCheck.path = "/healthz"
- `, fooPort, barPort)
- f.RunProcesses([]string{serverConf}, []string{clientConf})
-
- var contents []string
- framework.NewRequestExpect(f).Port(vhostPort).
- RequestModify(func(r *request.Request) {
- r.HTTP().HTTPHost("example.com")
- }).
- Ensure(func(resp *request.Response) bool {
- contents = append(contents, string(resp.Content))
- return true
- })
-
- framework.NewRequestExpect(f).Port(vhostPort).
- RequestModify(func(r *request.Request) {
- r.HTTP().HTTPHost("example.com")
- }).
- Ensure(func(resp *request.Response) bool {
- contents = append(contents, string(resp.Content))
- return true
- })
- framework.ExpectContainElements(contents, []string{"foo", "bar"})
-
- results := doFooBarHTTPRequest(vhostPort, "example.com")
- framework.ExpectContainElements(results, []string{"foo", "bar"})
-
- barServer.Close()
- time.Sleep(2 * time.Second)
- results = doFooBarHTTPRequest(vhostPort, "example.com")
- framework.ExpectContainElements(results, []string{"foo"})
- framework.ExpectNotContainElements(results, []string{"bar"})
-
- f.RunServer("", barServer)
- time.Sleep(2 * time.Second)
- results = doFooBarHTTPRequest(vhostPort, "example.com")
- framework.ExpectContainElements(results, []string{"foo", "bar"})
- })
- })
- })
|