mirror of
https://github.com/kubeshark/kubeshark.git
synced 2026-05-15 05:37:09 +00:00
258 lines
7.0 KiB
Go
258 lines
7.0 KiB
Go
package controllers
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"mizuserver/pkg/database"
|
|
"mizuserver/pkg/models"
|
|
"mizuserver/pkg/providers"
|
|
"mizuserver/pkg/up9"
|
|
"mizuserver/pkg/utils"
|
|
"mizuserver/pkg/validation"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/google/martian/har"
|
|
"github.com/romana/rlog"
|
|
)
|
|
|
|
func GetEntries(c *gin.Context) {
|
|
entriesFilter := &models.EntriesFilter{}
|
|
|
|
if err := c.BindQuery(entriesFilter); err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
err := validation.Validate(entriesFilter)
|
|
if err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
|
|
order := database.OperatorToOrderMapping[entriesFilter.Operator]
|
|
operatorSymbol := database.OperatorToSymbolMapping[entriesFilter.Operator]
|
|
var entries []models.MizuEntry
|
|
database.GetEntriesTable().
|
|
Order(fmt.Sprintf("timestamp %s", order)).
|
|
Where(fmt.Sprintf("timestamp %s %v", operatorSymbol, entriesFilter.Timestamp)).
|
|
Omit("entry"). // remove the "big" entry field
|
|
Limit(entriesFilter.Limit).
|
|
Find(&entries)
|
|
|
|
if len(entries) > 0 && order == database.OrderDesc {
|
|
// the entries always order from oldest to newest so we should revers
|
|
utils.ReverseSlice(entries)
|
|
}
|
|
|
|
baseEntries := make([]models.BaseEntryDetails, 0)
|
|
for _, data := range entries {
|
|
harEntry := models.BaseEntryDetails{}
|
|
if err := models.GetEntry(&data, &harEntry); err != nil {
|
|
continue
|
|
}
|
|
baseEntries = append(baseEntries, harEntry)
|
|
}
|
|
|
|
c.JSON(http.StatusOK, baseEntries)
|
|
}
|
|
|
|
func GetHARs(c *gin.Context) {
|
|
entriesFilter := &models.HarFetchRequestQuery{}
|
|
order := database.OrderDesc
|
|
if err := c.BindQuery(entriesFilter); err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
err := validation.Validate(entriesFilter)
|
|
if err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
|
|
var timestampFrom, timestampTo int64
|
|
|
|
if entriesFilter.From < 0 {
|
|
timestampFrom = 0
|
|
} else {
|
|
timestampFrom = entriesFilter.From
|
|
}
|
|
if entriesFilter.To <= 0 {
|
|
timestampTo = time.Now().UnixNano() / int64(time.Millisecond)
|
|
} else {
|
|
timestampTo = entriesFilter.To
|
|
}
|
|
|
|
var entries []models.MizuEntry
|
|
database.GetEntriesTable().
|
|
Where(fmt.Sprintf("timestamp BETWEEN %v AND %v", timestampFrom, timestampTo)).
|
|
Order(fmt.Sprintf("timestamp %s", order)).
|
|
Find(&entries)
|
|
|
|
if len(entries) > 0 {
|
|
// the entries always order from oldest to newest so we should revers
|
|
utils.ReverseSlice(entries)
|
|
}
|
|
|
|
harsObject := map[string]*models.ExtendedHAR{}
|
|
|
|
for _, entryData := range entries {
|
|
var harEntry har.Entry
|
|
_ = json.Unmarshal([]byte(entryData.Entry), &harEntry)
|
|
if entryData.ResolvedDestination != "" {
|
|
harEntry.Request.URL = utils.SetHostname(harEntry.Request.URL, entryData.ResolvedDestination)
|
|
}
|
|
|
|
var fileName string
|
|
sourceOfEntry := entryData.ResolvedSource
|
|
if sourceOfEntry != "" {
|
|
// naively assumes the proper service source is http
|
|
sourceOfEntry = fmt.Sprintf("http://%s", sourceOfEntry)
|
|
//replace / from the file name cause they end up creating a corrupted folder
|
|
fileName = fmt.Sprintf("%s.har", strings.ReplaceAll(sourceOfEntry, "/", "_"))
|
|
} else {
|
|
fileName = "unknown_source.har"
|
|
}
|
|
if harOfSource, ok := harsObject[fileName]; ok {
|
|
harOfSource.Log.Entries = append(harOfSource.Log.Entries, &harEntry)
|
|
} else {
|
|
var entriesHar []*har.Entry
|
|
entriesHar = append(entriesHar, &harEntry)
|
|
harsObject[fileName] = &models.ExtendedHAR{
|
|
Log: &models.ExtendedLog{
|
|
Version: "1.2",
|
|
Creator: &models.ExtendedCreator{
|
|
Creator: &har.Creator{
|
|
Name: "mizu",
|
|
Version: "0.0.2",
|
|
},
|
|
},
|
|
Entries: entriesHar,
|
|
},
|
|
}
|
|
// leave undefined when no source is present, otherwise modeler assumes source is empty string ""
|
|
if sourceOfEntry != "" {
|
|
harsObject[fileName].Log.Creator.Source = &sourceOfEntry
|
|
}
|
|
}
|
|
}
|
|
|
|
retObj := map[string][]byte{}
|
|
for k, v := range harsObject {
|
|
bytesData, _ := json.Marshal(v)
|
|
retObj[k] = bytesData
|
|
}
|
|
buffer := utils.ZipData(retObj)
|
|
c.Data(http.StatusOK, "application/octet-stream", buffer.Bytes())
|
|
}
|
|
|
|
func UploadEntries(c *gin.Context) {
|
|
rlog.Infof("Upload entries - started\n")
|
|
|
|
uploadParams := &models.UploadEntriesRequestQuery{}
|
|
if err := c.BindQuery(uploadParams); err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
if err := validation.Validate(uploadParams); err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
if up9.GetAnalyzeInfo().IsAnalyzing {
|
|
c.String(http.StatusBadRequest, "Cannot analyze, mizu is already analyzing")
|
|
return
|
|
}
|
|
|
|
rlog.Infof("Upload entries - creating token. dest %s\n", uploadParams.Dest)
|
|
token, err := up9.CreateAnonymousToken(uploadParams.Dest)
|
|
if err != nil {
|
|
c.String(http.StatusServiceUnavailable, "Cannot analyze, mizu is already analyzing")
|
|
return
|
|
}
|
|
rlog.Infof("Upload entries - uploading. token: %s model: %s\n", token.Token, token.Model)
|
|
go up9.UploadEntriesImpl(token.Token, token.Model, uploadParams.Dest, uploadParams.SleepIntervalSec)
|
|
c.String(http.StatusOK, "OK")
|
|
}
|
|
|
|
func GetFullEntries(c *gin.Context) {
|
|
entriesFilter := &models.HarFetchRequestQuery{}
|
|
if err := c.BindQuery(entriesFilter); err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
err := validation.Validate(entriesFilter)
|
|
if err != nil {
|
|
c.JSON(http.StatusBadRequest, err)
|
|
}
|
|
|
|
var timestampFrom, timestampTo int64
|
|
|
|
if entriesFilter.From < 0 {
|
|
timestampFrom = 0
|
|
} else {
|
|
timestampFrom = entriesFilter.From
|
|
}
|
|
if entriesFilter.To <= 0 {
|
|
timestampTo = time.Now().UnixNano() / int64(time.Millisecond)
|
|
} else {
|
|
timestampTo = entriesFilter.To
|
|
}
|
|
|
|
entriesArray := database.GetEntriesFromDb(timestampFrom, timestampTo)
|
|
result := make([]models.FullEntryDetails, 0)
|
|
for _, data := range entriesArray {
|
|
harEntry := models.FullEntryDetails{}
|
|
if err := models.GetEntry(&data, &harEntry); err != nil {
|
|
continue
|
|
}
|
|
result = append(result, harEntry)
|
|
}
|
|
c.JSON(http.StatusOK, result)
|
|
}
|
|
|
|
func GetEntry(c *gin.Context) {
|
|
var entryData models.MizuEntry
|
|
database.GetEntriesTable().
|
|
Where(map[string]string{"entryId": c.Param("entryId")}).
|
|
First(&entryData)
|
|
|
|
fullEntry := models.FullEntryDetails{}
|
|
if err := models.GetEntry(&entryData, &fullEntry); err != nil {
|
|
c.JSON(http.StatusInternalServerError, map[string]interface{}{
|
|
"error": true,
|
|
"msg": "Can't get entry details",
|
|
})
|
|
}
|
|
fullEntryWithPolicy := models.FullEntryWithPolicy{}
|
|
if err := models.GetEntry(&entryData, &fullEntryWithPolicy); err != nil {
|
|
c.JSON(http.StatusInternalServerError, map[string]interface{}{
|
|
"error": true,
|
|
"msg": "Can't get entry details",
|
|
})
|
|
}
|
|
c.JSON(http.StatusOK, fullEntryWithPolicy)
|
|
}
|
|
|
|
func DeleteAllEntries(c *gin.Context) {
|
|
database.GetEntriesTable().
|
|
Where("1 = 1").
|
|
Delete(&models.MizuEntry{})
|
|
|
|
c.JSON(http.StatusOK, map[string]string{
|
|
"msg": "Success",
|
|
})
|
|
|
|
}
|
|
|
|
func GetGeneralStats(c *gin.Context) {
|
|
c.JSON(http.StatusOK, providers.GetGeneralStats())
|
|
}
|
|
|
|
func GetTappingStatus(c *gin.Context) {
|
|
c.JSON(http.StatusOK, providers.TapStatus)
|
|
}
|
|
|
|
func AnalyzeInformation(c *gin.Context) {
|
|
c.JSON(http.StatusOK, up9.GetAnalyzeInfo())
|
|
}
|
|
|
|
func GetRecentTLSLinks(c *gin.Context) {
|
|
c.JSON(http.StatusOK, providers.GetAllRecentTLSAddresses())
|
|
}
|