2021-01-22 00:17:32 +03:00
|
|
|
|
package relay
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bytes"
|
|
|
|
|
"encoding/base64"
|
|
|
|
|
"fmt"
|
2021-01-30 13:17:37 +03:00
|
|
|
|
"github.com/emvi/logbuch"
|
2021-08-07 00:28:03 +03:00
|
|
|
|
"github.com/leandro-lugaresi/hub"
|
2021-01-22 00:17:32 +03:00
|
|
|
|
"github.com/muety/wakapi/config"
|
2021-03-26 15:10:10 +03:00
|
|
|
|
"github.com/muety/wakapi/middlewares"
|
2021-08-06 23:38:57 +03:00
|
|
|
|
"github.com/muety/wakapi/models"
|
2021-08-07 00:28:03 +03:00
|
|
|
|
"github.com/patrickmn/go-cache"
|
2021-01-22 00:17:32 +03:00
|
|
|
|
"io"
|
|
|
|
|
"io/ioutil"
|
|
|
|
|
"net/http"
|
|
|
|
|
"time"
|
|
|
|
|
)
|
|
|
|
|
|
2021-08-07 00:28:03 +03:00
|
|
|
|
const maxFailuresPerDay = 100
|
|
|
|
|
|
2021-01-22 00:17:32 +03:00
|
|
|
|
/* Middleware to conditionally relay heartbeats to Wakatime */
|
|
|
|
|
type WakatimeRelayMiddleware struct {
|
2021-08-07 00:28:03 +03:00
|
|
|
|
httpClient *http.Client
|
|
|
|
|
failureCache *cache.Cache
|
|
|
|
|
eventBus *hub.Hub
|
2021-01-22 00:17:32 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func NewWakatimeRelayMiddleware() *WakatimeRelayMiddleware {
|
|
|
|
|
return &WakatimeRelayMiddleware{
|
|
|
|
|
httpClient: &http.Client{
|
|
|
|
|
Timeout: 10 * time.Second,
|
|
|
|
|
},
|
2021-08-07 00:28:03 +03:00
|
|
|
|
failureCache: cache.New(24*time.Hour, 1*time.Hour),
|
|
|
|
|
eventBus: config.EventBus(),
|
2021-01-22 00:17:32 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (m *WakatimeRelayMiddleware) Handler(h http.Handler) http.Handler {
|
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
m.ServeHTTP(w, r, h.ServeHTTP)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (m *WakatimeRelayMiddleware) ServeHTTP(w http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
|
|
|
|
|
defer next(w, r)
|
|
|
|
|
|
|
|
|
|
if r.Method != http.MethodPost {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-26 15:10:10 +03:00
|
|
|
|
user := middlewares.GetPrincipal(r)
|
2021-01-22 00:17:32 +03:00
|
|
|
|
if user == nil || user.WakatimeApiKey == "" {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
body, _ := ioutil.ReadAll(r.Body)
|
|
|
|
|
r.Body.Close()
|
|
|
|
|
r.Body = ioutil.NopCloser(bytes.NewBuffer(body))
|
|
|
|
|
|
|
|
|
|
headers := http.Header{
|
|
|
|
|
"X-Machine-Name": r.Header.Values("X-Machine-Name"),
|
|
|
|
|
"Content-Type": r.Header.Values("Content-Type"),
|
|
|
|
|
"Accept": r.Header.Values("Accept"),
|
|
|
|
|
"User-Agent": r.Header.Values("User-Agent"),
|
|
|
|
|
"X-Origin": []string{
|
|
|
|
|
fmt.Sprintf("wakapi v%s", config.Get().Version),
|
|
|
|
|
},
|
|
|
|
|
"Authorization": []string{
|
|
|
|
|
fmt.Sprintf("Basic %s", base64.StdEncoding.EncodeToString([]byte(user.WakatimeApiKey))),
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
go m.send(
|
|
|
|
|
http.MethodPost,
|
2021-02-05 20:47:28 +03:00
|
|
|
|
config.WakatimeApiUrl+config.WakatimeApiHeartbeatsBulkUrl,
|
2021-01-22 00:17:32 +03:00
|
|
|
|
bytes.NewReader(body),
|
|
|
|
|
headers,
|
2021-08-06 23:38:57 +03:00
|
|
|
|
user,
|
2021-01-22 00:17:32 +03:00
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-06 23:38:57 +03:00
|
|
|
|
func (m *WakatimeRelayMiddleware) send(method, url string, body io.Reader, headers http.Header, forUser *models.User) {
|
2021-01-22 00:17:32 +03:00
|
|
|
|
request, err := http.NewRequest(method, url, body)
|
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Warn("error constructing relayed request – %v", err)
|
2021-01-24 23:39:35 +03:00
|
|
|
|
return
|
2021-01-22 00:17:32 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for k, v := range headers {
|
|
|
|
|
for _, h := range v {
|
|
|
|
|
request.Header.Set(k, h)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
response, err := m.httpClient.Do(request)
|
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Warn("error executing relayed request – %v", err)
|
2021-01-24 23:39:35 +03:00
|
|
|
|
return
|
2021-01-22 00:17:32 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if response.StatusCode < 200 || response.StatusCode >= 300 {
|
2021-08-06 23:38:57 +03:00
|
|
|
|
logbuch.Warn("failed to relay request for user %s, got status %d", forUser.ID, response.StatusCode)
|
2021-08-07 00:28:03 +03:00
|
|
|
|
|
|
|
|
|
// TODO: use leaky bucket instead of expiring cache?
|
|
|
|
|
if _, found := m.failureCache.Get(forUser.ID); !found {
|
|
|
|
|
m.failureCache.SetDefault(forUser.ID, 0)
|
|
|
|
|
}
|
|
|
|
|
if n, _ := m.failureCache.IncrementInt(forUser.ID, 1); n == maxFailuresPerDay {
|
|
|
|
|
m.eventBus.Publish(hub.Message{
|
|
|
|
|
Name: config.EventWakatimeFailure,
|
|
|
|
|
Fields: map[string]interface{}{config.FieldUser: forUser, config.FieldPayload: n},
|
|
|
|
|
})
|
|
|
|
|
} else if n%10 == 0 {
|
|
|
|
|
logbuch.Warn("%d / %d failed wakatime heartbeat relaying attempts for user %s within last 24 hours", n, maxFailuresPerDay, forUser.ID)
|
|
|
|
|
}
|
2021-01-22 00:17:32 +03:00
|
|
|
|
}
|
|
|
|
|
}
|