summaryrefslogtreecommitdiff
path: root/service/gzip
diff options
context:
space:
mode:
Diffstat (limited to 'service/gzip')
-rw-r--r--service/gzip/config.go23
-rw-r--r--service/gzip/config_test.go43
-rw-r--r--service/gzip/service.go32
-rw-r--r--service/gzip/service_test.go123
4 files changed, 221 insertions, 0 deletions
diff --git a/service/gzip/config.go b/service/gzip/config.go
new file mode 100644
index 00000000..7da48a67
--- /dev/null
+++ b/service/gzip/config.go
@@ -0,0 +1,23 @@
+package gzip
+
+import (
+ "github.com/spiral/roadrunner/service"
+)
+
+// Config describes file location and controls access to them.
+type Config struct {
+ // Dir contains name of directory to control access to.
+ Enable bool
+}
+
+// Hydrate must populate Config values using given Config source. Must return error if Config is not valid.
+func (c *Config) Hydrate(cfg service.Config) error {
+ return cfg.Unmarshal(c)
+}
+
+// InitDefaults sets missing values to their default values.
+func (c *Config) InitDefaults() error {
+ c.Enable = true
+
+ return nil
+}
diff --git a/service/gzip/config_test.go b/service/gzip/config_test.go
new file mode 100644
index 00000000..92bb1cb5
--- /dev/null
+++ b/service/gzip/config_test.go
@@ -0,0 +1,43 @@
+package gzip
+
+import (
+ "encoding/json"
+ "github.com/spiral/roadrunner/service"
+ "github.com/stretchr/testify/assert"
+ "testing"
+)
+
+type mockCfg struct{ cfg string }
+
+func (cfg *mockCfg) Get(name string) service.Config { return nil }
+func (cfg *mockCfg) Unmarshal(out interface{}) error { return json.Unmarshal([]byte(cfg.cfg), out) }
+
+func Test_Config_Hydrate(t *testing.T) {
+ cfg := &mockCfg{`{"enable": true}`}
+ c := &Config{}
+
+ assert.NoError(t, c.Hydrate(cfg))
+}
+
+func Test_Config_Hydrate_Error(t *testing.T) {
+ cfg := &mockCfg{`{"enable": "invalid"}`}
+ c := &Config{}
+
+ assert.Error(t, c.Hydrate(cfg))
+}
+
+func Test_Config_Hydrate_Error2(t *testing.T) {
+ cfg := &mockCfg{`{"enable": 1}`}
+ c := &Config{}
+
+ assert.Error(t, c.Hydrate(cfg))
+}
+
+func Test_Config_Defaults(t *testing.T) {
+ c := &Config{}
+ err := c.InitDefaults()
+ if err != nil {
+ t.Errorf("error during the InitDefaults: error %v", err)
+ }
+ assert.Equal(t, true, c.Enable)
+}
diff --git a/service/gzip/service.go b/service/gzip/service.go
new file mode 100644
index 00000000..4c6320ad
--- /dev/null
+++ b/service/gzip/service.go
@@ -0,0 +1,32 @@
+package gzip
+
+import (
+ rrhttp "github.com/spiral/roadrunner/service/http"
+ "github.com/NYTimes/gziphandler"
+ "net/http"
+)
+
+// ID contains default service name.
+const ID = "gzip"
+
+type Service struct {
+ cfg *Config
+}
+
+func (s *Service) Init(cfg *Config, r *rrhttp.Service) (bool, error) {
+ s.cfg = cfg
+
+ if s.cfg.Enable == false {
+ return false, nil
+ }
+
+ r.AddMiddleware(s.middleware)
+
+ return true, nil
+}
+
+func (s *Service) middleware(f http.HandlerFunc) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gziphandler.GzipHandler(f).ServeHTTP(w, r)
+ }
+}
diff --git a/service/gzip/service_test.go b/service/gzip/service_test.go
new file mode 100644
index 00000000..6f07a1f9
--- /dev/null
+++ b/service/gzip/service_test.go
@@ -0,0 +1,123 @@
+package gzip
+
+import (
+ "encoding/json"
+ "github.com/sirupsen/logrus"
+ "github.com/sirupsen/logrus/hooks/test"
+ "github.com/spiral/roadrunner/service"
+ rrhttp "github.com/spiral/roadrunner/service/http"
+ "github.com/stretchr/testify/assert"
+ "io/ioutil"
+ "net/http"
+ "os"
+ "testing"
+ "time"
+)
+
+type testCfg struct {
+ gzip string
+ httpCfg string
+ static string
+ target string
+}
+
+func (cfg *testCfg) Get(name string) service.Config {
+ if name == rrhttp.ID {
+ return &testCfg{target: cfg.httpCfg}
+ }
+
+ if name == ID {
+ return &testCfg{target: cfg.gzip}
+ }
+ return nil
+}
+func (cfg *testCfg) Unmarshal(out interface{}) error {
+ return json.Unmarshal([]byte(cfg.target), out)
+}
+
+func get(url string) (string, *http.Response, error) {
+ r, err := http.Get(url)
+ if err != nil {
+ return "", nil, err
+ }
+
+ b, err := ioutil.ReadAll(r.Body)
+ if err != nil {
+ return "", nil, err
+ }
+
+ err = r.Body.Close()
+ if err != nil {
+ return "", nil, err
+ }
+
+ return string(b), r, err
+}
+
+func Test_Disabled(t *testing.T) {
+ logger, _ := test.NewNullLogger()
+ logger.SetLevel(logrus.DebugLevel)
+
+ c := service.NewContainer(logger)
+ c.Register(ID, &Service{})
+
+ assert.NoError(t, c.Init(&testCfg{
+ gzip: `{"enable":false}`,
+ }))
+
+ s, st := c.Get(ID)
+ assert.NotNil(t, s)
+ assert.Equal(t, service.StatusInactive, st)
+}
+// func Test_Files(t *testing.T) {
+// logger, _ := test.NewNullLogger()
+// logger.SetLevel(logrus.DebugLevel)
+
+// c := service.NewContainer(logger)
+// c.Register(rrhttp.ID, &rrhttp.Service{})
+// c.Register(ID, &Service{})
+
+// assert.NoError(t, c.Init(&testCfg{
+// gzip: `{"enable":true}`,
+// static: `{"enable":true, "dir":"../../tests", "forbid":[]}`,
+// httpCfg: `{
+// "enable": true,
+// "address": ":6029",
+// "maxRequestSize": 1024,
+// "uploads": {
+// "dir": ` + tmpDir() + `,
+// "forbid": []
+// },
+// "workers":{
+// "command": "php ../../tests/http/client.php pid pipes",
+// "relay": "pipes",
+// "pool": {
+// "numWorkers": 1,
+// "allocateTimeout": 10000000,
+// "destroyTimeout": 10000000
+// }
+// }
+// }`}))
+
+// go func() {
+// err := c.Serve()
+// if err != nil {
+// t.Errorf("serve error: %v", err)
+// }
+// }()
+// time.Sleep(time.Millisecond * 1000)
+// defer c.Stop()
+
+// b, _, _ := get("http://localhost:6029/sample.txt")
+// assert.Equal(t, "sample", b)
+// //header should not contain content-encoding:gzip because content-length < gziphandler.DefaultMinSize
+// // b, _, _ := get("http://localhost:6029/gzip-large-file.txt")
+// //header should contain content-encoding:gzip because content-length > gziphandler.DefaultMinSize
+// }
+
+func tmpDir() string {
+ p := os.TempDir()
+ r, _ := json.Marshal(p)
+
+ return string(r)
+}