Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add logging and recording of requests #170

Closed
wants to merge 21 commits into from
Closed
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
c46de61
Add logging of requests to file and/or to stdout
tateexon Jun 13, 2024
2b9f1e1
Merge remote-tracking branch 'origin/main' into add_verbose_request_dump
tateexon Jun 14, 2024
e8dff5f
merge conflict fixes
tateexon Jun 14, 2024
3c18bdc
Handle error in config for io.ReadAll
tateexon Jun 22, 2024
72b3f4e
- Sync file wrties in the RequestWriter
tateexon Jun 22, 2024
acaa33c
Wrap request gorilla/handlers#CustomLoggingHandler to log requests
tateexon Jun 24, 2024
cd93b1f
clean up docker build warning
tateexon Jun 24, 2024
b7bafe8
use -v and update readme for version
tateexon Jun 24, 2024
ac2569a
Only set the body as binary when the Content-Type Header is a binary …
tateexon Jun 25, 2024
8c2e34c
Make binaryContentTypes global so it isn't building it every time.
tateexon Jun 25, 2024
fcc3cb7
replace -v --verbose with -l --log-level with 3 settings, 0 default, …
tateexon Jun 25, 2024
7c5b2d5
Remove some changes around build and git ignore for this PR
tateexon Jul 22, 2024
dbd4701
Remove deprecation fix from this PR
tateexon Jul 22, 2024
b0dbb5f
Add a serverconfig package
tateexon Jul 23, 2024
0a345b0
Better handle the log file close defer now that everything is an io.W…
tateexon Jul 23, 2024
6459695
reverse the order so we default to base64 encoded body
tateexon Jul 23, 2024
fdf0d40
remove log that added no value
tateexon Jul 23, 2024
cf5ca23
Remove gorilla code copied in
tateexon Jul 24, 2024
5cdd797
Add custom logger as json output
tateexon Jul 24, 2024
3dacb9d
Add max to the body that can be logged out
tateexon Jul 24, 2024
78a8ae0
Use a limited reader to only read up to the allowed maximum bytes fro…
tateexon Jul 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ bin
*.out
coverage.txt

.vscode
tateexon marked this conversation as resolved.
Show resolved Hide resolved

imposters

!internal/server/http/test/testdata/imposters/
6 changes: 4 additions & 2 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -1,15 +1,17 @@
FROM golang:1.21-alpine AS build

LABEL MAINTAINER = 'Friends of Go ([email protected])'
LABEL MAINTAINER='Friends of Go ([email protected])'

ARG TARGETOS=linux
ARG TARGETARCH=amd64
ARG TAG=""

RUN apk add --update git
RUN apk add ca-certificates
WORKDIR /go/src/github.com/friendsofgo/killgrave
COPY . .
RUN go mod tidy && TAG=$(git describe --tags --abbrev=0) \
RUN go mod tidy \
tateexon marked this conversation as resolved.
Show resolved Hide resolved
&& if [ -z "$TAG" ]; then TAG=$(git describe --tags --abbrev=0); fi \
&& LDFLAGS=$(echo "-s -w -X github.com/friendsofgo/killgrave/internal/app/cmd._version="docker-$TAG) \
&& CGO_ENABLED=0 GOOS="${TARGETOS}" GOARCH="${TARGETARCH}" go build -a -installsuffix cgo -o /go/bin/killgrave -ldflags "$LDFLAGS" cmd/killgrave/main.go

Expand Down
10 changes: 9 additions & 1 deletion Makefile
Original file line number Diff line number Diff line change
@@ -1,3 +1,11 @@
.PHONY: build
build:
go build -ldflags "-s -w -X 'github.com/friendsofgo/killgrave/internal/app/cmd._version=`git rev-parse --abbrev-ref HEAD`-`git rev-parse --short HEAD`'" -o bin/killgrave cmd/killgrave/main.go
go build -ldflags "-s -w -X 'github.com/friendsofgo/killgrave/internal/app/cmd._version=`git rev-parse --abbrev-ref HEAD`-`git rev-parse --short HEAD`'" -o bin/killgrave cmd/killgrave/main.go

.PHONY: build-docker
build-docker:
docker build --build-arg TAG=$(TAG) -t killgrave:$(TAG) .

.PHONY: test
test:
go test -v -vet=off -race ./...
4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,8 @@ Flags:
-s, --secure Run mock server using TLS (https)
-v, --version Version of Killgrave
-w, --watcher File watcher will reload the server on each file change
-l, --log-level The higher the log level the more detail you get (default 0, 1 adds requests, 2 adds request body)
-d, --dump-requests-path Print requests out to specified file
```

### Using Killgrave by config file
Expand All @@ -182,6 +184,8 @@ cors:
allow_credentials: true
watcher: true
secure: true
log_level: 1
dump_requests_path: "/abc/def.log
```

As you can see, you can configure all the options in a very easy way. For the above example, the file tree looks as follows, with the current working directory being `mymock`.
Expand Down
58 changes: 40 additions & 18 deletions internal/app/cmd/cmd.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@ import (

killgrave "github.com/friendsofgo/killgrave/internal"
server "github.com/friendsofgo/killgrave/internal/server/http"
"github.com/gorilla/handlers"
"github.com/gorilla/mux"
"github.com/radovskyb/watcher"
"github.com/spf13/cobra"
Expand All @@ -26,23 +25,29 @@ const (
_defaultPort = 3000
_defaultProxyMode = killgrave.ProxyNone
_defaultStrictSlash = true

_impostersFlag = "imposters"
_configFlag = "config"
_hostFlag = "host"
_portFlag = "port"
_watcherFlag = "watcher"
_secureFlag = "secure"
_proxyModeFlag = "proxy-mode"
_proxyURLFlag = "proxy-url"
_defaultLogLevel = 0

_impostersFlag = "imposters"
_configFlag = "config"
_hostFlag = "host"
_portFlag = "port"
_watcherFlag = "watcher"
_secureFlag = "secure"
_proxyModeFlag = "proxy-mode"
_proxyURLFlag = "proxy-url"
_logLevel = "log-level"
tateexon marked this conversation as resolved.
Show resolved Hide resolved
_dumpRequestsPathFlag = "dump-requests-path"
)

var (
errGetDataFromImpostersFlag = errors.New("error trying to get data from imposters flag")
errGetDataFromHostFlag = errors.New("error trying to get data from host flag")
errGetDataFromPortFlag = errors.New("error trying to get data from port flag")
errGetDataFromSecureFlag = errors.New("error trying to get data from secure flag")
errMandatoryURL = errors.New("the field proxy-url is mandatory if you selected a proxy mode")
errGetDataFromImpostersFlag = errors.New("error trying to get data from imposters flag")
errGetDataFromHostFlag = errors.New("error trying to get data from host flag")
errGetDataFromPortFlag = errors.New("error trying to get data from port flag")
errGetDataFromSecureFlag = errors.New("error trying to get data from secure flag")
errGetDataFromLogLevelFlag = errors.New("error trying to get data from log-level flag")
errGetDataLogLevelInvalid = errors.New("error setting log-level, must be between 0 and 2 inclusive")
errGetDataFromDumpRequestsPathFlag = errors.New("error trying to get data from dump-requests-path flag")
errMandatoryURL = errors.New("the field proxy-url is mandatory if you selected a proxy mode")
)

// NewKillgraveCmd returns cobra.Command to run killgrave command
Expand Down Expand Up @@ -77,6 +82,8 @@ func NewKillgraveCmd() *cobra.Command {
rootCmd.Flags().BoolP(_secureFlag, "s", false, "Run mock server using TLS (https)")
rootCmd.Flags().StringP(_proxyModeFlag, "m", _defaultProxyMode.String(), "Proxy mode, the options are all, missing or none")
rootCmd.Flags().StringP(_proxyURLFlag, "u", "", "The url where the proxy will redirect to")
rootCmd.Flags().IntP(_logLevel, "l", _defaultLogLevel, "Log level, the options are 0, 1, 2. Default is 0, 1 adds requests, 2 adds request body")
tateexon marked this conversation as resolved.
Show resolved Hide resolved
rootCmd.Flags().StringP(_dumpRequestsPathFlag, "d", "", "Path the requests will be dumped to")

rootCmd.SetVersionTemplate("Killgrave version: {{.Version}}\n")

Expand Down Expand Up @@ -115,8 +122,7 @@ func runServer(cfg killgrave.Config) server.Server {
httpAddr := fmt.Sprintf("%s:%d", cfg.Host, cfg.Port)

httpServer := http.Server{
Addr: httpAddr,
Handler: handlers.CORS(server.PrepareAccessControl(cfg.CORS)...)(router),
Addr: httpAddr,
}

proxyServer, err := server.NewProxy(cfg.Proxy.Url, cfg.Proxy.Mode)
Expand All @@ -135,6 +141,9 @@ func runServer(cfg killgrave.Config) server.Server {
proxyServer,
cfg.Secure,
imposterFs,
server.PrepareAccessControl(cfg.CORS),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same as above, I don't think this refactor is related (and/or strictly required) with the purpose of this pull request. Plus, I'm not even sure we really want it.

Copy link
Author

@tateexon tateexon Jul 23, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is needed so I can propagate cors here: https://github.com/friendsofgo/killgrave/pull/170/files#diff-09b422118d0773bb206a9e4cf4cbbaeeb017894e4007525bfbeb8d0163b3c824R102-R106
If we don't propagate it later then we can't get cors and the logging at the same time.

cfg.LogLevel,
cfg.DumpRequestsPath,
)
if err := s.Build(); err != nil {
log.Fatal(err)
Expand Down Expand Up @@ -185,7 +194,20 @@ func prepareConfig(cmd *cobra.Command) (killgrave.Config, error) {
return killgrave.Config{}, fmt.Errorf("%v: %w", err, errGetDataFromSecureFlag)
}

cfg, err := killgrave.NewConfig(impostersPath, host, port, secure)
logLevel, err := cmd.Flags().GetInt(_logLevel)
tateexon marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
return killgrave.Config{}, fmt.Errorf("%v: %w", err, errGetDataFromLogLevelFlag)
}
if logLevel < 0 || logLevel > 2 {
return killgrave.Config{}, fmt.Errorf("%v: %w", err, errGetDataLogLevelInvalid)
}

dumpRequestsPath, err := cmd.Flags().GetString(_dumpRequestsPathFlag)
if err != nil {
return killgrave.Config{}, fmt.Errorf("%v: %w", err, errGetDataFromDumpRequestsPathFlag)
}

cfg, err := killgrave.NewConfig(impostersPath, host, port, secure, logLevel, dumpRequestsPath)
if err != nil {
return killgrave.Config{}, err
}
Expand Down
35 changes: 21 additions & 14 deletions internal/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ package killgrave
import (
"errors"
"fmt"
"io/ioutil"
"io"
"os"
"path"

Expand All @@ -12,13 +12,15 @@ import (

// Config representation of config file yaml
type Config struct {
ImpostersPath string `yaml:"imposters_path"`
Port int `yaml:"port"`
Host string `yaml:"host"`
CORS ConfigCORS `yaml:"cors"`
Proxy ConfigProxy `yaml:"proxy"`
Secure bool `yaml:"secure"`
Watcher bool `yaml:"watcher"`
ImpostersPath string `yaml:"imposters_path"`
Port int `yaml:"port"`
Host string `yaml:"host"`
CORS ConfigCORS `yaml:"cors"`
Proxy ConfigProxy `yaml:"proxy"`
Secure bool `yaml:"secure"`
Watcher bool `yaml:"watcher"`
LogLevel int `yaml:"log_level"`
DumpRequestsPath string `yaml:"dump_requests_path"`
}

// ConfigCORS representation of section CORS of the yaml
Expand Down Expand Up @@ -111,7 +113,7 @@ func (cfg *Config) ConfigureProxy(proxyMode ProxyMode, proxyURL string) {
type ConfigOpt func(cfg *Config) error

// NewConfig initialize the config
func NewConfig(impostersPath, host string, port int, secure bool) (Config, error) {
func NewConfig(impostersPath, host string, port int, secure bool, logLevel int, dumpRequestsPath string) (Config, error) {
if impostersPath == "" {
return Config{}, errEmptyImpostersPath
}
Expand All @@ -125,10 +127,12 @@ func NewConfig(impostersPath, host string, port int, secure bool) (Config, error
}

cfg := Config{
ImpostersPath: impostersPath,
Host: host,
Port: port,
Secure: secure,
ImpostersPath: impostersPath,
Host: host,
Port: port,
Secure: secure,
LogLevel: logLevel,
DumpRequestsPath: dumpRequestsPath,
}

return cfg, nil
Expand All @@ -146,7 +150,10 @@ func NewConfigFromFile(cfgPath string) (Config, error) {
defer configFile.Close()

var cfg Config
bytes, _ := ioutil.ReadAll(configFile)
bytes, err := io.ReadAll(configFile)
tateexon marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
return Config{}, fmt.Errorf("%w: error while reading configFile file %s, using default configuration instead", err, cfgPath)
}
if err := yaml.Unmarshal(bytes, &cfg); err != nil {
return Config{}, fmt.Errorf("%w: error while unmarshalling configFile file %s, using default configuration instead", err, cfgPath)
}
Expand Down
4 changes: 2 additions & 2 deletions internal/config_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -203,7 +203,7 @@ func TestNewConfig(t *testing.T) {
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, err := NewConfig(tt.args.impostersPath, tt.args.host, tt.args.port, false)
got, err := NewConfig(tt.args.impostersPath, tt.args.host, tt.args.port, false, 0, "")
assert.Equal(t, tt.err, err)
assert.Equal(t, tt.want, got)
})
Expand All @@ -221,7 +221,7 @@ func TestConfig_ConfigureProxy(t *testing.T) {
},
}

got, err := NewConfig("imposters", "localhost", 80, false)
got, err := NewConfig("imposters", "localhost", 80, false, 0, "")
assert.NoError(t, err)

got.ConfigureProxy(ProxyAll, "https://friendsofgo.tech")
Expand Down
Loading