Skip to content

Commit 03b9f61

Browse files
2.0-RC8
1 parent 50fb441 commit 03b9f61

File tree

1,108 files changed

+140785
-128249
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

1,108 files changed

+140785
-128249
lines changed

.gitlab-ci.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ reuse-lint:
3939

4040
lint:
4141
extends: .on-non-tag
42-
image: golangci/golangci-lint:v1.42-alpine
42+
image: golangci/golangci-lint:v1.45-alpine
4343
stage: test
4444
allow_failure: true
4545
needs: []

.golangci.yml

+4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,10 @@ run:
1313
linters:
1414
enable-all: true
1515
disable:
16+
- forcetypeassert # TODO: this feels like a duplicated linter, not sure if we should disable it globally!
17+
- nilnil
18+
- ireturn
19+
- varnamelen
1620
- gomoddirectives # TODO: remove this as soon as as https://github.com/imdario/mergo/pull/198 and https://github.com/imdario/mergo/pull/197 are merged
1721
- promlinter
1822
- nolintlint

README.md

+12-1
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,7 @@ The following dependencies are needed during development:
9595

9696
- Bash
9797
- [Git](https://git-scm.com) on any recent (as in 2020) version.
98-
- [Go compiler](https://golang.org/) version 1.15 or newer.
98+
- [Go compiler](https://golang.org/) version 1.18 or newer.
9999
- [GCC](https://gcc.gnu.org/) version 9.3 or newer.
100100
- Libc development files. Both [glibc](https://www.gnu.org/software/libc/) and [musl](https://www.musl-libc.org/) have been successfully tested.
101101
- [GNU Make](https://www.gnu.org/software/make/manual/make.html) or compatible.
@@ -278,6 +278,7 @@ Here are all parameters you can set through environment variables, and their res
278278
- `LIGHTMETER_LOG_FORMAT=prepend-rfc3339` (`-log_format`)
279279
- `LIGHTMETER_LOG_FILE_PATTERNS=mail.log:mail.err:mail.warn:zimbra.log:maillog` (`-log_file_patterns`)
280280
- `LIGHTMETER_I_KNOW_WHAT_I_AM_DOING_NOT_USING_A_REVERSE_PROXY=true` (`-i_know_what_am_doing_not_using_a_reverse_proxy`)
281+
- `LIGHTMETER_NODE_TYPE="single"` (`-node_type`)
281282

282283
### Rotated files
283284

@@ -321,6 +322,16 @@ In case you are having an error similar to:
321322

322323
This means you should have a file `mail.log`, which means you should check your Postfix installation and ensure it's emitting logs properly.
323324

325+
### Support for multiple servers
326+
327+
There's some preliminary support for reading logs from multiple servers, when they are part of the `$mynetwork` postfix configuration.
328+
329+
More work is needed, but it works well enough for our internal use cases.
330+
331+
To enable it, set the environment variable `LIGHTMETER_NODE_TYPE=multi`, or use the command line argument `-node_type=multi`.
332+
333+
**TODO**: document how it works, caveats and so on. This somehow references #4.
334+
324335
### Reading from Logstash
325336

326337
**NOTE**: this is a very experimental feature, not well tested or supported. It can eat your logs!

VERSION.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
2.0.0-RC7
1+
2.0.0-RC8

api/dashboard.go

+95-8
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,10 @@ package api
66

77
import (
88
"context"
9+
"net/http"
10+
"strconv"
11+
"time"
12+
913
"gitlab.com/lightmeter/controlcenter/dashboard"
1014
"gitlab.com/lightmeter/controlcenter/httpauth/auth"
1115
"gitlab.com/lightmeter/controlcenter/httpmiddleware"
@@ -14,8 +18,6 @@ import (
1418
"gitlab.com/lightmeter/controlcenter/util/httputil"
1519
"gitlab.com/lightmeter/controlcenter/util/timeutil"
1620
"gitlab.com/lightmeter/controlcenter/version"
17-
"net/http"
18-
"time"
1921
)
2022

2123
type handler struct {
@@ -128,6 +130,80 @@ func (h deliveryStatusHandler) ServeHTTP(w http.ResponseWriter, r *http.Request)
128130
return servePairsFromTimeInterval(w, r, h.dashboard.DeliveryStatus, interval)
129131
}
130132

133+
type trafficBySenderOverTimeHandler struct {
134+
f func(context.Context, timeutil.TimeInterval, int) (dashboard.MailTrafficPerSenderOverTimeResult, error)
135+
}
136+
137+
// @Summary Messages sent by mailbox over time
138+
// @Param from query string true "Initial date in the format 1999-12-23"
139+
// @Param to query string true "Final date in the format 1999-12-23"
140+
// @Param granularity query integer 12 "Time granularity in hours"
141+
// @Produce json
142+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
143+
// @Failure 422 {string} string "desc"
144+
// @Router /api/v0/sentMailsByMailbox [get]
145+
146+
// @Summary Messages bounced by mailbox over time
147+
// @Param from query string true "Initial date in the format 1999-12-23"
148+
// @Param to query string true "Final date in the format 1999-12-23"
149+
// @Param granularity query integer 12 "Time granularity in hours"
150+
// @Produce json
151+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
152+
// @Failure 422 {string} string "desc"
153+
// @Router /api/v0/bouncedMailsByMailbox [get]
154+
155+
// @Summary Messages deferred by mailbox over time
156+
// @Param from query string true "Initial date in the format 1999-12-23"
157+
// @Param to query string true "Final date in the format 1999-12-23"
158+
// @Param granularity query integer 12 "Time granularity in hours"
159+
// @Produce json
160+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
161+
// @Failure 422 {string} string "desc"
162+
// @Router /api/v0/deferredMailsByMailbox [get]
163+
164+
// @Summary Messages expired by mailbox over time
165+
// @Param from query string true "Initial date in the format 1999-12-23"
166+
// @Param to query string true "Final date in the format 1999-12-23"
167+
// @Param granularity query integer 12 "Time granularity in hours"
168+
// @Produce json
169+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
170+
// @Failure 422 {string} string "desc"
171+
// @Router /api/v0/expiredMailsByMailbox [get]
172+
173+
// @Summary Messages received by mailbox over time
174+
// @Param from query string true "Initial date in the format 1999-12-23"
175+
// @Param to query string true "Final date in the format 1999-12-23"
176+
// @Param granularity query integer 12 "Time granularity in hours"
177+
// @Produce json
178+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
179+
// @Failure 422 {string} string "desc"
180+
// @Router /api/v0/receivedMailsByMailbox [get]
181+
182+
// @Summary Number of inbound replies
183+
// @Param from query string true "Initial date in the format 1999-12-23"
184+
// @Param to query string true "Final date in the format 1999-12-23"
185+
// @Param granularity query integer 12 "Time granularity in hours"
186+
// @Produce json
187+
// @Success 200 {object} dashboard.MailTrafficPerSenderOverTimeResult
188+
// @Failure 422 {string} string "desc"
189+
// @Router /api/v0/inboundRepliesByMailbox [get]
190+
191+
func (h trafficBySenderOverTimeHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) error {
192+
interval := httpmiddleware.GetIntervalFromContext(r)
193+
194+
granularity, err := strconv.Atoi(r.Form.Get("granularity"))
195+
if err != nil {
196+
return httperror.NewHTTPStatusCodeError(http.StatusUnprocessableEntity, err)
197+
}
198+
199+
result, err := h.f(r.Context(), interval, granularity)
200+
if err != nil {
201+
return err
202+
}
203+
204+
return httputil.WriteJson(w, result, http.StatusOK)
205+
}
206+
131207
type appVersionHandler struct{}
132208

133209
type appVersion struct {
@@ -144,14 +220,25 @@ func (appVersionHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) error
144220
return httputil.WriteJson(w, appVersion{Version: version.Version, Commit: version.Commit, TagOrBranch: version.TagOrBranch}, http.StatusOK)
145221
}
146222

147-
func HttpDashboard(auth *auth.Authenticator, mux *http.ServeMux, timezone *time.Location, dashboard dashboard.Dashboard) {
223+
func HttpDashboard(auth *auth.Authenticator, mux *http.ServeMux, timezone *time.Location, d dashboard.Dashboard) {
148224
authenticated := httpmiddleware.WithDefaultStack(auth, httpmiddleware.RequestWithInterval(timezone))
149225
unauthenticated := httpmiddleware.WithDefaultStackWithoutAuth()
150226

151-
mux.Handle("/api/v0/countByStatus", authenticated.WithEndpoint(countByStatusHandler{dashboard}))
152-
mux.Handle("/api/v0/topBusiestDomains", authenticated.WithEndpoint(topBusiestDomainsHandler{dashboard}))
153-
mux.Handle("/api/v0/topBouncedDomains", authenticated.WithEndpoint(topBouncedDomainsHandler{dashboard}))
154-
mux.Handle("/api/v0/topDeferredDomains", authenticated.WithEndpoint(topDeferredDomainsHandler{dashboard}))
155-
mux.Handle("/api/v0/deliveryStatus", authenticated.WithEndpoint(deliveryStatusHandler{dashboard}))
227+
for k, v := range map[string]func(context.Context, timeutil.TimeInterval, int) (dashboard.MailTrafficPerSenderOverTimeResult, error){
228+
"/api/v0/sentMailsByMailbox": d.SentMailsByMailbox,
229+
"/api/v0/bouncedMailsByMailbox": d.BouncedMailsByMailbox,
230+
"/api/v0/deferredMailsByMailbox": d.DeferredMailsByMailbox,
231+
"/api/v0/expiredMailsByMailbox": d.ExpiredMailsByMailbox,
232+
"/api/v0/receivedMailsByMailbox": d.ReceivedMailsByMailbox,
233+
"/api/v0/inboundRepliesByMailbox": d.InboundRepliesByMailbox,
234+
} {
235+
mux.Handle(k, authenticated.WithEndpoint(trafficBySenderOverTimeHandler{v}))
236+
}
237+
238+
mux.Handle("/api/v0/countByStatus", authenticated.WithEndpoint(countByStatusHandler{d}))
239+
mux.Handle("/api/v0/topBusiestDomains", authenticated.WithEndpoint(topBusiestDomainsHandler{d}))
240+
mux.Handle("/api/v0/topBouncedDomains", authenticated.WithEndpoint(topBouncedDomainsHandler{d}))
241+
mux.Handle("/api/v0/topDeferredDomains", authenticated.WithEndpoint(topDeferredDomainsHandler{d}))
242+
mux.Handle("/api/v0/deliveryStatus", authenticated.WithEndpoint(deliveryStatusHandler{d}))
156243
mux.Handle("/api/v0/appVersion", unauthenticated.WithEndpoint(appVersionHandler{}))
157244
}

api/detective.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ func requireDetectiveAuth(auth *httpauth.Authenticator, settingsReader metadata.
9191
isAuthenticated := isUserAuthenticated(auth, r)
9292

9393
settings := detectivesettings.Settings{}
94-
err := settingsReader.RetrieveJson(r.Context(), detectivesettings.SettingKey, &settings)
94+
err := settingsReader.RetrieveJson(r.Context(), detectivesettings.SettingsKey, &settings)
9595
if err != nil && !errors.Is(err, metadata.ErrNoSuchKey) {
9696
return httperror.NewHTTPStatusCodeError(http.StatusInternalServerError, errorutil.Wrap(err))
9797
}

api/detective_auth_test.go

+8-7
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,13 @@ package api
66

77
import (
88
"context"
9+
"net/http"
10+
"net/http/cookiejar"
11+
"net/http/httptest"
12+
"net/url"
13+
"testing"
14+
"time"
15+
916
"github.com/golang/mock/gomock"
1017
. "github.com/smartystreets/goconvey/convey"
1118
"gitlab.com/lightmeter/controlcenter/detective"
@@ -17,12 +24,6 @@ import (
1724
"gitlab.com/lightmeter/controlcenter/pkg/runner"
1825
detectivesettings "gitlab.com/lightmeter/controlcenter/settings/detective"
1926
"gitlab.com/lightmeter/controlcenter/util/testutil"
20-
"net/http"
21-
"net/http/cookiejar"
22-
"net/http/httptest"
23-
"net/url"
24-
"testing"
25-
"time"
2627
)
2728

2829
func init() {
@@ -49,7 +50,7 @@ func buildTestEnv(t *testing.T) (*httptest.Server, *mock_detective.MockDetective
4950

5051
detective := mock_detective.NewMockDetective(ctrl)
5152

52-
auth := auth.NewAuthenticator(registrar, dir)
53+
auth := auth.NewAuthenticator(registrar, dir, nil)
5354
mux := http.NewServeMux()
5455

5556
settingdDB, removeDB := testutil.TempDBConnectionMigrated(t, "master")

api/gen.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,4 @@
44

55
package api
66

7-
//go:generate go run -mod vendor github.com/swaggo/swag/cmd/swag init --generalInfo api.go
7+
//go:generate go run -mod vendor github.com/swaggo/swag/cmd/swag init --parseDependency --parseInternal --generalInfo api.go

api/rawlogs.go

+12-5
Original file line numberDiff line numberDiff line change
@@ -5,19 +5,21 @@
55
package api
66

77
import (
8+
"bufio"
89
"compress/gzip"
910
"errors"
1011
"fmt"
12+
"io"
13+
"net/http"
14+
"strconv"
15+
"time"
16+
1117
"gitlab.com/lightmeter/controlcenter/httpauth/auth"
1218
"gitlab.com/lightmeter/controlcenter/httpmiddleware"
1319
"gitlab.com/lightmeter/controlcenter/rawlogsdb"
1420
"gitlab.com/lightmeter/controlcenter/util/errorutil"
1521
"gitlab.com/lightmeter/controlcenter/util/httputil"
1622
"gitlab.com/lightmeter/controlcenter/util/timeutil"
17-
"io"
18-
"net/http"
19-
"strconv"
20-
"time"
2123
)
2224

2325
//nolint:structcheck,unused
@@ -123,7 +125,12 @@ func (h fetchRawLogLinesToWriterHandler) ServeHTTP(w http.ResponseWriter, r *htt
123125

124126
defer errorutil.UpdateErrorFromCall(releaseWriter, &err)
125127

126-
if err := h.accessor.FetchLogsInIntervalToWriter(r.Context(), interval, writer); err != nil {
128+
// use a 4KB buffer to improve write throughput, reducing the number of write() calls in the final socket
129+
bufferedWriter := bufio.NewWriterSize(writer, 4096)
130+
131+
defer errorutil.UpdateErrorFromCall(bufferedWriter.Flush, &err)
132+
133+
if err := h.accessor.FetchLogsInIntervalToWriter(r.Context(), interval, bufferedWriter); err != nil {
127134
return errorutil.Wrap(err)
128135
}
129136

app_test.go

+7-5
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,11 @@ package main
66

77
import (
88
"context"
9+
"os"
10+
"path"
11+
"testing"
12+
"time"
13+
914
. "github.com/smartystreets/goconvey/convey"
1015
"gitlab.com/lightmeter/controlcenter/config"
1116
"gitlab.com/lightmeter/controlcenter/detective/escalator"
@@ -16,10 +21,6 @@ import (
1621
"gitlab.com/lightmeter/controlcenter/util/testutil"
1722
"gitlab.com/lightmeter/controlcenter/util/timeutil"
1823
"gitlab.com/lightmeter/controlcenter/workspace"
19-
"os"
20-
"path"
21-
"testing"
22-
"time"
2324
)
2425

2526
func init() {
@@ -55,9 +56,10 @@ func TestMain(t *testing.T) {
5556
// Only import the logs in a workspace
5657
config := config.Config{
5758
WorkspaceDirectory: wsDir,
58-
DirToWatch: path.Join(logsDir, "logs_sample"),
59+
DirsToWatch: []string{path.Join(logsDir, "logs_sample")},
5960
ImportOnly: true,
6061
LogFormat: "default",
62+
MultiNodeType: "single",
6163
}
6264

6365
// first execution

0 commit comments

Comments
 (0)