mirror of
https://github.com/kubeshark/kubeshark.git
synced 2025-06-21 13:58:49 +00:00
Merge pull request #744 from up9inc/develop
Develop -> main (Release 0.25.0)
This commit is contained in:
commit
d987bb17d2
@ -1,4 +1,4 @@
|
|||||||
name: PR validation
|
name: Build
|
||||||
|
|
||||||
on:
|
on:
|
||||||
pull_request:
|
pull_request:
|
||||||
@ -12,7 +12,7 @@ concurrency:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build-cli:
|
build-cli:
|
||||||
name: Build CLI executable
|
name: CLI executable build
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Set up Go 1.16
|
- name: Set up Go 1.16
|
||||||
@ -27,7 +27,7 @@ jobs:
|
|||||||
run: make cli
|
run: make cli
|
||||||
|
|
||||||
build-agent:
|
build-agent:
|
||||||
name: Build Agent docker image
|
name: Agent docker image build
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code into the Go module directory
|
- name: Check out code into the Go module directory
|
90
.github/workflows/static_code_analysis.yml
vendored
Normal file
90
.github/workflows/static_code_analysis.yml
vendored
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
name: Static code analysis
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- 'develop'
|
||||||
|
- 'main'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
golangci:
|
||||||
|
name: Go lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-go@v2
|
||||||
|
with:
|
||||||
|
go-version: '^1.16'
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
sudo apt update
|
||||||
|
sudo apt install -y libpcap-dev
|
||||||
|
|
||||||
|
- name: Go lint - agent
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: agent
|
||||||
|
args: --timeout=3m
|
||||||
|
|
||||||
|
- name: Go lint - shared
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: shared
|
||||||
|
args: --timeout=3m
|
||||||
|
|
||||||
|
- name: Go lint - tap
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap
|
||||||
|
args: --timeout=3m
|
||||||
|
|
||||||
|
- name: Go lint - CLI
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: cli
|
||||||
|
args: --timeout=3m
|
||||||
|
|
||||||
|
- name: Go lint - acceptanceTests
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: acceptanceTests
|
||||||
|
args: --timeout=3m
|
||||||
|
|
||||||
|
- name: Go lint - tap/api
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap/api
|
||||||
|
|
||||||
|
- name: Go lint - tap/extensions/amqp
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap/extensions/amqp
|
||||||
|
|
||||||
|
- name: Go lint - tap/extensions/http
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap/extensions/http
|
||||||
|
|
||||||
|
- name: Go lint - tap/extensions/kafka
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap/extensions/kafka
|
||||||
|
|
||||||
|
- name: Go lint - tap/extensions/redis
|
||||||
|
uses: golangci/golangci-lint-action@v2
|
||||||
|
with:
|
||||||
|
version: latest
|
||||||
|
working-directory: tap/extensions/redis
|
@ -1,14 +1,10 @@
|
|||||||
name: tests validation
|
name: Test
|
||||||
|
|
||||||
on:
|
on:
|
||||||
pull_request:
|
pull_request:
|
||||||
branches:
|
branches:
|
||||||
- 'develop'
|
- 'develop'
|
||||||
- 'main'
|
- 'main'
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- 'develop'
|
|
||||||
- 'main'
|
|
||||||
|
|
||||||
concurrency:
|
concurrency:
|
||||||
group: mizu-tests-validation-${{ github.ref }}
|
group: mizu-tests-validation-${{ github.ref }}
|
||||||
@ -16,7 +12,7 @@ concurrency:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
run-tests-cli:
|
run-tests-cli:
|
||||||
name: Run CLI tests
|
name: CLI Tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Set up Go 1.16
|
- name: Set up Go 1.16
|
||||||
@ -34,7 +30,7 @@ jobs:
|
|||||||
uses: codecov/codecov-action@v2
|
uses: codecov/codecov-action@v2
|
||||||
|
|
||||||
run-tests-agent:
|
run-tests-agent:
|
||||||
name: Run Agent tests
|
name: Agent Tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Set up Go 1.16
|
- name: Set up Go 1.16
|
@ -72,10 +72,10 @@ ARG SEM_VER=0.0.0
|
|||||||
WORKDIR /app/agent-build
|
WORKDIR /app/agent-build
|
||||||
|
|
||||||
RUN go build -ldflags="-extldflags=-static -s -w \
|
RUN go build -ldflags="-extldflags=-static -s -w \
|
||||||
-X 'mizuserver/pkg/version.GitCommitHash=${COMMIT_HASH}' \
|
-X 'github.com/up9inc/mizu/agent/pkg/version.GitCommitHash=${COMMIT_HASH}' \
|
||||||
-X 'mizuserver/pkg/version.Branch=${GIT_BRANCH}' \
|
-X 'github.com/up9inc/mizu/agent/pkg/version.Branch=${GIT_BRANCH}' \
|
||||||
-X 'mizuserver/pkg/version.BuildTimestamp=${BUILD_TIMESTAMP}' \
|
-X 'github.com/up9inc/mizu/agent/pkg/version.BuildTimestamp=${BUILD_TIMESTAMP}' \
|
||||||
-X 'mizuserver/pkg/version.SemVer=${SEM_VER}'" -o mizuagent .
|
-X 'github.com/up9inc/mizu/agent/pkg/version.SemVer=${SEM_VER}'" -o mizuagent .
|
||||||
|
|
||||||
# Download Basenine executable, verify the sha1sum
|
# Download Basenine executable, verify the sha1sum
|
||||||
ADD https://github.com/up9inc/basenine/releases/download/v0.4.13/basenine_linux_${GOARCH} ./basenine_linux_${GOARCH}
|
ADD https://github.com/up9inc/basenine/releases/download/v0.4.13/basenine_linux_${GOARCH} ./basenine_linux_${GOARCH}
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
"viewportHeight": 1080,
|
"viewportHeight": 1080,
|
||||||
"video": false,
|
"video": false,
|
||||||
"screenshotOnRunFailure": false,
|
"screenshotOnRunFailure": false,
|
||||||
"defaultCommandTimeout": 5000,
|
"defaultCommandTimeout": 6000,
|
||||||
"testFiles": [
|
"testFiles": [
|
||||||
"tests/GuiPort.js",
|
"tests/GuiPort.js",
|
||||||
"tests/MultipleNamespaces.js",
|
"tests/MultipleNamespaces.js",
|
||||||
|
@ -3,6 +3,7 @@ import {resizeToHugeMizu, resizeToNormalMizu} from "../testHelpers/TrafficHelper
|
|||||||
const greenFilterColor = 'rgb(210, 250, 210)';
|
const greenFilterColor = 'rgb(210, 250, 210)';
|
||||||
const redFilterColor = 'rgb(250, 214, 220)';
|
const redFilterColor = 'rgb(250, 214, 220)';
|
||||||
const refreshWaitTimeout = 10000;
|
const refreshWaitTimeout = 10000;
|
||||||
|
const bodyJsonClass = '.hljs';
|
||||||
|
|
||||||
it('opening mizu', function () {
|
it('opening mizu', function () {
|
||||||
cy.visit(Cypress.env('testUrl'));
|
cy.visit(Cypress.env('testUrl'));
|
||||||
@ -25,6 +26,37 @@ it('filtering guide check', function () {
|
|||||||
cy.get('#modal-modal-title').should('not.exist');
|
cy.get('#modal-modal-title').should('not.exist');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('right side sanity test', function () {
|
||||||
|
cy.get('#entryDetailedTitleBodySize').then(sizeTopLine => {
|
||||||
|
const sizeOnTopLine = sizeTopLine.text().replace(' B', '');
|
||||||
|
cy.contains('Response').click();
|
||||||
|
cy.contains('Body Size (bytes)').parent().next().then(size => {
|
||||||
|
const bodySizeByDetails = size.text();
|
||||||
|
expect(sizeOnTopLine).to.equal(bodySizeByDetails, 'The body size in the top line should match the details in the response');
|
||||||
|
|
||||||
|
if (parseInt(bodySizeByDetails) < 0) {
|
||||||
|
throw new Error(`The body size cannot be negative. got the size: ${bodySizeByDetails}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
cy.get('#entryDetailedTitleElapsedTime').then(timeInMs => {
|
||||||
|
const time = timeInMs.text();
|
||||||
|
if (time < '0ms') {
|
||||||
|
throw new Error(`The time in the top line cannot be negative ${time}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
cy.get('#rightSideContainer [title="Status Code"]').then(status => {
|
||||||
|
const statusCode = status.text();
|
||||||
|
cy.contains('Status').parent().next().then(statusInDetails => {
|
||||||
|
const statusCodeInDetails = statusInDetails.text();
|
||||||
|
|
||||||
|
expect(statusCode).to.equal(statusCodeInDetails, 'The status code in the top line should match the status code in details');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
checkIllegalFilter('invalid filter');
|
checkIllegalFilter('invalid filter');
|
||||||
|
|
||||||
checkFilter({
|
checkFilter({
|
||||||
@ -161,11 +193,12 @@ function checkFilter(filterDetails){
|
|||||||
if (!applyByEnter)
|
if (!applyByEnter)
|
||||||
cy.get('[type="submit"]').click();
|
cy.get('[type="submit"]').click();
|
||||||
|
|
||||||
// only one entry in DOM after filtering, checking all four checks on it
|
// only one entry in DOM after filtering, checking all checks on it
|
||||||
leftTextCheck(totalEntries - 1, leftSidePath, leftSideExpectedText);
|
leftTextCheck(totalEntries - 1, leftSidePath, leftSideExpectedText);
|
||||||
leftOnHoverCheck(totalEntries - 1, leftSidePath, name);
|
leftOnHoverCheck(totalEntries - 1, leftSidePath, name);
|
||||||
rightTextCheck(rightSidePath, rightSideExpectedText);
|
rightTextCheck(rightSidePath, rightSideExpectedText);
|
||||||
rightOnHoverCheck(rightSidePath, name);
|
rightOnHoverCheck(rightSidePath, name);
|
||||||
|
checkRightSideResponseBody();
|
||||||
|
|
||||||
cy.get('[title="Fetch old records"]').click();
|
cy.get('[title="Fetch old records"]').click();
|
||||||
resizeToHugeMizu();
|
resizeToHugeMizu();
|
||||||
@ -196,6 +229,7 @@ function deeperChcek(leftSidePath, rightSidePath, filterName, leftSideExpectedTe
|
|||||||
cy.get(`#list #entry-${entryNum}`).click();
|
cy.get(`#list #entry-${entryNum}`).click();
|
||||||
rightTextCheck(rightSidePath, rightSideExpectedText);
|
rightTextCheck(rightSidePath, rightSideExpectedText);
|
||||||
rightOnHoverCheck(rightSidePath, filterName);
|
rightOnHoverCheck(rightSidePath, filterName);
|
||||||
|
checkRightSideResponseBody();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -216,3 +250,88 @@ function rightOnHoverCheck(path, expectedText) {
|
|||||||
cy.get(`.TrafficPage-Container > :nth-child(2) ${path}`).trigger('mouseover');
|
cy.get(`.TrafficPage-Container > :nth-child(2) ${path}`).trigger('mouseover');
|
||||||
cy.get(`.TrafficPage-Container > :nth-child(2) .Queryable-Tooltip`).should('have.text', expectedText);
|
cy.get(`.TrafficPage-Container > :nth-child(2) .Queryable-Tooltip`).should('have.text', expectedText);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function checkRightSideResponseBody() {
|
||||||
|
cy.contains('Response').click();
|
||||||
|
clickCheckbox('Decode Base64');
|
||||||
|
|
||||||
|
cy.get(`${bodyJsonClass}`).then(value => {
|
||||||
|
const encodedBody = value.text();
|
||||||
|
cy.log(encodedBody);
|
||||||
|
|
||||||
|
const decodedBody = atob(encodedBody);
|
||||||
|
const responseBody = JSON.parse(decodedBody);
|
||||||
|
|
||||||
|
const expectdJsonBody = {
|
||||||
|
args: RegExp({}),
|
||||||
|
url: RegExp('http://.*/get'),
|
||||||
|
headers: {
|
||||||
|
"User-Agent": RegExp('[REDACTED]'),
|
||||||
|
"Accept-Encoding": RegExp('gzip'),
|
||||||
|
"X-Forwarded-Uri": RegExp('/api/v1/namespaces/.*/services/.*/proxy/get')
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
expect(responseBody.args).to.match(expectdJsonBody.args);
|
||||||
|
expect(responseBody.url).to.match(expectdJsonBody.url);
|
||||||
|
expect(responseBody.headers['User-Agent']).to.match(expectdJsonBody.headers['User-Agent']);
|
||||||
|
expect(responseBody.headers['Accept-Encoding']).to.match(expectdJsonBody.headers['Accept-Encoding']);
|
||||||
|
expect(responseBody.headers['X-Forwarded-Uri']).to.match(expectdJsonBody.headers['X-Forwarded-Uri']);
|
||||||
|
|
||||||
|
cy.get(`${bodyJsonClass}`).should('have.text', encodedBody);
|
||||||
|
clickCheckbox('Decode Base64');
|
||||||
|
|
||||||
|
cy.get(`${bodyJsonClass} > `).its('length').should('be.gt', 1).then(linesNum => {
|
||||||
|
cy.get(`${bodyJsonClass} > >`).its('length').should('be.gt', linesNum).then(jsonItemsNum => {
|
||||||
|
checkPrettyAndLineNums(jsonItemsNum, decodedBody);
|
||||||
|
|
||||||
|
clickCheckbox('Line numbers');
|
||||||
|
checkPrettyOrNothing(jsonItemsNum, decodedBody);
|
||||||
|
|
||||||
|
clickCheckbox('Pretty');
|
||||||
|
checkPrettyOrNothing(jsonItemsNum, decodedBody);
|
||||||
|
|
||||||
|
clickCheckbox('Line numbers');
|
||||||
|
checkOnlyLineNumberes(jsonItemsNum, decodedBody);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function clickCheckbox(type) {
|
||||||
|
cy.contains(`${type}`).prev().children().click();
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkPrettyAndLineNums(jsonItemsLen, decodedBody) {
|
||||||
|
decodedBody = decodedBody.replaceAll(' ', '');
|
||||||
|
cy.get(`${bodyJsonClass} >`).then(elements => {
|
||||||
|
const lines = Object.values(elements);
|
||||||
|
lines.forEach((line, index) => {
|
||||||
|
if (line.getAttribute) {
|
||||||
|
const cleanLine = getCleanLine(line);
|
||||||
|
const currentLineFromDecodedText = decodedBody.substring(0, cleanLine.length);
|
||||||
|
|
||||||
|
expect(cleanLine).to.equal(currentLineFromDecodedText, `expected the text in line number ${index + 1} to match the text that generated by the base64 decoding`)
|
||||||
|
|
||||||
|
decodedBody = decodedBody.substring(cleanLine.length);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCleanLine(lineElement) {
|
||||||
|
return (lineElement.innerText.substring(0, lineElement.innerText.length - 1)).replaceAll(' ', '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkPrettyOrNothing(jsonItems, decodedBody) {
|
||||||
|
cy.get(`${bodyJsonClass} > `).should('have.length', jsonItems).then(text => {
|
||||||
|
const json = text.text();
|
||||||
|
expect(json).to.equal(decodedBody);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkOnlyLineNumberes(jsonItems, decodedText) {
|
||||||
|
cy.get(`${bodyJsonClass} >`).should('have.length', 1).and('have.text', decodedText);
|
||||||
|
cy.get(`${bodyJsonClass} > >`).should('have.length', jsonItems)
|
||||||
|
}
|
||||||
|
@ -3,7 +3,6 @@ package acceptanceTests
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -11,12 +10,10 @@ import (
|
|||||||
"os/exec"
|
"os/exec"
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
|
||||||
"syscall"
|
"syscall"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gorilla/websocket"
|
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -28,7 +25,6 @@ const (
|
|||||||
defaultServiceName = "httpbin"
|
defaultServiceName = "httpbin"
|
||||||
defaultEntriesCount = 50
|
defaultEntriesCount = 50
|
||||||
waitAfterTapPodsReady = 3 * time.Second
|
waitAfterTapPodsReady = 3 * time.Second
|
||||||
cleanCommandTimeout = 1 * time.Minute
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type PodDescriptor struct {
|
type PodDescriptor struct {
|
||||||
@ -36,18 +32,6 @@ type PodDescriptor struct {
|
|||||||
Namespace string
|
Namespace string
|
||||||
}
|
}
|
||||||
|
|
||||||
func isPodDescriptorInPodArray(pods []map[string]interface{}, podDescriptor PodDescriptor) bool {
|
|
||||||
for _, pod := range pods {
|
|
||||||
podNamespace := pod["namespace"].(string)
|
|
||||||
podName := pod["name"].(string)
|
|
||||||
|
|
||||||
if podDescriptor.Namespace == podNamespace && strings.Contains(podName, podDescriptor.Name) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func getCliPath() (string, error) {
|
func getCliPath() (string, error) {
|
||||||
dir, filePathErr := os.Getwd()
|
dir, filePathErr := os.Getwd()
|
||||||
if filePathErr != nil {
|
if filePathErr != nil {
|
||||||
@ -84,10 +68,6 @@ func getApiServerUrl(port uint16) string {
|
|||||||
return fmt.Sprintf("http://localhost:%v", port)
|
return fmt.Sprintf("http://localhost:%v", port)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getWebSocketUrl(port uint16) string {
|
|
||||||
return fmt.Sprintf("ws://localhost:%v/ws", port)
|
|
||||||
}
|
|
||||||
|
|
||||||
func getDefaultCommandArgs() []string {
|
func getDefaultCommandArgs() []string {
|
||||||
setFlag := "--set"
|
setFlag := "--set"
|
||||||
telemetry := "telemetry=false"
|
telemetry := "telemetry=false"
|
||||||
@ -130,20 +110,6 @@ func getDefaultConfigCommandArgs() []string {
|
|||||||
return append([]string{configCommand}, defaultCmdArgs...)
|
return append([]string{configCommand}, defaultCmdArgs...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getDefaultCleanCommandArgs() []string {
|
|
||||||
cleanCommand := "clean"
|
|
||||||
defaultCmdArgs := getDefaultCommandArgs()
|
|
||||||
|
|
||||||
return append([]string{cleanCommand}, defaultCmdArgs...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func getDefaultViewCommandArgs() []string {
|
|
||||||
viewCommand := "view"
|
|
||||||
defaultCmdArgs := getDefaultCommandArgs()
|
|
||||||
|
|
||||||
return append([]string{viewCommand}, defaultCmdArgs...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func runCypressTests(t *testing.T, cypressRunCmd string) {
|
func runCypressTests(t *testing.T, cypressRunCmd string) {
|
||||||
cypressCmd := exec.Command("bash", "-c", cypressRunCmd)
|
cypressCmd := exec.Command("bash", "-c", cypressRunCmd)
|
||||||
t.Logf("running command: %v", cypressCmd.String())
|
t.Logf("running command: %v", cypressCmd.String())
|
||||||
@ -268,36 +234,6 @@ func executeHttpPostRequestWithHeaders(url string, headers map[string]string, bo
|
|||||||
return executeHttpRequest(response, requestErr)
|
return executeHttpRequest(response, requestErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
func runMizuClean() error {
|
|
||||||
cliPath, err := getCliPath()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanCmdArgs := getDefaultCleanCommandArgs()
|
|
||||||
|
|
||||||
cleanCmd := exec.Command(cliPath, cleanCmdArgs...)
|
|
||||||
|
|
||||||
commandDone := make(chan error)
|
|
||||||
go func() {
|
|
||||||
if err := cleanCmd.Run(); err != nil {
|
|
||||||
commandDone <- err
|
|
||||||
}
|
|
||||||
commandDone <- nil
|
|
||||||
}()
|
|
||||||
|
|
||||||
select {
|
|
||||||
case err = <-commandDone:
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
case <-time.After(cleanCommandTimeout):
|
|
||||||
return errors.New("clean command timed out")
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func cleanupCommand(cmd *exec.Cmd) error {
|
func cleanupCommand(cmd *exec.Cmd) error {
|
||||||
if err := cmd.Process.Signal(syscall.SIGQUIT); err != nil {
|
if err := cmd.Process.Signal(syscall.SIGQUIT); err != nil {
|
||||||
return err
|
return err
|
||||||
@ -310,17 +246,6 @@ func cleanupCommand(cmd *exec.Cmd) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func getPods(tapStatusInterface interface{}) ([]map[string]interface{}, error) {
|
|
||||||
tapPodsInterface := tapStatusInterface.([]interface{})
|
|
||||||
|
|
||||||
var pods []map[string]interface{}
|
|
||||||
for _, podInterface := range tapPodsInterface {
|
|
||||||
pods = append(pods, podInterface.(map[string]interface{}))
|
|
||||||
}
|
|
||||||
|
|
||||||
return pods, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func getLogsPath() (string, error) {
|
func getLogsPath() (string, error) {
|
||||||
dir, filePathErr := os.Getwd()
|
dir, filePathErr := os.Getwd()
|
||||||
if filePathErr != nil {
|
if filePathErr != nil {
|
||||||
@ -331,77 +256,6 @@ func getLogsPath() (string, error) {
|
|||||||
return logsPath, nil
|
return logsPath, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// waitTimeout waits for the waitgroup for the specified max timeout.
|
|
||||||
// Returns true if waiting timed out.
|
|
||||||
func waitTimeout(wg *sync.WaitGroup, timeout time.Duration) bool {
|
|
||||||
channel := make(chan struct{})
|
|
||||||
go func() {
|
|
||||||
defer close(channel)
|
|
||||||
wg.Wait()
|
|
||||||
}()
|
|
||||||
select {
|
|
||||||
case <-channel:
|
|
||||||
return false // completed normally
|
|
||||||
case <-time.After(timeout):
|
|
||||||
return true // timed out
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkEntriesAtLeast checks whether the number of entries greater than or equal to n
|
|
||||||
func checkEntriesAtLeast(entries []map[string]interface{}, n int) error {
|
|
||||||
if len(entries) < n {
|
|
||||||
return fmt.Errorf("Unexpected entries result - Expected more than %d entries", n-1)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// getDBEntries retrieves the entries from the database before the given timestamp.
|
|
||||||
// Also limits the results according to the limit parameter.
|
|
||||||
// Timeout for the WebSocket connection is defined by the timeout parameter.
|
|
||||||
func getDBEntries(timestamp int64, limit int, timeout time.Duration) (entries []map[string]interface{}, err error) {
|
|
||||||
query := fmt.Sprintf("timestamp < %d and limit(%d)", timestamp, limit)
|
|
||||||
webSocketUrl := getWebSocketUrl(defaultApiServerPort)
|
|
||||||
|
|
||||||
var connection *websocket.Conn
|
|
||||||
connection, _, err = websocket.DefaultDialer.Dial(webSocketUrl, nil)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer connection.Close()
|
|
||||||
|
|
||||||
handleWSConnection := func(wg *sync.WaitGroup) {
|
|
||||||
defer wg.Done()
|
|
||||||
for {
|
|
||||||
_, message, err := connection.ReadMessage()
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var data map[string]interface{}
|
|
||||||
if err = json.Unmarshal([]byte(message), &data); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if data["messageType"] == "entry" {
|
|
||||||
entries = append(entries, data)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
err = connection.WriteMessage(websocket.TextMessage, []byte(query))
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var wg sync.WaitGroup
|
|
||||||
go handleWSConnection(&wg)
|
|
||||||
wg.Add(1)
|
|
||||||
|
|
||||||
waitTimeout(&wg, timeout)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func Contains(slice []string, containsValue string) bool {
|
func Contains(slice []string, containsValue string) bool {
|
||||||
for _, sliceValue := range slice {
|
for _, sliceValue := range slice {
|
||||||
if sliceValue == containsValue {
|
if sliceValue == containsValue {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
module mizuserver
|
module github.com/up9inc/mizu/agent
|
||||||
|
|
||||||
go 1.16
|
go 1.16
|
||||||
|
|
||||||
|
@ -6,17 +6,6 @@ import (
|
|||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"mizuserver/pkg/api"
|
|
||||||
"mizuserver/pkg/config"
|
|
||||||
"mizuserver/pkg/controllers"
|
|
||||||
"mizuserver/pkg/elastic"
|
|
||||||
"mizuserver/pkg/middlewares"
|
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"mizuserver/pkg/oas"
|
|
||||||
"mizuserver/pkg/routes"
|
|
||||||
"mizuserver/pkg/servicemap"
|
|
||||||
"mizuserver/pkg/up9"
|
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
@ -26,6 +15,21 @@ import (
|
|||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/oas"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/routes"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/servicemap"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/up9"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/elastic"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/api"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/config"
|
||||||
|
|
||||||
v1 "k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
|
|
||||||
"github.com/antelman107/net-wait-go/wait"
|
"github.com/antelman107/net-wait-go/wait"
|
||||||
|
@ -5,22 +5,23 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/elastic"
|
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"mizuserver/pkg/holder"
|
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"mizuserver/pkg/servicemap"
|
"github.com/up9inc/mizu/agent/pkg/elastic"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/holder"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
|
||||||
"mizuserver/pkg/models"
|
"github.com/up9inc/mizu/agent/pkg/servicemap"
|
||||||
"mizuserver/pkg/oas"
|
|
||||||
"mizuserver/pkg/resolver"
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
"mizuserver/pkg/utils"
|
"github.com/up9inc/mizu/agent/pkg/oas"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/resolver"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
@ -42,10 +43,8 @@ func StartResolving(namespace string) {
|
|||||||
res.Start(ctx)
|
res.Start(ctx)
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
select {
|
err := <-errOut
|
||||||
case err := <-errOut:
|
logger.Log.Infof("name resolving error %s", err)
|
||||||
logger.Log.Infof("name resolving error %s", err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
@ -67,7 +66,7 @@ func startReadingFiles(workingDir string) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
for true {
|
for {
|
||||||
dir, _ := os.Open(workingDir)
|
dir, _ := os.Open(workingDir)
|
||||||
dirFiles, _ := dir.Readdir(-1)
|
dirFiles, _ := dir.Readdir(-1)
|
||||||
|
|
||||||
@ -124,7 +123,9 @@ func startReadingChannel(outputItems <-chan *tapApi.OutputChannelItem, extension
|
|||||||
if extension.Protocol.Name == "http" {
|
if extension.Protocol.Name == "http" {
|
||||||
if !disableOASValidation {
|
if !disableOASValidation {
|
||||||
var httpPair tapApi.HTTPRequestResponsePair
|
var httpPair tapApi.HTTPRequestResponsePair
|
||||||
json.Unmarshal([]byte(mizuEntry.HTTPPair), &httpPair)
|
if err := json.Unmarshal([]byte(mizuEntry.HTTPPair), &httpPair); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
contract := handleOAS(ctx, doc, router, httpPair.Request.Payload.RawRequest, httpPair.Response.Payload.RawResponse, contractContent)
|
contract := handleOAS(ctx, doc, router, httpPair.Request.Payload.RawRequest, httpPair.Response.Payload.RawResponse, contractContent)
|
||||||
mizuEntry.ContractStatus = contract.Status
|
mizuEntry.ContractStatus = contract.Status
|
||||||
|
@ -2,18 +2,17 @@ package api
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
basenine "github.com/up9inc/basenine/client/go"
|
basenine "github.com/up9inc/basenine/client/go"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/debounce"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
tapApi "github.com/up9inc/mizu/tap/api"
|
tapApi "github.com/up9inc/mizu/tap/api"
|
||||||
)
|
)
|
||||||
@ -42,7 +41,7 @@ var connectedWebsocketIdCounter = 0
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
websocketUpgrader.CheckOrigin = func(r *http.Request) bool { return true } // like cors for web socket
|
websocketUpgrader.CheckOrigin = func(r *http.Request) bool { return true } // like cors for web socket
|
||||||
connectedWebsockets = make(map[int]*SocketConnection, 0)
|
connectedWebsockets = make(map[int]*SocketConnection)
|
||||||
}
|
}
|
||||||
|
|
||||||
func WebSocketRoutes(app *gin.Engine, eventHandlers EventHandlers, startTime int64) {
|
func WebSocketRoutes(app *gin.Engine, eventHandlers EventHandlers, startTime int64) {
|
||||||
@ -94,7 +93,10 @@ func websocketHandler(w http.ResponseWriter, r *http.Request, eventHandlers Even
|
|||||||
eventHandlers.WebSocketConnect(socketId, isTapper)
|
eventHandlers.WebSocketConnect(socketId, isTapper)
|
||||||
|
|
||||||
startTimeBytes, _ := models.CreateWebsocketStartTimeMessage(startTime)
|
startTimeBytes, _ := models.CreateWebsocketStartTimeMessage(startTime)
|
||||||
SendToSocket(socketId, startTimeBytes)
|
|
||||||
|
if err = SendToSocket(socketId, startTimeBytes); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
for {
|
for {
|
||||||
_, msg, err := ws.ReadMessage()
|
_, msg, err := ws.ReadMessage()
|
||||||
@ -117,7 +119,9 @@ func websocketHandler(w http.ResponseWriter, r *http.Request, eventHandlers Even
|
|||||||
AutoClose: 5000,
|
AutoClose: 5000,
|
||||||
Text: fmt.Sprintf("Syntax error: %s", err.Error()),
|
Text: fmt.Sprintf("Syntax error: %s", err.Error()),
|
||||||
})
|
})
|
||||||
SendToSocket(socketId, toastBytes)
|
if err := SendToSocket(socketId, toastBytes); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,7 +141,9 @@ func websocketHandler(w http.ResponseWriter, r *http.Request, eventHandlers Even
|
|||||||
base := tapApi.Summarize(entry)
|
base := tapApi.Summarize(entry)
|
||||||
|
|
||||||
baseEntryBytes, _ := models.CreateBaseEntryWebSocketMessage(base)
|
baseEntryBytes, _ := models.CreateBaseEntryWebSocketMessage(base)
|
||||||
SendToSocket(socketId, baseEntryBytes)
|
if err := SendToSocket(socketId, baseEntryBytes); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -156,7 +162,9 @@ func websocketHandler(w http.ResponseWriter, r *http.Request, eventHandlers Even
|
|||||||
}
|
}
|
||||||
|
|
||||||
metadataBytes, _ := models.CreateWebsocketQueryMetadataMessage(metadata)
|
metadataBytes, _ := models.CreateWebsocketQueryMetadataMessage(metadata)
|
||||||
SendToSocket(socketId, metadataBytes)
|
if err := SendToSocket(socketId, metadataBytes); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -183,14 +191,10 @@ func socketCleanup(socketId int, socketConnection *SocketConnection) {
|
|||||||
socketConnection.eventHandlers.WebSocketDisconnect(socketId, socketConnection.isTapper)
|
socketConnection.eventHandlers.WebSocketDisconnect(socketId, socketConnection.isTapper)
|
||||||
}
|
}
|
||||||
|
|
||||||
var db = debounce.NewDebouncer(time.Second*5, func() {
|
|
||||||
logger.Log.Error("Successfully sent to socket")
|
|
||||||
})
|
|
||||||
|
|
||||||
func SendToSocket(socketId int, message []byte) error {
|
func SendToSocket(socketId int, message []byte) error {
|
||||||
socketObj := connectedWebsockets[socketId]
|
socketObj := connectedWebsockets[socketId]
|
||||||
if socketObj == nil {
|
if socketObj == nil {
|
||||||
return errors.New("Socket is disconnected")
|
return fmt.Errorf("Socket %v is disconnected", socketId)
|
||||||
}
|
}
|
||||||
|
|
||||||
var sent = false
|
var sent = false
|
||||||
@ -204,7 +208,10 @@ func SendToSocket(socketId int, message []byte) error {
|
|||||||
socketObj.lock.Lock() // gorilla socket panics from concurrent writes to a single socket
|
socketObj.lock.Lock() // gorilla socket panics from concurrent writes to a single socket
|
||||||
err := socketObj.connection.WriteMessage(1, message)
|
err := socketObj.connection.WriteMessage(1, message)
|
||||||
socketObj.lock.Unlock()
|
socketObj.lock.Unlock()
|
||||||
|
|
||||||
sent = true
|
sent = true
|
||||||
return err
|
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to write message to socket %v, err: %w", socketId, err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -3,12 +3,13 @@ package api
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"mizuserver/pkg/providers/tappers"
|
|
||||||
"mizuserver/pkg/up9"
|
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/up9"
|
||||||
|
|
||||||
tapApi "github.com/up9inc/mizu/tap/api"
|
tapApi "github.com/up9inc/mizu/tap/api"
|
||||||
|
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
@ -54,9 +55,8 @@ func (h *RoutesEventHandlers) WebSocketDisconnect(socketId int, isTapper bool) {
|
|||||||
func BroadcastToBrowserClients(message []byte) {
|
func BroadcastToBrowserClients(message []byte) {
|
||||||
for _, socketId := range browserClientSocketUUIDs {
|
for _, socketId := range browserClientSocketUUIDs {
|
||||||
go func(socketId int) {
|
go func(socketId int) {
|
||||||
err := SendToSocket(socketId, message)
|
if err := SendToSocket(socketId, message); err != nil {
|
||||||
if err != nil {
|
logger.Log.Error(err)
|
||||||
logger.Log.Errorf("error sending message to socket ID %d: %v", socketId, err)
|
|
||||||
}
|
}
|
||||||
}(socketId)
|
}(socketId)
|
||||||
}
|
}
|
||||||
|
@ -2,21 +2,22 @@ package controllers
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"net/http"
|
||||||
|
"regexp"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/config"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tapConfig"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappedPods"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappers"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/kubernetes"
|
"github.com/up9inc/mizu/shared/kubernetes"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
tapApi "github.com/up9inc/mizu/tap/api"
|
tapApi "github.com/up9inc/mizu/tap/api"
|
||||||
v1 "k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
"mizuserver/pkg/config"
|
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"mizuserver/pkg/providers/tapConfig"
|
|
||||||
"mizuserver/pkg/providers/tappedPods"
|
|
||||||
"mizuserver/pkg/providers/tappers"
|
|
||||||
"net/http"
|
|
||||||
"regexp"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var cancelTapperSyncer context.CancelFunc
|
var cancelTapperSyncer context.CancelFunc
|
||||||
|
@ -2,13 +2,14 @@ package controllers
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"mizuserver/pkg/validation"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/validation"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
basenine "github.com/up9inc/basenine/client/go"
|
basenine "github.com/up9inc/basenine/client/go"
|
||||||
@ -26,7 +27,7 @@ func InitExtensionsMap(ref map[string]*tapApi.Extension) {
|
|||||||
func Error(c *gin.Context, err error) bool {
|
func Error(c *gin.Context, err error) bool {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Log.Errorf("Error getting entry: %v", err)
|
logger.Log.Errorf("Error getting entry: %v", err)
|
||||||
c.Error(err)
|
_ = c.Error(err)
|
||||||
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{
|
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{
|
||||||
"error": true,
|
"error": true,
|
||||||
"type": "error",
|
"type": "error",
|
||||||
@ -131,7 +132,9 @@ func GetEntry(c *gin.Context) {
|
|||||||
_, rulesMatched, _isRulesEnabled := models.RunValidationRulesState(*harEntry, entry.Destination.Name)
|
_, rulesMatched, _isRulesEnabled := models.RunValidationRulesState(*harEntry, entry.Destination.Name)
|
||||||
isRulesEnabled = _isRulesEnabled
|
isRulesEnabled = _isRulesEnabled
|
||||||
inrec, _ := json.Marshal(rulesMatched)
|
inrec, _ := json.Marshal(rulesMatched)
|
||||||
json.Unmarshal(inrec, &rules)
|
if err := json.Unmarshal(inrec, &rules); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
c.JSON(http.StatusOK, tapApi.EntryWrapper{
|
c.JSON(http.StatusOK, tapApi.EntryWrapper{
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
"github.com/up9inc/mizu/shared"
|
|
||||||
"mizuserver/pkg/version"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/version"
|
||||||
|
"github.com/up9inc/mizu/shared"
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetVersion(c *gin.Context) {
|
func GetVersion(c *gin.Context) {
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
"github.com/chanced/openapi"
|
"github.com/chanced/openapi"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/oas"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
"mizuserver/pkg/oas"
|
|
||||||
"net/http"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetOASServers(c *gin.Context) {
|
func GetOASServers(c *gin.Context) {
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
"mizuserver/pkg/oas"
|
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/oas"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestGetOASServers(t *testing.T) {
|
func TestGetOASServers(t *testing.T) {
|
||||||
@ -15,7 +17,6 @@ func TestGetOASServers(t *testing.T) {
|
|||||||
|
|
||||||
GetOASServers(c)
|
GetOASServers(c)
|
||||||
t.Logf("Written body: %s", recorder.Body.String())
|
t.Logf("Written body: %s", recorder.Body.String())
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetOASAllSpecs(t *testing.T) {
|
func TestGetOASAllSpecs(t *testing.T) {
|
||||||
@ -26,7 +27,6 @@ func TestGetOASAllSpecs(t *testing.T) {
|
|||||||
|
|
||||||
GetOASAllSpecs(c)
|
GetOASAllSpecs(c)
|
||||||
t.Logf("Written body: %s", recorder.Body.String())
|
t.Logf("Written body: %s", recorder.Body.String())
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetOASSpec(t *testing.T) {
|
func TestGetOASSpec(t *testing.T) {
|
||||||
@ -39,5 +39,4 @@ func TestGetOASSpec(t *testing.T) {
|
|||||||
|
|
||||||
GetOASSpec(c)
|
GetOASSpec(c)
|
||||||
t.Logf("Written body: %s", recorder.Body.String())
|
t.Logf("Written body: %s", recorder.Body.String())
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/servicemap"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/servicemap"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@ import (
|
|||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"mizuserver/pkg/servicemap"
|
"github.com/up9inc/mizu/agent/pkg/servicemap"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/stretchr/testify/suite"
|
"github.com/stretchr/testify/suite"
|
||||||
|
@ -2,17 +2,18 @@ package controllers
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/api"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/holder"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappedPods"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/up9"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/validation"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
"mizuserver/pkg/api"
|
|
||||||
"mizuserver/pkg/holder"
|
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"mizuserver/pkg/providers/tappedPods"
|
|
||||||
"mizuserver/pkg/providers/tappers"
|
|
||||||
"mizuserver/pkg/up9"
|
|
||||||
"mizuserver/pkg/validation"
|
|
||||||
"net/http"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func HealthCheck(c *gin.Context) {
|
func HealthCheck(c *gin.Context) {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package controllers
|
package controllers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/providers"
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
|
@ -123,11 +123,11 @@ func NewRequest(request map[string]interface{}) (harRequest *Request, err error)
|
|||||||
cookies := make([]Cookie, 0) // BuildCookies(request["_cookies"].([]interface{}))
|
cookies := make([]Cookie, 0) // BuildCookies(request["_cookies"].([]interface{}))
|
||||||
|
|
||||||
postData, _ := request["postData"].(map[string]interface{})
|
postData, _ := request["postData"].(map[string]interface{})
|
||||||
mimeType, _ := postData["mimeType"]
|
mimeType := postData["mimeType"]
|
||||||
if mimeType == nil || len(mimeType.(string)) == 0 {
|
if mimeType == nil || len(mimeType.(string)) == 0 {
|
||||||
mimeType = "text/html"
|
mimeType = "text/html"
|
||||||
}
|
}
|
||||||
text, _ := postData["text"]
|
text := postData["text"]
|
||||||
postDataText := ""
|
postDataText := ""
|
||||||
if text != nil {
|
if text != nil {
|
||||||
postDataText = text.(string)
|
postDataText = text.(string)
|
||||||
@ -176,12 +176,12 @@ func NewResponse(response map[string]interface{}) (harResponse *Response, err er
|
|||||||
cookies := make([]Cookie, 0) // BuildCookies(response["_cookies"].([]interface{}))
|
cookies := make([]Cookie, 0) // BuildCookies(response["_cookies"].([]interface{}))
|
||||||
|
|
||||||
content, _ := response["content"].(map[string]interface{})
|
content, _ := response["content"].(map[string]interface{})
|
||||||
mimeType, _ := content["mimeType"]
|
mimeType := content["mimeType"]
|
||||||
if mimeType == nil || len(mimeType.(string)) == 0 {
|
if mimeType == nil || len(mimeType.(string)) == 0 {
|
||||||
mimeType = "text/html"
|
mimeType = "text/html"
|
||||||
}
|
}
|
||||||
encoding, _ := content["encoding"]
|
encoding := content["encoding"]
|
||||||
text, _ := content["text"]
|
text := content["text"]
|
||||||
bodyText := ""
|
bodyText := ""
|
||||||
if text != nil {
|
if text != nil {
|
||||||
bodyText = text.(string)
|
bodyText = text.(string)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package holder
|
package holder
|
||||||
|
|
||||||
import "mizuserver/pkg/resolver"
|
import "github.com/up9inc/mizu/agent/pkg/resolver"
|
||||||
|
|
||||||
var k8sResolver *resolver.Resolver
|
var k8sResolver *resolver.Resolver
|
||||||
|
|
||||||
@ -11,4 +11,3 @@ func SetResolver(param *resolver.Resolver) {
|
|||||||
func GetResolver() *resolver.Resolver {
|
func GetResolver() *resolver.Resolver {
|
||||||
return k8sResolver
|
return k8sResolver
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package middlewares
|
package middlewares
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/config"
|
"github.com/up9inc/mizu/agent/pkg/config"
|
||||||
"mizuserver/pkg/providers"
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
ory "github.com/ory/kratos-client-go"
|
ory "github.com/ory/kratos-client-go"
|
||||||
|
@ -2,9 +2,10 @@ package models
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/rules"
|
||||||
tapApi "github.com/up9inc/mizu/tap/api"
|
tapApi "github.com/up9inc/mizu/tap/api"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"mizuserver/pkg/rules"
|
|
||||||
|
|
||||||
basenine "github.com/up9inc/basenine/client/go"
|
basenine "github.com/up9inc/basenine/client/go"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
|
@ -4,19 +4,21 @@ import (
|
|||||||
"bufio"
|
"bufio"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
func getFiles(baseDir string) (result []string, err error) {
|
func getFiles(baseDir string) (result []string, err error) {
|
||||||
result = make([]string, 0, 0)
|
result = make([]string, 0)
|
||||||
logger.Log.Infof("Reading files from tree: %s", baseDir)
|
logger.Log.Infof("Reading files from tree: %s", baseDir)
|
||||||
|
|
||||||
// https://yourbasic.org/golang/list-files-in-directory/
|
// https://yourbasic.org/golang/list-files-in-directory/
|
||||||
@ -121,8 +123,6 @@ func feedEntry(entry *har.Entry, isSync bool) {
|
|||||||
|
|
||||||
if strings.Contains(entry.Request.URL, "taboola") {
|
if strings.Contains(entry.Request.URL, "taboola") {
|
||||||
logger.Log.Debugf("Interesting: %s", entry.Request.URL)
|
logger.Log.Debugf("Interesting: %s", entry.Request.URL)
|
||||||
} else {
|
|
||||||
//return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if isSync {
|
if isSync {
|
||||||
|
@ -33,11 +33,7 @@ func IsGibberish(str string) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
noise := noiseLevel(str)
|
noise := noiseLevel(str)
|
||||||
if noise >= 0.2 {
|
return noise >= 0.2
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func noiseLevel(str string) (score float64) {
|
func noiseLevel(str string) (score float64) {
|
||||||
|
@ -3,10 +3,11 @@ package oas
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"net/url"
|
"net/url"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -3,15 +3,17 @@ package oas
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"github.com/chanced/openapi"
|
|
||||||
"github.com/google/uuid"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mime"
|
"mime"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"net/url"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
|
||||||
|
"github.com/chanced/openapi"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
type reqResp struct { // hello, generics in Go
|
type reqResp struct { // hello, generics in Go
|
||||||
@ -272,7 +274,7 @@ func handleRequest(req *har.Request, opObj *openapi.Operation, isSuccess bool) e
|
|||||||
|
|
||||||
if reqBody != nil {
|
if reqBody != nil {
|
||||||
reqCtype := getReqCtype(req)
|
reqCtype := getReqCtype(req)
|
||||||
reqMedia, err := fillContent(reqResp{Req: req}, reqBody.Content, reqCtype, err)
|
reqMedia, err := fillContent(reqResp{Req: req}, reqBody.Content, reqCtype)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -294,7 +296,7 @@ func handleResponse(resp *har.Response, opObj *openapi.Operation, isSuccess bool
|
|||||||
|
|
||||||
respCtype := getRespCtype(resp)
|
respCtype := getRespCtype(resp)
|
||||||
respContent := respObj.Content
|
respContent := respObj.Content
|
||||||
respMedia, err := fillContent(reqResp{Resp: resp}, respContent, respCtype, err)
|
respMedia, err := fillContent(reqResp{Resp: resp}, respContent, respCtype)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -342,11 +344,9 @@ func handleRespHeaders(reqHeaders []har.Header, respObj *openapi.ResponseObj) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func fillContent(reqResp reqResp, respContent openapi.Content, ctype string, err error) (*openapi.MediaType, error) {
|
func fillContent(reqResp reqResp, respContent openapi.Content, ctype string) (*openapi.MediaType, error) {
|
||||||
content, found := respContent[ctype]
|
content, found := respContent[ctype]
|
||||||
if !found {
|
if !found {
|
||||||
respContent[ctype] = &openapi.MediaType{}
|
respContent[ctype] = &openapi.MediaType{}
|
||||||
@ -367,14 +367,16 @@ func fillContent(reqResp reqResp, respContent openapi.Content, ctype string, err
|
|||||||
any, isJSON := anyJSON(text)
|
any, isJSON := anyJSON(text)
|
||||||
if isJSON {
|
if isJSON {
|
||||||
// re-marshal with forced indent
|
// re-marshal with forced indent
|
||||||
exampleMsg, err = json.MarshalIndent(any, "", "\t")
|
if msg, err := json.MarshalIndent(any, "", "\t"); err != nil {
|
||||||
if err != nil {
|
|
||||||
panic("Failed to re-marshal value, super-strange")
|
panic("Failed to re-marshal value, super-strange")
|
||||||
|
} else {
|
||||||
|
exampleMsg = msg
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
exampleMsg, err = json.Marshal(text)
|
if msg, err := json.Marshal(text); err != nil {
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
return nil, err
|
||||||
|
} else {
|
||||||
|
exampleMsg = msg
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,15 +2,16 @@ package oas
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"github.com/chanced/openapi"
|
|
||||||
"github.com/op/go-logging"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/chanced/openapi"
|
||||||
|
"github.com/op/go-logging"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
// if started via env, write file into subdir
|
// if started via env, write file into subdir
|
||||||
@ -171,8 +172,6 @@ func loadStartingOAS() {
|
|||||||
gen.StartFromSpec(doc)
|
gen.StartFromSpec(doc)
|
||||||
|
|
||||||
GetOasGeneratorInstance().ServiceSpecs.Store("catalogue", gen)
|
GetOasGeneratorInstance().ServiceSpecs.Store("catalogue", gen)
|
||||||
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestEntriesNegative(t *testing.T) {
|
func TestEntriesNegative(t *testing.T) {
|
||||||
@ -224,5 +223,4 @@ func TestLoadValid3_1(t *testing.T) {
|
|||||||
t.Log(err)
|
t.Log(err)
|
||||||
t.FailNow()
|
t.FailNow()
|
||||||
}
|
}
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
package oas
|
package oas
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/chanced/openapi"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"net/url"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/chanced/openapi"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
type NodePath = []string
|
type NodePath = []string
|
||||||
@ -162,9 +163,7 @@ func (n *Node) listPaths() *openapi.Paths {
|
|||||||
strChunk = *n.constant
|
strChunk = *n.constant
|
||||||
} else if n.pathParam != nil {
|
} else if n.pathParam != nil {
|
||||||
strChunk = "{" + n.pathParam.Name + "}"
|
strChunk = "{" + n.pathParam.Name + "}"
|
||||||
} else {
|
} // else -> this is the root node
|
||||||
// this is the root node
|
|
||||||
}
|
|
||||||
|
|
||||||
// add self
|
// add self
|
||||||
if n.pathObj != nil {
|
if n.pathObj != nil {
|
||||||
|
@ -3,11 +3,13 @@ package oas
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"github.com/chanced/openapi"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
|
||||||
|
"github.com/chanced/openapi"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
func exampleResolver(ref string) (*openapi.ExampleObj, error) {
|
func exampleResolver(ref string) (*openapi.ExampleObj, error) {
|
||||||
@ -32,16 +34,14 @@ func headerResolver(ref string) (*openapi.HeaderObj, error) {
|
|||||||
|
|
||||||
func initParams(obj **openapi.ParameterList) {
|
func initParams(obj **openapi.ParameterList) {
|
||||||
if *obj == nil {
|
if *obj == nil {
|
||||||
var params openapi.ParameterList
|
var params openapi.ParameterList = make([]openapi.Parameter, 0)
|
||||||
params = make([]openapi.Parameter, 0)
|
|
||||||
*obj = ¶ms
|
*obj = ¶ms
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func initHeaders(respObj *openapi.ResponseObj) {
|
func initHeaders(respObj *openapi.ResponseObj) {
|
||||||
if respObj.Headers == nil {
|
if respObj.Headers == nil {
|
||||||
var created openapi.Headers
|
var created openapi.Headers = map[string]openapi.Header{}
|
||||||
created = map[string]openapi.Header{}
|
|
||||||
respObj.Headers = created
|
respObj.Headers = created
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -85,7 +85,7 @@ func findParamByName(params *openapi.ParameterList, in openapi.In, name string)
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if paramObj.Name == name || (caseInsensitive && strings.ToLower(paramObj.Name) == strings.ToLower(name)) {
|
if paramObj.Name == name || (caseInsensitive && strings.EqualFold(paramObj.Name, name)) {
|
||||||
pathParam = paramObj
|
pathParam = paramObj
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -102,7 +102,7 @@ func findHeaderByName(headers *openapi.Headers, name string) *openapi.HeaderObj
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if strings.ToLower(hname) == strings.ToLower(name) {
|
if strings.EqualFold(hname, name) {
|
||||||
return hdrObj
|
return hdrObj
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,10 @@ package providers
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"mizuserver/pkg/config"
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/config"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
|
|
||||||
ory "github.com/ory/kratos-client-go"
|
ory "github.com/ory/kratos-client-go"
|
||||||
)
|
)
|
||||||
@ -38,7 +41,9 @@ func CreateAdminUser(password string, ctx context.Context) (token *string, err e
|
|||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
//Delete the user to prevent a half-setup situation where admin user is created without admin privileges
|
//Delete the user to prevent a half-setup situation where admin user is created without admin privileges
|
||||||
DeleteUser(identityId, ctx)
|
if err := DeleteUser(identityId, ctx); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
return nil, err, nil
|
return nil, err, nil
|
||||||
}
|
}
|
||||||
|
@ -2,8 +2,9 @@ package providers_test
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/providers"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNoEntryAddedCount(t *testing.T) {
|
func TestNoEntryAddedCount(t *testing.T) {
|
||||||
|
@ -3,12 +3,13 @@ package providers
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/patrickmn/go-cache"
|
|
||||||
"github.com/up9inc/mizu/shared"
|
|
||||||
"github.com/up9inc/mizu/tap"
|
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"os"
|
"os"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/patrickmn/go-cache"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/shared"
|
||||||
|
"github.com/up9inc/mizu/tap"
|
||||||
)
|
)
|
||||||
|
|
||||||
const tlsLinkRetainmentTime = time.Minute * 15
|
const tlsLinkRetainmentTime = time.Minute * 15
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
package tapConfig
|
package tapConfig
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/up9inc/mizu/shared"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mizuserver/pkg/models"
|
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
"os"
|
"os"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/models"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
"github.com/up9inc/mizu/shared"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
const FilePath = shared.DataDirPath + "tap-config.json"
|
const FilePath = shared.DataDirPath + "tap-config.json"
|
||||||
|
@ -1,13 +1,14 @@
|
|||||||
package tappedPods
|
package tappedPods
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/up9inc/mizu/shared"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mizuserver/pkg/providers/tappers"
|
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/providers/tappers"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
"github.com/up9inc/mizu/shared"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
const FilePath = shared.DataDirPath + "tapped-pods.json"
|
const FilePath = shared.DataDirPath + "tapped-pods.json"
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
package tappers
|
package tappers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/up9inc/mizu/shared"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
"os"
|
"os"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
"github.com/up9inc/mizu/shared"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
const FilePath = shared.DataDirPath + "tappers-status.json"
|
const FilePath = shared.DataDirPath + "tappers-status.json"
|
||||||
|
@ -85,11 +85,11 @@ func DeleteUser(identityId string, ctx context.Context) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if result == nil {
|
if result == nil {
|
||||||
return errors.New("unknown error occured during user deletion")
|
return fmt.Errorf("unknown error occured during user deletion %v", identityId)
|
||||||
}
|
}
|
||||||
|
|
||||||
if result.StatusCode < 200 || result.StatusCode > 299 {
|
if result.StatusCode < 200 || result.StatusCode > 299 {
|
||||||
return errors.New(fmt.Sprintf("user deletion returned bad status %d", result.StatusCode))
|
return fmt.Errorf("user deletion %v returned bad status %d", identityId, result.StatusCode)
|
||||||
} else {
|
} else {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,8 @@ more on keto here: https://www.ory.sh/keto/docs/
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
|
||||||
ketoClient "github.com/ory/keto-client-go/client"
|
ketoClient "github.com/ory/keto-client-go/client"
|
||||||
ketoRead "github.com/ory/keto-client-go/client/read"
|
ketoRead "github.com/ory/keto-client-go/client/read"
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
"mizuserver/pkg/middlewares"
|
"github.com/up9inc/mizu/agent/pkg/middlewares"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"mizuserver/pkg/controllers"
|
"github.com/up9inc/mizu/agent/pkg/controllers"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
)
|
)
|
||||||
|
@ -4,11 +4,12 @@ import (
|
|||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
|
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
@ -108,7 +109,7 @@ func PassedValidationRules(rulesMatched []RulesMatched) (bool, int64, int) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, rule := range rulesMatched {
|
for _, rule := range rulesMatched {
|
||||||
if rule.Matched == false {
|
if !rule.Matched {
|
||||||
return false, responseTime, numberOfRulesMatched
|
return false, responseTime, numberOfRulesMatched
|
||||||
} else {
|
} else {
|
||||||
if strings.ToLower(rule.Rule.Type) == "slo" {
|
if strings.ToLower(rule.Rule.Type) == "slo" {
|
||||||
|
@ -6,8 +6,6 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"mizuserver/pkg/har"
|
|
||||||
"mizuserver/pkg/utils"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"regexp"
|
"regexp"
|
||||||
@ -15,6 +13,9 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/har"
|
||||||
|
"github.com/up9inc/mizu/agent/pkg/utils"
|
||||||
|
|
||||||
basenine "github.com/up9inc/basenine/client/go"
|
basenine "github.com/up9inc/basenine/client/go"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
|
@ -31,10 +31,14 @@ func StartServer(app *gin.Engine) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
_ = <-signals
|
<-signals
|
||||||
logger.Log.Infof("Shutting down...")
|
logger.Log.Infof("Shutting down...")
|
||||||
ctx, _ := context.WithTimeout(context.Background(), 5*time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||||
_ = srv.Shutdown(ctx)
|
defer cancel()
|
||||||
|
err := srv.Shutdown(ctx)
|
||||||
|
if err != nil {
|
||||||
|
logger.Log.Errorf("%v", err)
|
||||||
|
}
|
||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
}()
|
}()
|
||||||
|
|
||||||
@ -91,7 +95,7 @@ func UniqueStringSlice(s []string) []string {
|
|||||||
uniqueMap := map[string]bool{}
|
uniqueMap := map[string]bool{}
|
||||||
|
|
||||||
for _, val := range s {
|
for _, val := range s {
|
||||||
if uniqueMap[val] == true {
|
if uniqueMap[val] {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
uniqueMap[val] = true
|
uniqueMap[val] = true
|
||||||
|
@ -50,7 +50,9 @@ func init() {
|
|||||||
rootCmd.AddCommand(configCmd)
|
rootCmd.AddCommand(configCmd)
|
||||||
|
|
||||||
defaultConfig := config.ConfigStruct{}
|
defaultConfig := config.ConfigStruct{}
|
||||||
defaults.Set(&defaultConfig)
|
if err := defaults.Set(&defaultConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
configCmd.Flags().BoolP(configStructs.RegenerateConfigName, "r", defaultConfig.Config.Regenerate, fmt.Sprintf("Regenerate the config file with default values to path %s or to chosen path using --%s", defaultConfig.ConfigFilePath, config.ConfigFilePathCommandName))
|
configCmd.Flags().BoolP(configStructs.RegenerateConfigName, "r", defaultConfig.Config.Regenerate, fmt.Sprintf("Regenerate the config file with default values to path %s or to chosen path using --%s", defaultConfig.ConfigFilePath, config.ConfigFilePathCommandName))
|
||||||
}
|
}
|
||||||
|
@ -31,7 +31,9 @@ func runMizuInstall() {
|
|||||||
var defaultMaxEntriesDBSizeBytes int64 = 200 * 1000 * 1000
|
var defaultMaxEntriesDBSizeBytes int64 = 200 * 1000 * 1000
|
||||||
|
|
||||||
defaultResources := shared.Resources{}
|
defaultResources := shared.Resources{}
|
||||||
defaults.Set(&defaultResources)
|
if err := defaults.Set(&defaultResources); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
mizuAgentConfig := getInstallMizuAgentConfig(defaultMaxEntriesDBSizeBytes, defaultResources)
|
mizuAgentConfig := getInstallMizuAgentConfig(defaultMaxEntriesDBSizeBytes, defaultResources)
|
||||||
serializedMizuConfig, err := getSerializedMizuAgentConfig(mizuAgentConfig)
|
serializedMizuConfig, err := getSerializedMizuAgentConfig(mizuAgentConfig)
|
||||||
|
@ -43,7 +43,9 @@ func init() {
|
|||||||
rootCmd.AddCommand(logsCmd)
|
rootCmd.AddCommand(logsCmd)
|
||||||
|
|
||||||
defaultLogsConfig := configStructs.LogsConfig{}
|
defaultLogsConfig := configStructs.LogsConfig{}
|
||||||
defaults.Set(&defaultLogsConfig)
|
if err := defaults.Set(&defaultLogsConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
logsCmd.Flags().StringP(configStructs.FileLogsName, "f", defaultLogsConfig.FileStr, "Path for zip file (default current <pwd>\\mizu_logs.zip)")
|
logsCmd.Flags().StringP(configStructs.FileLogsName, "f", defaultLogsConfig.FileStr, "Path for zip file (default current <pwd>\\mizu_logs.zip)")
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,9 @@ Further info is available at https://github.com/up9inc/mizu`,
|
|||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
defaultConfig := config.ConfigStruct{}
|
defaultConfig := config.ConfigStruct{}
|
||||||
defaults.Set(&defaultConfig)
|
if err := defaults.Set(&defaultConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
rootCmd.PersistentFlags().StringSlice(config.SetCommandName, []string{}, fmt.Sprintf("Override values using --%s", config.SetCommandName))
|
rootCmd.PersistentFlags().StringSlice(config.SetCommandName, []string{}, fmt.Sprintf("Override values using --%s", config.SetCommandName))
|
||||||
rootCmd.PersistentFlags().String(config.ConfigFilePathCommandName, defaultConfig.ConfigFilePath, fmt.Sprintf("Override config file path using --%s", config.ConfigFilePathCommandName))
|
rootCmd.PersistentFlags().String(config.ConfigFilePathCommandName, defaultConfig.ConfigFilePath, fmt.Sprintf("Override config file path using --%s", config.ConfigFilePathCommandName))
|
||||||
|
@ -104,7 +104,9 @@ func init() {
|
|||||||
rootCmd.AddCommand(tapCmd)
|
rootCmd.AddCommand(tapCmd)
|
||||||
|
|
||||||
defaultTapConfig := configStructs.TapConfig{}
|
defaultTapConfig := configStructs.TapConfig{}
|
||||||
defaults.Set(&defaultTapConfig)
|
if err := defaults.Set(&defaultTapConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
tapCmd.Flags().Uint16P(configStructs.GuiPortTapName, "p", defaultTapConfig.GuiPort, "Provide a custom port for the web interface webserver")
|
tapCmd.Flags().Uint16P(configStructs.GuiPortTapName, "p", defaultTapConfig.GuiPort, "Provide a custom port for the web interface webserver")
|
||||||
tapCmd.Flags().StringSliceP(configStructs.NamespacesTapName, "n", defaultTapConfig.Namespaces, "Namespaces selector")
|
tapCmd.Flags().StringSliceP(configStructs.NamespacesTapName, "n", defaultTapConfig.Namespaces, "Namespaces selector")
|
||||||
|
@ -36,7 +36,9 @@ func init() {
|
|||||||
rootCmd.AddCommand(versionCmd)
|
rootCmd.AddCommand(versionCmd)
|
||||||
|
|
||||||
defaultVersionConfig := configStructs.VersionConfig{}
|
defaultVersionConfig := configStructs.VersionConfig{}
|
||||||
defaults.Set(&defaultVersionConfig)
|
if err := defaults.Set(&defaultVersionConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
versionCmd.Flags().BoolP(configStructs.DebugInfoVersionName, "d", defaultVersionConfig.DebugInfo, "Provide all information about version")
|
versionCmd.Flags().BoolP(configStructs.DebugInfoVersionName, "d", defaultVersionConfig.DebugInfo, "Provide all information about version")
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@ import (
|
|||||||
"github.com/up9inc/mizu/cli/config"
|
"github.com/up9inc/mizu/cli/config"
|
||||||
"github.com/up9inc/mizu/cli/config/configStructs"
|
"github.com/up9inc/mizu/cli/config/configStructs"
|
||||||
"github.com/up9inc/mizu/cli/telemetry"
|
"github.com/up9inc/mizu/cli/telemetry"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
var viewCmd = &cobra.Command{
|
var viewCmd = &cobra.Command{
|
||||||
@ -22,10 +23,14 @@ func init() {
|
|||||||
rootCmd.AddCommand(viewCmd)
|
rootCmd.AddCommand(viewCmd)
|
||||||
|
|
||||||
defaultViewConfig := configStructs.ViewConfig{}
|
defaultViewConfig := configStructs.ViewConfig{}
|
||||||
defaults.Set(&defaultViewConfig)
|
if err := defaults.Set(&defaultViewConfig); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
|
|
||||||
viewCmd.Flags().Uint16P(configStructs.GuiPortViewName, "p", defaultViewConfig.GuiPort, "Provide a custom port for the web interface webserver")
|
viewCmd.Flags().Uint16P(configStructs.GuiPortViewName, "p", defaultViewConfig.GuiPort, "Provide a custom port for the web interface webserver")
|
||||||
viewCmd.Flags().StringP(configStructs.UrlViewName, "u", defaultViewConfig.Url, "Provide a custom host")
|
viewCmd.Flags().StringP(configStructs.UrlViewName, "u", defaultViewConfig.Url, "Provide a custom host")
|
||||||
|
|
||||||
viewCmd.Flags().MarkHidden(configStructs.UrlViewName)
|
if err := viewCmd.Flags().MarkHidden(configStructs.UrlViewName); err != nil {
|
||||||
|
logger.Log.Debug(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -60,12 +60,12 @@ func (config *TapConfig) MaxEntriesDBSizeBytes() int64 {
|
|||||||
func (config *TapConfig) Validate() error {
|
func (config *TapConfig) Validate() error {
|
||||||
_, compileErr := regexp.Compile(config.PodRegexStr)
|
_, compileErr := regexp.Compile(config.PodRegexStr)
|
||||||
if compileErr != nil {
|
if compileErr != nil {
|
||||||
return errors.New(fmt.Sprintf("%s is not a valid regex %s", config.PodRegexStr, compileErr))
|
return fmt.Errorf("%s is not a valid regex %s", config.PodRegexStr, compileErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
_, parseHumanDataSizeErr := units.HumanReadableToBytes(config.HumanMaxEntriesDBSize)
|
_, parseHumanDataSizeErr := units.HumanReadableToBytes(config.HumanMaxEntriesDBSize)
|
||||||
if parseHumanDataSizeErr != nil {
|
if parseHumanDataSizeErr != nil {
|
||||||
return errors.New(fmt.Sprintf("Could not parse --%s value %s", HumanMaxEntriesDBSizeTapName, config.HumanMaxEntriesDBSize))
|
return fmt.Errorf("Could not parse --%s value %s", HumanMaxEntriesDBSizeTapName, config.HumanMaxEntriesDBSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
if config.Workspace != "" {
|
if config.Workspace != "" {
|
||||||
@ -76,7 +76,7 @@ func (config *TapConfig) Validate() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if config.Analysis && config.Workspace != "" {
|
if config.Analysis && config.Workspace != "" {
|
||||||
return errors.New(fmt.Sprintf("Can't run with both --%s and --%s flags", AnalysisTapName, WorkspaceTapName))
|
return fmt.Errorf("Can't run with both --%s and --%s flags", AnalysisTapName, WorkspaceTapName)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
package fsUtils
|
package fsUtils
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
)
|
)
|
||||||
@ -18,7 +17,7 @@ func EnsureDir(dirName string) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if !info.IsDir() {
|
if !info.IsDir() {
|
||||||
return errors.New(fmt.Sprintf("path exists but is not a directory: %s", dirName))
|
return fmt.Errorf("path exists but is not a directory: %s", dirName)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -23,9 +23,5 @@ func IsTokenValid(tokenString string, envName string) bool {
|
|||||||
}
|
}
|
||||||
defer response.Body.Close()
|
defer response.Body.Close()
|
||||||
|
|
||||||
if response.StatusCode != http.StatusOK {
|
return response.StatusCode == http.StatusOK
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
This document describes in details all permissions required for full and correct operation of Mizu.
|
This document describes in details all permissions required for full and correct operation of Mizu.
|
||||||
|
|
||||||
## Editting permissions
|
## Editing permissions
|
||||||
|
|
||||||
During installation, Mizu creates a `ServiceAccount` and the roles it requires. No further action is required.
|
During installation, Mizu creates a `ServiceAccount` and the roles it requires. No further action is required.
|
||||||
However, if there is a need, it is possible to make changes to Mizu permissions.
|
However, if there is a need, it is possible to make changes to Mizu permissions.
|
||||||
|
@ -44,7 +44,7 @@ func (d *Debouncer) SetOn() error {
|
|||||||
if d.canceled {
|
if d.canceled {
|
||||||
return fmt.Errorf("debouncer cancelled")
|
return fmt.Errorf("debouncer cancelled")
|
||||||
}
|
}
|
||||||
if d.running == true {
|
if d.running {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -93,6 +93,7 @@ github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDD
|
|||||||
github.com/docker/go-units v0.4.0 h1:3uh0PgVws3nIA0Q+MwDC8yjEPf9zjRfZZWXZYDct3Tw=
|
github.com/docker/go-units v0.4.0 h1:3uh0PgVws3nIA0Q+MwDC8yjEPf9zjRfZZWXZYDct3Tw=
|
||||||
github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||||
github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE=
|
github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE=
|
||||||
|
github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153 h1:yUdfgN0XgIJw7foRItutHYUIhlcKzcSf5vDpdhQAKTc=
|
||||||
github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
|
github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
|
||||||
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
||||||
github.com/emicklei/go-restful v2.9.5+incompatible/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
github.com/emicklei/go-restful v2.9.5+incompatible/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
||||||
@ -307,6 +308,7 @@ github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS4
|
|||||||
github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY=
|
github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY=
|
||||||
github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
||||||
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
||||||
|
github.com/moby/spdystream v0.2.0 h1:cjW1zVyyoiM0T7b6UoySUFqzXMoqRckQtXwGPiBhOM8=
|
||||||
github.com/moby/spdystream v0.2.0/go.mod h1:f7i0iNDQJ059oMTcWxx8MA/zKFIuD/lY+0GqbN2Wy8c=
|
github.com/moby/spdystream v0.2.0/go.mod h1:f7i0iNDQJ059oMTcWxx8MA/zKFIuD/lY+0GqbN2Wy8c=
|
||||||
github.com/moby/term v0.0.0-20201216013528-df9cb8a40635/go.mod h1:FBS0z0QWA44HXygs7VXDUOGoN/1TV3RuWkLO04am3wc=
|
github.com/moby/term v0.0.0-20201216013528-df9cb8a40635/go.mod h1:FBS0z0QWA44HXygs7VXDUOGoN/1TV3RuWkLO04am3wc=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
@ -34,7 +34,7 @@ func (wh *EventWatchHelper) Filter(wEvent *WatchEvent) (bool, error) {
|
|||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if strings.ToLower(event.Regarding.Kind) != strings.ToLower(wh.Kind) {
|
if strings.EqualFold(event.Regarding.Kind, wh.Kind) {
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,21 +3,22 @@ package kubernetes
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"regexp"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/op/go-logging"
|
"github.com/op/go-logging"
|
||||||
"github.com/up9inc/mizu/shared"
|
"github.com/up9inc/mizu/shared"
|
||||||
"github.com/up9inc/mizu/shared/debounce"
|
"github.com/up9inc/mizu/shared/debounce"
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
"github.com/up9inc/mizu/tap/api"
|
"github.com/up9inc/mizu/tap/api"
|
||||||
core "k8s.io/api/core/v1"
|
core "k8s.io/api/core/v1"
|
||||||
"regexp"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const updateTappersDelay = 5 * time.Second
|
const updateTappersDelay = 5 * time.Second
|
||||||
|
|
||||||
type TappedPodChangeEvent struct {
|
type TappedPodChangeEvent struct {
|
||||||
Added []core.Pod
|
Added []core.Pod
|
||||||
Removed []core.Pod
|
Removed []core.Pod
|
||||||
}
|
}
|
||||||
|
|
||||||
// MizuTapperSyncer uses a k8s pod watch to update tapper daemonsets when targeted pods are removed or created
|
// MizuTapperSyncer uses a k8s pod watch to update tapper daemonsets when targeted pods are removed or created
|
||||||
@ -222,10 +223,14 @@ func (tapperSyncer *MizuTapperSyncer) watchPodsForTapping() {
|
|||||||
switch wEvent.Type {
|
switch wEvent.Type {
|
||||||
case EventAdded:
|
case EventAdded:
|
||||||
logger.Log.Debugf("Added matching pod %s, ns: %s", pod.Name, pod.Namespace)
|
logger.Log.Debugf("Added matching pod %s, ns: %s", pod.Name, pod.Namespace)
|
||||||
restartTappersDebouncer.SetOn()
|
if err := restartTappersDebouncer.SetOn(); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
case EventDeleted:
|
case EventDeleted:
|
||||||
logger.Log.Debugf("Removed matching pod %s, ns: %s", pod.Name, pod.Namespace)
|
logger.Log.Debugf("Removed matching pod %s, ns: %s", pod.Name, pod.Namespace)
|
||||||
restartTappersDebouncer.SetOn()
|
if err := restartTappersDebouncer.SetOn(); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
case EventModified:
|
case EventModified:
|
||||||
logger.Log.Debugf("Modified matching pod %s, ns: %s, phase: %s, ip: %s", pod.Name, pod.Namespace, pod.Status.Phase, pod.Status.PodIP)
|
logger.Log.Debugf("Modified matching pod %s, ns: %s, phase: %s, ip: %s", pod.Name, pod.Namespace, pod.Status.Phase, pod.Status.PodIP)
|
||||||
// Act only if the modified pod has already obtained an IP address.
|
// Act only if the modified pod has already obtained an IP address.
|
||||||
@ -235,7 +240,9 @@ func (tapperSyncer *MizuTapperSyncer) watchPodsForTapping() {
|
|||||||
// - Pod reaches ready state
|
// - Pod reaches ready state
|
||||||
// Ready/unready transitions might also trigger this event.
|
// Ready/unready transitions might also trigger this event.
|
||||||
if pod.Status.PodIP != "" {
|
if pod.Status.PodIP != "" {
|
||||||
restartTappersDebouncer.SetOn()
|
if err := restartTappersDebouncer.SetOn(); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
case EventBookmark:
|
case EventBookmark:
|
||||||
break
|
break
|
||||||
@ -285,8 +292,8 @@ func (tapperSyncer *MizuTapperSyncer) updateCurrentlyTappedPods() (err error, ch
|
|||||||
tapperSyncer.CurrentlyTappedPods = podsToTap
|
tapperSyncer.CurrentlyTappedPods = podsToTap
|
||||||
tapperSyncer.nodeToTappedPodMap = GetNodeHostToTappedPodsMap(tapperSyncer.CurrentlyTappedPods)
|
tapperSyncer.nodeToTappedPodMap = GetNodeHostToTappedPodsMap(tapperSyncer.CurrentlyTappedPods)
|
||||||
tapperSyncer.TapPodChangesOut <- TappedPodChangeEvent{
|
tapperSyncer.TapPodChangesOut <- TappedPodChangeEvent{
|
||||||
Added: addedPods,
|
Added: addedPods,
|
||||||
Removed: removedPods,
|
Removed: removedPods,
|
||||||
}
|
}
|
||||||
return nil, true
|
return nil, true
|
||||||
}
|
}
|
||||||
|
@ -194,19 +194,19 @@ func (provider *Provider) GetMizuApiServerPodObject(opts *ApiServerOptions, moun
|
|||||||
|
|
||||||
cpuLimit, err := resource.ParseQuantity(opts.Resources.CpuLimit)
|
cpuLimit, err := resource.ParseQuantity(opts.Resources.CpuLimit)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.New(fmt.Sprintf("invalid cpu limit for %s container", opts.PodName))
|
return nil, fmt.Errorf("invalid cpu limit for %s container", opts.PodName)
|
||||||
}
|
}
|
||||||
memLimit, err := resource.ParseQuantity(opts.Resources.MemoryLimit)
|
memLimit, err := resource.ParseQuantity(opts.Resources.MemoryLimit)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.New(fmt.Sprintf("invalid memory limit for %s container", opts.PodName))
|
return nil, fmt.Errorf("invalid memory limit for %s container", opts.PodName)
|
||||||
}
|
}
|
||||||
cpuRequests, err := resource.ParseQuantity(opts.Resources.CpuRequests)
|
cpuRequests, err := resource.ParseQuantity(opts.Resources.CpuRequests)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.New(fmt.Sprintf("invalid cpu request for %s container", opts.PodName))
|
return nil, fmt.Errorf("invalid cpu request for %s container", opts.PodName)
|
||||||
}
|
}
|
||||||
memRequests, err := resource.ParseQuantity(opts.Resources.MemoryRequests)
|
memRequests, err := resource.ParseQuantity(opts.Resources.MemoryRequests)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.New(fmt.Sprintf("invalid memory request for %s container", opts.PodName))
|
return nil, fmt.Errorf("invalid memory request for %s container", opts.PodName)
|
||||||
}
|
}
|
||||||
|
|
||||||
command := []string{"./mizuagent", "--api-server"}
|
command := []string{"./mizuagent", "--api-server"}
|
||||||
@ -395,7 +395,7 @@ func (provider *Provider) CreatePod(ctx context.Context, namespace string, podSp
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (provider *Provider) CreateDeployment(ctx context.Context, namespace string, deploymentName string, podSpec *core.Pod) (*v1.Deployment, error) {
|
func (provider *Provider) CreateDeployment(ctx context.Context, namespace string, deploymentName string, podSpec *core.Pod) (*v1.Deployment, error) {
|
||||||
if _, keyExists := podSpec.ObjectMeta.Labels["app"]; keyExists == false {
|
if _, keyExists := podSpec.ObjectMeta.Labels["app"]; !keyExists {
|
||||||
return nil, errors.New("pod spec must contain 'app' label")
|
return nil, errors.New("pod spec must contain 'app' label")
|
||||||
}
|
}
|
||||||
podTemplate := &core.PodTemplateSpec{
|
podTemplate := &core.PodTemplateSpec{
|
||||||
@ -764,7 +764,7 @@ func (provider *Provider) handleRemovalError(err error) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (provider *Provider) CreateConfigMap(ctx context.Context, namespace string, configMapName string, serializedValidationRules string, serializedContract string, serializedMizuConfig string) error {
|
func (provider *Provider) CreateConfigMap(ctx context.Context, namespace string, configMapName string, serializedValidationRules string, serializedContract string, serializedMizuConfig string) error {
|
||||||
configMapData := make(map[string]string, 0)
|
configMapData := make(map[string]string)
|
||||||
if serializedValidationRules != "" {
|
if serializedValidationRules != "" {
|
||||||
configMapData[shared.ValidationRulesFileName] = serializedValidationRules
|
configMapData[shared.ValidationRulesFileName] = serializedValidationRules
|
||||||
}
|
}
|
||||||
@ -854,19 +854,19 @@ func (provider *Provider) ApplyMizuTapperDaemonSet(ctx context.Context, namespac
|
|||||||
)
|
)
|
||||||
cpuLimit, err := resource.ParseQuantity(resources.CpuLimit)
|
cpuLimit, err := resource.ParseQuantity(resources.CpuLimit)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New(fmt.Sprintf("invalid cpu limit for %s container", tapperPodName))
|
return fmt.Errorf("invalid cpu limit for %s container", tapperPodName)
|
||||||
}
|
}
|
||||||
memLimit, err := resource.ParseQuantity(resources.MemoryLimit)
|
memLimit, err := resource.ParseQuantity(resources.MemoryLimit)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New(fmt.Sprintf("invalid memory limit for %s container", tapperPodName))
|
return fmt.Errorf("invalid memory limit for %s container", tapperPodName)
|
||||||
}
|
}
|
||||||
cpuRequests, err := resource.ParseQuantity(resources.CpuRequests)
|
cpuRequests, err := resource.ParseQuantity(resources.CpuRequests)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New(fmt.Sprintf("invalid cpu request for %s container", tapperPodName))
|
return fmt.Errorf("invalid cpu request for %s container", tapperPodName)
|
||||||
}
|
}
|
||||||
memRequests, err := resource.ParseQuantity(resources.MemoryRequests)
|
memRequests, err := resource.ParseQuantity(resources.MemoryRequests)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New(fmt.Sprintf("invalid memory request for %s container", tapperPodName))
|
return fmt.Errorf("invalid memory request for %s container", tapperPodName)
|
||||||
}
|
}
|
||||||
agentResourceLimits := core.ResourceList{
|
agentResourceLimits := core.ResourceList{
|
||||||
"cpu": cpuLimit,
|
"cpu": cpuLimit,
|
||||||
|
@ -10,7 +10,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func GetNodeHostToTappedPodsMap(tappedPods []core.Pod) map[string][]core.Pod {
|
func GetNodeHostToTappedPodsMap(tappedPods []core.Pod) map[string][]core.Pod {
|
||||||
nodeToTappedPodMap := make(map[string][]core.Pod, 0)
|
nodeToTappedPodMap := make(map[string][]core.Pod)
|
||||||
for _, pod := range tappedPods {
|
for _, pod := range tappedPods {
|
||||||
minimizedPod := getMinimizedPod(pod)
|
minimizedPod := getMinimizedPod(pod)
|
||||||
|
|
||||||
|
@ -4,11 +4,12 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/up9inc/mizu/shared/debounce"
|
|
||||||
"github.com/up9inc/mizu/shared/logger"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/up9inc/mizu/shared/debounce"
|
||||||
|
"github.com/up9inc/mizu/shared/logger"
|
||||||
|
|
||||||
"k8s.io/apimachinery/pkg/watch"
|
"k8s.io/apimachinery/pkg/watch"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -31,7 +32,7 @@ func FilteredWatch(ctx context.Context, watcherCreator WatchCreator, targetNames
|
|||||||
|
|
||||||
go func(targetNamespace string) {
|
go func(targetNamespace string) {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
watchRestartDebouncer := debounce.NewDebouncer(1 * time.Minute, func() {})
|
watchRestartDebouncer := debounce.NewDebouncer(1*time.Minute, func() {})
|
||||||
|
|
||||||
for {
|
for {
|
||||||
watcher, err := watcherCreator.NewWatcher(ctx, targetNamespace)
|
watcher, err := watcherCreator.NewWatcher(ctx, targetNamespace)
|
||||||
@ -44,7 +45,7 @@ func FilteredWatch(ctx context.Context, watcherCreator WatchCreator, targetNames
|
|||||||
watcher.Stop()
|
watcher.Stop()
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case <- ctx.Done():
|
case <-ctx.Done():
|
||||||
return
|
return
|
||||||
default:
|
default:
|
||||||
break
|
break
|
||||||
@ -55,7 +56,9 @@ func FilteredWatch(ctx context.Context, watcherCreator WatchCreator, targetNames
|
|||||||
break
|
break
|
||||||
} else {
|
} else {
|
||||||
if !watchRestartDebouncer.IsOn() {
|
if !watchRestartDebouncer.IsOn() {
|
||||||
watchRestartDebouncer.SetOn()
|
if err := watchRestartDebouncer.SetOn(); err != nil {
|
||||||
|
logger.Log.Error(err)
|
||||||
|
}
|
||||||
logger.Log.Debug("k8s watch channel closed, restarting watcher")
|
logger.Log.Debug("k8s watch channel closed, restarting watcher")
|
||||||
time.Sleep(time.Second * 5)
|
time.Sleep(time.Second * 5)
|
||||||
continue
|
continue
|
||||||
|
@ -6,11 +6,12 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/google/martian/har"
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/martian/har"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Protocol struct {
|
type Protocol struct {
|
||||||
@ -282,7 +283,7 @@ func (h HTTPPayload) MarshalJSON() ([]byte, error) {
|
|||||||
RawResponse: &HTTPResponseWrapper{Response: h.Data.(*http.Response)},
|
RawResponse: &HTTPResponseWrapper{Response: h.Data.(*http.Response)},
|
||||||
})
|
})
|
||||||
default:
|
default:
|
||||||
panic(fmt.Sprintf("HTTP payload cannot be marshaled: %s", h.Type))
|
panic(fmt.Sprintf("HTTP payload cannot be marshaled: %v", h.Type))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -312,7 +313,7 @@ type HTTPRequestWrapper struct {
|
|||||||
func (r *HTTPRequestWrapper) MarshalJSON() ([]byte, error) {
|
func (r *HTTPRequestWrapper) MarshalJSON() ([]byte, error) {
|
||||||
body, _ := ioutil.ReadAll(r.Request.Body)
|
body, _ := ioutil.ReadAll(r.Request.Body)
|
||||||
r.Request.Body = ioutil.NopCloser(bytes.NewBuffer(body))
|
r.Request.Body = ioutil.NopCloser(bytes.NewBuffer(body))
|
||||||
return json.Marshal(&struct {
|
return json.Marshal(&struct { //nolint
|
||||||
Body string `json:"Body,omitempty"`
|
Body string `json:"Body,omitempty"`
|
||||||
GetBody string `json:"GetBody,omitempty"`
|
GetBody string `json:"GetBody,omitempty"`
|
||||||
Cancel string `json:"Cancel,omitempty"`
|
Cancel string `json:"Cancel,omitempty"`
|
||||||
@ -330,7 +331,7 @@ type HTTPResponseWrapper struct {
|
|||||||
func (r *HTTPResponseWrapper) MarshalJSON() ([]byte, error) {
|
func (r *HTTPResponseWrapper) MarshalJSON() ([]byte, error) {
|
||||||
body, _ := ioutil.ReadAll(r.Response.Body)
|
body, _ := ioutil.ReadAll(r.Response.Body)
|
||||||
r.Response.Body = ioutil.NopCloser(bytes.NewBuffer(body))
|
r.Response.Body = ioutil.NopCloser(bytes.NewBuffer(body))
|
||||||
return json.Marshal(&struct {
|
return json.Marshal(&struct { //nolint
|
||||||
Body string `json:"Body,omitempty"`
|
Body string `json:"Body,omitempty"`
|
||||||
GetBody string `json:"GetBody,omitempty"`
|
GetBody string `json:"GetBody,omitempty"`
|
||||||
Cancel string `json:"Cancel,omitempty"`
|
Cancel string `json:"Cancel,omitempty"`
|
||||||
|
@ -24,14 +24,14 @@ var connectionMethodMap = map[int]string{
|
|||||||
61: "connection unblocked",
|
61: "connection unblocked",
|
||||||
}
|
}
|
||||||
|
|
||||||
var channelMethodMap = map[int]string{
|
// var channelMethodMap = map[int]string{
|
||||||
10: "channel open",
|
// 10: "channel open",
|
||||||
11: "channel open-ok",
|
// 11: "channel open-ok",
|
||||||
20: "channel flow",
|
// 20: "channel flow",
|
||||||
21: "channel flow-ok",
|
// 21: "channel flow-ok",
|
||||||
40: "channel close",
|
// 40: "channel close",
|
||||||
41: "channel close-ok",
|
// 41: "channel close-ok",
|
||||||
}
|
// }
|
||||||
|
|
||||||
var exchangeMethodMap = map[int]string{
|
var exchangeMethodMap = map[int]string{
|
||||||
10: "exchange declare",
|
10: "exchange declare",
|
||||||
@ -78,14 +78,14 @@ var basicMethodMap = map[int]string{
|
|||||||
120: "basic nack",
|
120: "basic nack",
|
||||||
}
|
}
|
||||||
|
|
||||||
var txMethodMap = map[int]string{
|
// var txMethodMap = map[int]string{
|
||||||
10: "tx select",
|
// 10: "tx select",
|
||||||
11: "tx select-ok",
|
// 11: "tx select-ok",
|
||||||
20: "tx commit",
|
// 20: "tx commit",
|
||||||
21: "tx commit-ok",
|
// 21: "tx commit-ok",
|
||||||
30: "tx rollback",
|
// 30: "tx rollback",
|
||||||
31: "tx rollback-ok",
|
// 31: "tx rollback-ok",
|
||||||
}
|
// }
|
||||||
|
|
||||||
type AMQPWrapper struct {
|
type AMQPWrapper struct {
|
||||||
Method string `json:"method"`
|
Method string `json:"method"`
|
||||||
@ -550,14 +550,12 @@ func representConnectionStart(event map[string]interface{}) []interface{} {
|
|||||||
headers := make([]api.TableData, 0)
|
headers := make([]api.TableData, 0)
|
||||||
for name, value := range event["serverProperties"].(map[string]interface{}) {
|
for name, value := range event["serverProperties"].(map[string]interface{}) {
|
||||||
var outcome string
|
var outcome string
|
||||||
switch value.(type) {
|
switch v := value.(type) {
|
||||||
case string:
|
case string:
|
||||||
outcome = value.(string)
|
outcome = v
|
||||||
break
|
|
||||||
case map[string]interface{}:
|
case map[string]interface{}:
|
||||||
x, _ := json.Marshal(value)
|
x, _ := json.Marshal(value)
|
||||||
outcome = string(x)
|
outcome = string(x)
|
||||||
break
|
|
||||||
default:
|
default:
|
||||||
panic("Unknown data type for the server property!")
|
panic("Unknown data type for the server property!")
|
||||||
}
|
}
|
||||||
|
@ -47,7 +47,6 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
|
|
||||||
var remaining int
|
var remaining int
|
||||||
var header *HeaderFrame
|
var header *HeaderFrame
|
||||||
var body []byte
|
|
||||||
|
|
||||||
connectionInfo := &api.ConnectionInfo{
|
connectionInfo := &api.ConnectionInfo{
|
||||||
ClientIP: tcpID.SrcIP,
|
ClientIP: tcpID.SrcIP,
|
||||||
@ -102,13 +101,10 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
eventBasicPublish.Properties = header.Properties
|
eventBasicPublish.Properties = header.Properties
|
||||||
case *BasicDeliver:
|
case *BasicDeliver:
|
||||||
eventBasicDeliver.Properties = header.Properties
|
eventBasicDeliver.Properties = header.Properties
|
||||||
default:
|
|
||||||
frame = nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case *BodyFrame:
|
case *BodyFrame:
|
||||||
// continue until terminated
|
// continue until terminated
|
||||||
body = append(body, f.Body...)
|
|
||||||
remaining -= len(f.Body)
|
remaining -= len(f.Body)
|
||||||
switch lastMethodFrameMessage.(type) {
|
switch lastMethodFrameMessage.(type) {
|
||||||
case *BasicPublish:
|
case *BasicPublish:
|
||||||
@ -119,9 +115,6 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
eventBasicDeliver.Body = f.Body
|
eventBasicDeliver.Body = f.Body
|
||||||
superIdentifier.Protocol = &protocol
|
superIdentifier.Protocol = &protocol
|
||||||
emitAMQP(*eventBasicDeliver, amqpRequest, basicMethodMap[60], connectionInfo, superTimer.CaptureTime, emitter)
|
emitAMQP(*eventBasicDeliver, amqpRequest, basicMethodMap[60], connectionInfo, superTimer.CaptureTime, emitter)
|
||||||
default:
|
|
||||||
body = nil
|
|
||||||
frame = nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case *MethodFrame:
|
case *MethodFrame:
|
||||||
@ -211,10 +204,6 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
}
|
}
|
||||||
superIdentifier.Protocol = &protocol
|
superIdentifier.Protocol = &protocol
|
||||||
emitAMQP(*eventConnectionClose, amqpRequest, connectionMethodMap[50], connectionInfo, superTimer.CaptureTime, emitter)
|
emitAMQP(*eventConnectionClose, amqpRequest, connectionMethodMap[50], connectionInfo, superTimer.CaptureTime, emitter)
|
||||||
|
|
||||||
default:
|
|
||||||
frame = nil
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -231,32 +220,24 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
switch request["method"] {
|
switch request["method"] {
|
||||||
case basicMethodMap[40]:
|
case basicMethodMap[40]:
|
||||||
summary = reqDetails["exchange"].(string)
|
summary = reqDetails["exchange"].(string)
|
||||||
break
|
|
||||||
case basicMethodMap[60]:
|
case basicMethodMap[60]:
|
||||||
summary = reqDetails["exchange"].(string)
|
summary = reqDetails["exchange"].(string)
|
||||||
break
|
|
||||||
case exchangeMethodMap[10]:
|
case exchangeMethodMap[10]:
|
||||||
summary = reqDetails["exchange"].(string)
|
summary = reqDetails["exchange"].(string)
|
||||||
break
|
|
||||||
case queueMethodMap[10]:
|
case queueMethodMap[10]:
|
||||||
summary = reqDetails["queue"].(string)
|
summary = reqDetails["queue"].(string)
|
||||||
break
|
|
||||||
case connectionMethodMap[10]:
|
case connectionMethodMap[10]:
|
||||||
summary = fmt.Sprintf(
|
summary = fmt.Sprintf(
|
||||||
"%s.%s",
|
"%s.%s",
|
||||||
strconv.Itoa(int(reqDetails["versionMajor"].(float64))),
|
strconv.Itoa(int(reqDetails["versionMajor"].(float64))),
|
||||||
strconv.Itoa(int(reqDetails["versionMinor"].(float64))),
|
strconv.Itoa(int(reqDetails["versionMinor"].(float64))),
|
||||||
)
|
)
|
||||||
break
|
|
||||||
case connectionMethodMap[50]:
|
case connectionMethodMap[50]:
|
||||||
summary = reqDetails["replyText"].(string)
|
summary = reqDetails["replyText"].(string)
|
||||||
break
|
|
||||||
case queueMethodMap[20]:
|
case queueMethodMap[20]:
|
||||||
summary = reqDetails["queue"].(string)
|
summary = reqDetails["queue"].(string)
|
||||||
break
|
|
||||||
case basicMethodMap[20]:
|
case basicMethodMap[20]:
|
||||||
summary = reqDetails["queue"].(string)
|
summary = reqDetails["queue"].(string)
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
request["url"] = summary
|
request["url"] = summary
|
||||||
@ -288,33 +269,25 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
|
|
||||||
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
||||||
bodySize = 0
|
bodySize = 0
|
||||||
representation := make(map[string]interface{}, 0)
|
representation := make(map[string]interface{})
|
||||||
var repRequest []interface{}
|
var repRequest []interface{}
|
||||||
switch request["method"].(string) {
|
switch request["method"].(string) {
|
||||||
case basicMethodMap[40]:
|
case basicMethodMap[40]:
|
||||||
repRequest = representBasicPublish(request)
|
repRequest = representBasicPublish(request)
|
||||||
break
|
|
||||||
case basicMethodMap[60]:
|
case basicMethodMap[60]:
|
||||||
repRequest = representBasicDeliver(request)
|
repRequest = representBasicDeliver(request)
|
||||||
break
|
|
||||||
case queueMethodMap[10]:
|
case queueMethodMap[10]:
|
||||||
repRequest = representQueueDeclare(request)
|
repRequest = representQueueDeclare(request)
|
||||||
break
|
|
||||||
case exchangeMethodMap[10]:
|
case exchangeMethodMap[10]:
|
||||||
repRequest = representExchangeDeclare(request)
|
repRequest = representExchangeDeclare(request)
|
||||||
break
|
|
||||||
case connectionMethodMap[10]:
|
case connectionMethodMap[10]:
|
||||||
repRequest = representConnectionStart(request)
|
repRequest = representConnectionStart(request)
|
||||||
break
|
|
||||||
case connectionMethodMap[50]:
|
case connectionMethodMap[50]:
|
||||||
repRequest = representConnectionClose(request)
|
repRequest = representConnectionClose(request)
|
||||||
break
|
|
||||||
case queueMethodMap[20]:
|
case queueMethodMap[20]:
|
||||||
repRequest = representQueueBind(request)
|
repRequest = representQueueBind(request)
|
||||||
break
|
|
||||||
case basicMethodMap[20]:
|
case basicMethodMap[20]:
|
||||||
repRequest = representBasicConsume(request)
|
repRequest = representBasicConsume(request)
|
||||||
break
|
|
||||||
}
|
}
|
||||||
representation["request"] = repRequest
|
representation["request"] = repRequest
|
||||||
object, err = json.Marshal(representation)
|
object, err = json.Marshal(representation)
|
||||||
|
@ -135,20 +135,6 @@ func readDecimal(r io.Reader) (v Decimal, err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func readFloat32(r io.Reader) (v float32, err error) {
|
|
||||||
if err = binary.Read(r, binary.BigEndian, &v); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func readFloat64(r io.Reader) (v float64, err error) {
|
|
||||||
if err = binary.Read(r, binary.BigEndian, &v); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func readTimestamp(r io.Reader) (v time.Time, err error) {
|
func readTimestamp(r io.Reader) (v time.Time, err error) {
|
||||||
var sec int64
|
var sec int64
|
||||||
if err = binary.Read(r, binary.BigEndian, &sec); err != nil {
|
if err = binary.Read(r, binary.BigEndian, &sec); err != nil {
|
||||||
|
@ -78,15 +78,6 @@ type Error struct {
|
|||||||
Recover bool // true when this error can be recovered by retrying later or with different parameters
|
Recover bool // true when this error can be recovered by retrying later or with different parameters
|
||||||
}
|
}
|
||||||
|
|
||||||
func newError(code uint16, text string) *Error {
|
|
||||||
return &Error{
|
|
||||||
Code: int(code),
|
|
||||||
Reason: text,
|
|
||||||
Recover: isSoftExceptionCode(int(code)),
|
|
||||||
Server: true,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e Error) Error() string {
|
func (e Error) Error() string {
|
||||||
return fmt.Sprintf("Exception (%d) Reason: %q", e.Code, e.Reason)
|
return fmt.Sprintf("Exception (%d) Reason: %q", e.Code, e.Reason)
|
||||||
}
|
}
|
||||||
@ -262,19 +253,6 @@ func (t Table) Validate() error {
|
|||||||
return validateField(t)
|
return validateField(t)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Heap interface for maintaining delivery tags
|
|
||||||
type tagSet []uint64
|
|
||||||
|
|
||||||
func (set tagSet) Len() int { return len(set) }
|
|
||||||
func (set tagSet) Less(i, j int) bool { return (set)[i] < (set)[j] }
|
|
||||||
func (set tagSet) Swap(i, j int) { (set)[i], (set)[j] = (set)[j], (set)[i] }
|
|
||||||
func (set *tagSet) Push(tag interface{}) { *set = append(*set, tag.(uint64)) }
|
|
||||||
func (set *tagSet) Pop() interface{} {
|
|
||||||
val := (*set)[len(*set)-1]
|
|
||||||
*set = (*set)[:len(*set)-1]
|
|
||||||
return val
|
|
||||||
}
|
|
||||||
|
|
||||||
type Message interface {
|
type Message interface {
|
||||||
id() (uint16, uint16)
|
id() (uint16, uint16)
|
||||||
wait() bool
|
wait() bool
|
||||||
@ -282,12 +260,6 @@ type Message interface {
|
|||||||
write(io.Writer) error
|
write(io.Writer) error
|
||||||
}
|
}
|
||||||
|
|
||||||
type messageWithContent interface {
|
|
||||||
Message
|
|
||||||
getContent() (Properties, []byte)
|
|
||||||
setContent(Properties, []byte)
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
The base interface implemented as:
|
The base interface implemented as:
|
||||||
|
|
||||||
@ -322,22 +294,6 @@ type AmqpReader struct {
|
|||||||
R io.Reader
|
R io.Reader
|
||||||
}
|
}
|
||||||
|
|
||||||
type writer struct {
|
|
||||||
w io.Writer
|
|
||||||
}
|
|
||||||
|
|
||||||
// Implements the frame interface for Connection RPC
|
|
||||||
type protocolHeader struct{}
|
|
||||||
|
|
||||||
func (protocolHeader) write(w io.Writer) error {
|
|
||||||
_, err := w.Write([]byte{'A', 'M', 'Q', 'P', 0, 0, 9, 1})
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (protocolHeader) channel() uint16 {
|
|
||||||
panic("only valid as initial handshake")
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Method frames carry the high-level protocol commands (which we call "methods").
|
Method frames carry the high-level protocol commands (which we call "methods").
|
||||||
One method frame carries one command. The method frame payload has this format:
|
One method frame carries one command. The method frame payload has this format:
|
||||||
|
@ -6,7 +6,6 @@
|
|||||||
package amqp
|
package amqp
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"errors"
|
"errors"
|
||||||
@ -15,18 +14,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (w *writer) WriteFrame(frame frame) (err error) {
|
|
||||||
if err = frame.write(w.w); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if buf, ok := w.w.(*bufio.Writer); ok {
|
|
||||||
err = buf.Flush()
|
|
||||||
}
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *MethodFrame) write(w io.Writer) (err error) {
|
func (f *MethodFrame) write(w io.Writer) (err error) {
|
||||||
var payload bytes.Buffer
|
var payload bytes.Buffer
|
||||||
|
|
||||||
@ -412,5 +399,5 @@ func writeTable(w io.Writer, table Table) (err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return writeLongstr(w, string(buf.Bytes()))
|
return writeLongstr(w, buf.String())
|
||||||
}
|
}
|
||||||
|
@ -81,7 +81,7 @@ func representSliceAsTable(slice []interface{}, selectorPrefix string) (represen
|
|||||||
selector := fmt.Sprintf("%s[%d]", selectorPrefix, i)
|
selector := fmt.Sprintf("%s[%d]", selectorPrefix, i)
|
||||||
table = append(table, api.TableData{
|
table = append(table, api.TableData{
|
||||||
Name: strconv.Itoa(i),
|
Name: strconv.Itoa(i),
|
||||||
Value: item.(interface{}),
|
Value: item,
|
||||||
Selector: selector,
|
Selector: selector,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -92,11 +92,15 @@ func (d dissecting) Ping() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, counterPair *api.CounterPair, superTimer *api.SuperTimer, superIdentifier *api.SuperIdentifier, emitter api.Emitter, options *api.TrafficFilteringOptions) error {
|
func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, counterPair *api.CounterPair, superTimer *api.SuperTimer, superIdentifier *api.SuperIdentifier, emitter api.Emitter, options *api.TrafficFilteringOptions) error {
|
||||||
isHTTP2, err := checkIsHTTP2Connection(b, isClient)
|
var err error
|
||||||
|
isHTTP2, _ := checkIsHTTP2Connection(b, isClient)
|
||||||
|
|
||||||
var http2Assembler *Http2Assembler
|
var http2Assembler *Http2Assembler
|
||||||
if isHTTP2 {
|
if isHTTP2 {
|
||||||
prepareHTTP2Connection(b, isClient)
|
err = prepareHTTP2Connection(b, isClient)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
http2Assembler = createHTTP2Assembler(b)
|
http2Assembler = createHTTP2Assembler(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -105,7 +109,13 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
if switchingProtocolsHTTP2 {
|
if switchingProtocolsHTTP2 {
|
||||||
switchingProtocolsHTTP2 = false
|
switchingProtocolsHTTP2 = false
|
||||||
isHTTP2, err = checkIsHTTP2Connection(b, isClient)
|
isHTTP2, err = checkIsHTTP2Connection(b, isClient)
|
||||||
prepareHTTP2Connection(b, isClient)
|
if err != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
err = prepareHTTP2Connection(b, isClient)
|
||||||
|
if err != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
http2Assembler = createHTTP2Assembler(b)
|
http2Assembler = createHTTP2Assembler(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -340,11 +350,11 @@ func representRequest(request map[string]interface{}) (repRequest []interface{})
|
|||||||
})
|
})
|
||||||
|
|
||||||
postData, _ := request["postData"].(map[string]interface{})
|
postData, _ := request["postData"].(map[string]interface{})
|
||||||
mimeType, _ := postData["mimeType"]
|
mimeType := postData["mimeType"]
|
||||||
if mimeType == nil || len(mimeType.(string)) == 0 {
|
if mimeType == nil || len(mimeType.(string)) == 0 {
|
||||||
mimeType = "text/html"
|
mimeType = "text/html"
|
||||||
}
|
}
|
||||||
text, _ := postData["text"]
|
text := postData["text"]
|
||||||
if text != nil {
|
if text != nil {
|
||||||
repRequest = append(repRequest, api.SectionData{
|
repRequest = append(repRequest, api.SectionData{
|
||||||
Type: api.BODY,
|
Type: api.BODY,
|
||||||
@ -424,12 +434,12 @@ func representResponse(response map[string]interface{}) (repResponse []interface
|
|||||||
})
|
})
|
||||||
|
|
||||||
content, _ := response["content"].(map[string]interface{})
|
content, _ := response["content"].(map[string]interface{})
|
||||||
mimeType, _ := content["mimeType"]
|
mimeType := content["mimeType"]
|
||||||
if mimeType == nil || len(mimeType.(string)) == 0 {
|
if mimeType == nil || len(mimeType.(string)) == 0 {
|
||||||
mimeType = "text/html"
|
mimeType = "text/html"
|
||||||
}
|
}
|
||||||
encoding, _ := content["encoding"]
|
encoding := content["encoding"]
|
||||||
text, _ := content["text"]
|
text := content["text"]
|
||||||
if text != nil {
|
if text != nil {
|
||||||
repResponse = append(repResponse, api.SectionData{
|
repResponse = append(repResponse, api.SectionData{
|
||||||
Type: api.BODY,
|
Type: api.BODY,
|
||||||
@ -445,7 +455,7 @@ func representResponse(response map[string]interface{}) (repResponse []interface
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
||||||
representation := make(map[string]interface{}, 0)
|
representation := make(map[string]interface{})
|
||||||
repRequest := representRequest(request)
|
repRequest := representRequest(request)
|
||||||
repResponse, bodySize := representResponse(response)
|
repResponse, bodySize := representResponse(response)
|
||||||
representation["request"] = repRequest
|
representation["request"] = repRequest
|
||||||
|
@ -89,7 +89,7 @@ func filterResponseBody(response *http.Response, options *api.TrafficFilteringOp
|
|||||||
}
|
}
|
||||||
|
|
||||||
func filterHeaders(headers *http.Header) {
|
func filterHeaders(headers *http.Header) {
|
||||||
for key, _ := range *headers {
|
for key := range *headers {
|
||||||
if strings.ToLower(key) == userAgent {
|
if strings.ToLower(key) == userAgent {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -103,7 +103,7 @@ func filterHeaders(headers *http.Header) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func getContentTypeHeaderValue(headers http.Header) string {
|
func getContentTypeHeaderValue(headers http.Header) string {
|
||||||
for key, _ := range headers {
|
for key := range headers {
|
||||||
if strings.ToLower(key) == "content-type" {
|
if strings.ToLower(key) == "content-type" {
|
||||||
return headers.Get(key)
|
return headers.Get(key)
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,6 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"math"
|
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
)
|
)
|
||||||
@ -201,25 +200,6 @@ func newPageBuffer() *pageBuffer {
|
|||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (pb *pageBuffer) refTo(ref *pageRef, begin, end int64) {
|
|
||||||
length := end - begin
|
|
||||||
|
|
||||||
if length > math.MaxUint32 {
|
|
||||||
panic("reference to contiguous buffer pages exceeds the maximum size of 4 GB")
|
|
||||||
}
|
|
||||||
|
|
||||||
ref.pages = append(ref.buffer[:0], pb.pages.slice(begin, end)...)
|
|
||||||
ref.pages.ref()
|
|
||||||
ref.offset = begin
|
|
||||||
ref.length = uint32(length)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (pb *pageBuffer) ref(begin, end int64) *pageRef {
|
|
||||||
ref := new(pageRef)
|
|
||||||
pb.refTo(ref, begin, end)
|
|
||||||
return ref
|
|
||||||
}
|
|
||||||
|
|
||||||
func (pb *pageBuffer) unref() {
|
func (pb *pageBuffer) unref() {
|
||||||
pb.refc.unref(func() {
|
pb.refc.unref(func() {
|
||||||
pb.pages.unref()
|
pb.pages.unref()
|
||||||
@ -353,12 +333,12 @@ func (pb *pageBuffer) Write(b []byte) (int, error) {
|
|||||||
free := tail.Cap() - tail.Len()
|
free := tail.Cap() - tail.Len()
|
||||||
|
|
||||||
if len(b) <= free {
|
if len(b) <= free {
|
||||||
tail.Write(b)
|
_, _ = tail.Write(b)
|
||||||
pb.length += len(b)
|
pb.length += len(b)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
tail.Write(b[:free])
|
_, _ = tail.Write(b[:free])
|
||||||
b = b[free:]
|
b = b[free:]
|
||||||
|
|
||||||
pb.length += free
|
pb.length += free
|
||||||
@ -374,7 +354,7 @@ func (pb *pageBuffer) WriteAt(b []byte, off int64) (int, error) {
|
|||||||
return n, err
|
return n, err
|
||||||
}
|
}
|
||||||
if n < len(b) {
|
if n < len(b) {
|
||||||
pb.Write(b[n:])
|
_, _ = pb.Write(b[n:])
|
||||||
}
|
}
|
||||||
return len(b), nil
|
return len(b), nil
|
||||||
}
|
}
|
||||||
@ -406,12 +386,6 @@ var (
|
|||||||
|
|
||||||
type contiguousPages []*page
|
type contiguousPages []*page
|
||||||
|
|
||||||
func (pages contiguousPages) ref() {
|
|
||||||
for _, p := range pages {
|
|
||||||
p.ref()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (pages contiguousPages) unref() {
|
func (pages contiguousPages) unref() {
|
||||||
for _, p := range pages {
|
for _, p := range pages {
|
||||||
p.unref()
|
p.unref()
|
||||||
@ -480,7 +454,6 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type pageRef struct {
|
type pageRef struct {
|
||||||
buffer [2]*page
|
|
||||||
pages contiguousPages
|
pages contiguousPages
|
||||||
offset int64
|
offset int64
|
||||||
cursor int64
|
cursor int64
|
||||||
@ -590,28 +563,6 @@ var (
|
|||||||
_ io.WriterTo = (*pageRef)(nil)
|
_ io.WriterTo = (*pageRef)(nil)
|
||||||
)
|
)
|
||||||
|
|
||||||
type pageRefAllocator struct {
|
|
||||||
refs []pageRef
|
|
||||||
head int
|
|
||||||
size int
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *pageRefAllocator) newPageRef() *pageRef {
|
|
||||||
if a.head == len(a.refs) {
|
|
||||||
a.refs = make([]pageRef, a.size)
|
|
||||||
a.head = 0
|
|
||||||
}
|
|
||||||
ref := &a.refs[a.head]
|
|
||||||
a.head++
|
|
||||||
return ref
|
|
||||||
}
|
|
||||||
|
|
||||||
func unref(x interface{}) {
|
|
||||||
if r, _ := x.(interface{ unref() }); r != nil {
|
|
||||||
r.unref()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func seek(cursor, limit, offset int64, whence int) (int64, error) {
|
func seek(cursor, limit, offset int64, whence int) (int64, error) {
|
||||||
switch whence {
|
switch whence {
|
||||||
case io.SeekStart:
|
case io.SeekStart:
|
||||||
@ -631,15 +582,3 @@ func seek(cursor, limit, offset int64, whence int) (int64, error) {
|
|||||||
}
|
}
|
||||||
return offset, nil
|
return offset, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func closeBytes(b Bytes) {
|
|
||||||
if b != nil {
|
|
||||||
b.Close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func resetBytes(b Bytes) {
|
|
||||||
if r, _ := b.(interface{ Reset() }); r != nil {
|
|
||||||
r.Reset()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
package kafka
|
package kafka
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
|
|
||||||
"github.com/segmentio/kafka-go/compress"
|
"github.com/segmentio/kafka-go/compress"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -12,19 +10,3 @@ type CompressionCodec = compress.Codec
|
|||||||
|
|
||||||
// TODO: this file should probably go away once the internals of the package
|
// TODO: this file should probably go away once the internals of the package
|
||||||
// have moved to use the protocol package.
|
// have moved to use the protocol package.
|
||||||
const (
|
|
||||||
compressionCodecMask = 0x07
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
errUnknownCodec = errors.New("the compression code is invalid or its codec has not been imported")
|
|
||||||
)
|
|
||||||
|
|
||||||
// resolveCodec looks up a codec by Code()
|
|
||||||
func resolveCodec(code int8) (CompressionCodec, error) {
|
|
||||||
codec := compress.Compression(code).Codec()
|
|
||||||
if codec == nil {
|
|
||||||
return nil, errUnknownCodec
|
|
||||||
}
|
|
||||||
return codec, nil
|
|
||||||
}
|
|
||||||
|
@ -58,14 +58,6 @@ func (d *decoder) ReadByte() (byte, error) {
|
|||||||
return c, d.err
|
return c, d.err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *decoder) done() bool {
|
|
||||||
return d.remain == 0 || d.err != nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) setCRC(table *crc32.Table) {
|
|
||||||
d.table, d.crc32 = table, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) decodeBool(v value) {
|
func (d *decoder) decodeBool(v value) {
|
||||||
v.setBool(d.readBool())
|
v.setBool(d.readBool())
|
||||||
}
|
}
|
||||||
@ -199,19 +191,6 @@ func (d *decoder) read(n int) []byte {
|
|||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *decoder) writeTo(w io.Writer, n int) {
|
|
||||||
limit := d.remain
|
|
||||||
if n < limit {
|
|
||||||
d.remain = n
|
|
||||||
}
|
|
||||||
c, err := io.Copy(w, d)
|
|
||||||
if int(c) < n && err == nil {
|
|
||||||
err = io.ErrUnexpectedEOF
|
|
||||||
}
|
|
||||||
d.remain = limit - int(c)
|
|
||||||
d.setError(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) setError(err error) {
|
func (d *decoder) setError(err error) {
|
||||||
if d.err == nil && err != nil {
|
if d.err == nil && err != nil {
|
||||||
d.err = err
|
d.err = err
|
||||||
@ -272,14 +251,6 @@ func (d *decoder) readString() string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *decoder) readVarString() string {
|
|
||||||
if n := d.readVarInt(); n < 0 {
|
|
||||||
return ""
|
|
||||||
} else {
|
|
||||||
return bytesToString(d.read(int(n)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) readCompactString() string {
|
func (d *decoder) readCompactString() string {
|
||||||
if n := d.readUnsignedVarInt(); n < 1 {
|
if n := d.readUnsignedVarInt(); n < 1 {
|
||||||
return ""
|
return ""
|
||||||
@ -296,32 +267,6 @@ func (d *decoder) readBytes() []byte {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *decoder) readBytesTo(w io.Writer) bool {
|
|
||||||
if n := d.readInt32(); n < 0 {
|
|
||||||
return false
|
|
||||||
} else {
|
|
||||||
d.writeTo(w, int(n))
|
|
||||||
return d.err == nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) readVarBytes() []byte {
|
|
||||||
if n := d.readVarInt(); n < 0 {
|
|
||||||
return nil
|
|
||||||
} else {
|
|
||||||
return d.read(int(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) readVarBytesTo(w io.Writer) bool {
|
|
||||||
if n := d.readVarInt(); n < 0 {
|
|
||||||
return false
|
|
||||||
} else {
|
|
||||||
d.writeTo(w, int(n))
|
|
||||||
return d.err == nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) readCompactBytes() []byte {
|
func (d *decoder) readCompactBytes() []byte {
|
||||||
if n := d.readUnsignedVarInt(); n < 1 {
|
if n := d.readUnsignedVarInt(); n < 1 {
|
||||||
return nil
|
return nil
|
||||||
@ -330,15 +275,6 @@ func (d *decoder) readCompactBytes() []byte {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *decoder) readCompactBytesTo(w io.Writer) bool {
|
|
||||||
if n := d.readUnsignedVarInt(); n < 1 {
|
|
||||||
return false
|
|
||||||
} else {
|
|
||||||
d.writeTo(w, int(n-1))
|
|
||||||
return d.err == nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *decoder) readVarInt() int64 {
|
func (d *decoder) readVarInt() int64 {
|
||||||
n := 11 // varints are at most 11 bytes
|
n := 11 // varints are at most 11 bytes
|
||||||
|
|
||||||
|
@ -14,37 +14,3 @@ func discardN(r *bufio.Reader, sz int, n int) (int, error) {
|
|||||||
}
|
}
|
||||||
return sz - n, err
|
return sz - n, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func discardInt8(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return discardN(r, sz, 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func discardInt16(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return discardN(r, sz, 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
func discardInt32(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return discardN(r, sz, 4)
|
|
||||||
}
|
|
||||||
|
|
||||||
func discardInt64(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return discardN(r, sz, 8)
|
|
||||||
}
|
|
||||||
|
|
||||||
func discardString(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return readStringWith(r, sz, func(r *bufio.Reader, sz int, n int) (int, error) {
|
|
||||||
if n < 0 {
|
|
||||||
return sz, nil
|
|
||||||
}
|
|
||||||
return discardN(r, sz, n)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func discardBytes(r *bufio.Reader, sz int) (int, error) {
|
|
||||||
return readBytesWith(r, sz, func(r *bufio.Reader, sz int, n int) (int, error) {
|
|
||||||
if n < 0 {
|
|
||||||
return sz, nil
|
|
||||||
}
|
|
||||||
return discardN(r, sz, n)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
@ -3,7 +3,6 @@ package kafka
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"fmt"
|
|
||||||
"hash/crc32"
|
"hash/crc32"
|
||||||
"io"
|
"io"
|
||||||
"reflect"
|
"reflect"
|
||||||
@ -95,10 +94,6 @@ func (e *encoder) WriteString(s string) (int, error) {
|
|||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) setCRC(table *crc32.Table) {
|
|
||||||
e.table, e.crc32 = table, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) update(b []byte) {
|
func (e *encoder) update(b []byte) {
|
||||||
if e.table != nil {
|
if e.table != nil {
|
||||||
e.crc32 = crc32.Update(e.crc32, e.table, b)
|
e.crc32 = crc32.Update(e.crc32, e.table, b)
|
||||||
@ -133,10 +128,6 @@ func (e *encoder) encodeString(v value) {
|
|||||||
e.writeString(v.string())
|
e.writeString(v.string())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) encodeVarString(v value) {
|
|
||||||
e.writeVarString(v.string())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) encodeCompactString(v value) {
|
func (e *encoder) encodeCompactString(v value) {
|
||||||
e.writeCompactString(v.string())
|
e.writeCompactString(v.string())
|
||||||
}
|
}
|
||||||
@ -145,10 +136,6 @@ func (e *encoder) encodeNullString(v value) {
|
|||||||
e.writeNullString(v.string())
|
e.writeNullString(v.string())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) encodeVarNullString(v value) {
|
|
||||||
e.writeVarNullString(v.string())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) encodeCompactNullString(v value) {
|
func (e *encoder) encodeCompactNullString(v value) {
|
||||||
e.writeCompactNullString(v.string())
|
e.writeCompactNullString(v.string())
|
||||||
}
|
}
|
||||||
@ -157,10 +144,6 @@ func (e *encoder) encodeBytes(v value) {
|
|||||||
e.writeBytes(v.bytes())
|
e.writeBytes(v.bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) encodeVarBytes(v value) {
|
|
||||||
e.writeVarBytes(v.bytes())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) encodeCompactBytes(v value) {
|
func (e *encoder) encodeCompactBytes(v value) {
|
||||||
e.writeCompactBytes(v.bytes())
|
e.writeCompactBytes(v.bytes())
|
||||||
}
|
}
|
||||||
@ -169,10 +152,6 @@ func (e *encoder) encodeNullBytes(v value) {
|
|||||||
e.writeNullBytes(v.bytes())
|
e.writeNullBytes(v.bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) encodeVarNullBytes(v value) {
|
|
||||||
e.writeVarNullBytes(v.bytes())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) encodeCompactNullBytes(v value) {
|
func (e *encoder) encodeCompactNullBytes(v value) {
|
||||||
e.writeCompactNullBytes(v.bytes())
|
e.writeCompactNullBytes(v.bytes())
|
||||||
}
|
}
|
||||||
@ -228,37 +207,32 @@ func (e *encoder) encodeCompactNullArray(v value, elemType reflect.Type, encodeE
|
|||||||
|
|
||||||
func (e *encoder) writeInt8(i int8) {
|
func (e *encoder) writeInt8(i int8) {
|
||||||
writeInt8(e.buffer[:1], i)
|
writeInt8(e.buffer[:1], i)
|
||||||
e.Write(e.buffer[:1])
|
_, _ = e.Write(e.buffer[:1])
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeInt16(i int16) {
|
func (e *encoder) writeInt16(i int16) {
|
||||||
writeInt16(e.buffer[:2], i)
|
writeInt16(e.buffer[:2], i)
|
||||||
e.Write(e.buffer[:2])
|
_, _ = e.Write(e.buffer[:2])
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeInt32(i int32) {
|
func (e *encoder) writeInt32(i int32) {
|
||||||
writeInt32(e.buffer[:4], i)
|
writeInt32(e.buffer[:4], i)
|
||||||
e.Write(e.buffer[:4])
|
_, _ = e.Write(e.buffer[:4])
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeInt64(i int64) {
|
func (e *encoder) writeInt64(i int64) {
|
||||||
writeInt64(e.buffer[:8], i)
|
writeInt64(e.buffer[:8], i)
|
||||||
e.Write(e.buffer[:8])
|
_, _ = e.Write(e.buffer[:8])
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeString(s string) {
|
func (e *encoder) writeString(s string) {
|
||||||
e.writeInt16(int16(len(s)))
|
e.writeInt16(int16(len(s)))
|
||||||
e.WriteString(s)
|
_, _ = e.WriteString(s)
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarString(s string) {
|
|
||||||
e.writeVarInt(int64(len(s)))
|
|
||||||
e.WriteString(s)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeCompactString(s string) {
|
func (e *encoder) writeCompactString(s string) {
|
||||||
e.writeUnsignedVarInt(uint64(len(s)) + 1)
|
e.writeUnsignedVarInt(uint64(len(s)) + 1)
|
||||||
e.WriteString(s)
|
_, _ = e.WriteString(s)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeNullString(s string) {
|
func (e *encoder) writeNullString(s string) {
|
||||||
@ -266,16 +240,7 @@ func (e *encoder) writeNullString(s string) {
|
|||||||
e.writeInt16(-1)
|
e.writeInt16(-1)
|
||||||
} else {
|
} else {
|
||||||
e.writeInt16(int16(len(s)))
|
e.writeInt16(int16(len(s)))
|
||||||
e.WriteString(s)
|
_, _ = e.WriteString(s)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarNullString(s string) {
|
|
||||||
if s == "" {
|
|
||||||
e.writeVarInt(-1)
|
|
||||||
} else {
|
|
||||||
e.writeVarInt(int64(len(s)))
|
|
||||||
e.WriteString(s)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -284,23 +249,18 @@ func (e *encoder) writeCompactNullString(s string) {
|
|||||||
e.writeUnsignedVarInt(0)
|
e.writeUnsignedVarInt(0)
|
||||||
} else {
|
} else {
|
||||||
e.writeUnsignedVarInt(uint64(len(s)) + 1)
|
e.writeUnsignedVarInt(uint64(len(s)) + 1)
|
||||||
e.WriteString(s)
|
_, _ = e.WriteString(s)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeBytes(b []byte) {
|
func (e *encoder) writeBytes(b []byte) {
|
||||||
e.writeInt32(int32(len(b)))
|
e.writeInt32(int32(len(b)))
|
||||||
e.Write(b)
|
_, _ = e.Write(b)
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarBytes(b []byte) {
|
|
||||||
e.writeVarInt(int64(len(b)))
|
|
||||||
e.Write(b)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeCompactBytes(b []byte) {
|
func (e *encoder) writeCompactBytes(b []byte) {
|
||||||
e.writeUnsignedVarInt(uint64(len(b)) + 1)
|
e.writeUnsignedVarInt(uint64(len(b)) + 1)
|
||||||
e.Write(b)
|
_, _ = e.Write(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeNullBytes(b []byte) {
|
func (e *encoder) writeNullBytes(b []byte) {
|
||||||
@ -308,16 +268,7 @@ func (e *encoder) writeNullBytes(b []byte) {
|
|||||||
e.writeInt32(-1)
|
e.writeInt32(-1)
|
||||||
} else {
|
} else {
|
||||||
e.writeInt32(int32(len(b)))
|
e.writeInt32(int32(len(b)))
|
||||||
e.Write(b)
|
_, _ = e.Write(b)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarNullBytes(b []byte) {
|
|
||||||
if b == nil {
|
|
||||||
e.writeVarInt(-1)
|
|
||||||
} else {
|
|
||||||
e.writeVarInt(int64(len(b)))
|
|
||||||
e.Write(b)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -326,69 +277,10 @@ func (e *encoder) writeCompactNullBytes(b []byte) {
|
|||||||
e.writeUnsignedVarInt(0)
|
e.writeUnsignedVarInt(0)
|
||||||
} else {
|
} else {
|
||||||
e.writeUnsignedVarInt(uint64(len(b)) + 1)
|
e.writeUnsignedVarInt(uint64(len(b)) + 1)
|
||||||
e.Write(b)
|
_, _ = e.Write(b)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *encoder) writeBytesFrom(b Bytes) error {
|
|
||||||
size := int64(b.Len())
|
|
||||||
e.writeInt32(int32(size))
|
|
||||||
n, err := io.Copy(e, b)
|
|
||||||
if err == nil && n != size {
|
|
||||||
err = fmt.Errorf("size of bytes does not match the number of bytes that were written (size=%d, written=%d): %w", size, n, io.ErrUnexpectedEOF)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeNullBytesFrom(b Bytes) error {
|
|
||||||
if b == nil {
|
|
||||||
e.writeInt32(-1)
|
|
||||||
return nil
|
|
||||||
} else {
|
|
||||||
size := int64(b.Len())
|
|
||||||
e.writeInt32(int32(size))
|
|
||||||
n, err := io.Copy(e, b)
|
|
||||||
if err == nil && n != size {
|
|
||||||
err = fmt.Errorf("size of nullable bytes does not match the number of bytes that were written (size=%d, written=%d): %w", size, n, io.ErrUnexpectedEOF)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarNullBytesFrom(b Bytes) error {
|
|
||||||
if b == nil {
|
|
||||||
e.writeVarInt(-1)
|
|
||||||
return nil
|
|
||||||
} else {
|
|
||||||
size := int64(b.Len())
|
|
||||||
e.writeVarInt(size)
|
|
||||||
n, err := io.Copy(e, b)
|
|
||||||
if err == nil && n != size {
|
|
||||||
err = fmt.Errorf("size of nullable bytes does not match the number of bytes that were written (size=%d, written=%d): %w", size, n, io.ErrUnexpectedEOF)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeCompactNullBytesFrom(b Bytes) error {
|
|
||||||
if b == nil {
|
|
||||||
e.writeUnsignedVarInt(0)
|
|
||||||
return nil
|
|
||||||
} else {
|
|
||||||
size := int64(b.Len())
|
|
||||||
e.writeUnsignedVarInt(uint64(size + 1))
|
|
||||||
n, err := io.Copy(e, b)
|
|
||||||
if err == nil && n != size {
|
|
||||||
err = fmt.Errorf("size of compact nullable bytes does not match the number of bytes that were written (size=%d, written=%d): %w", size, n, io.ErrUnexpectedEOF)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeVarInt(i int64) {
|
|
||||||
e.writeUnsignedVarInt(uint64((i << 1) ^ (i >> 63)))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *encoder) writeUnsignedVarInt(i uint64) {
|
func (e *encoder) writeUnsignedVarInt(i uint64) {
|
||||||
b := e.buffer[:]
|
b := e.buffer[:]
|
||||||
n := 0
|
n := 0
|
||||||
@ -404,7 +296,7 @@ func (e *encoder) writeUnsignedVarInt(i uint64) {
|
|||||||
n++
|
n++
|
||||||
}
|
}
|
||||||
|
|
||||||
e.Write(b[:n])
|
_, _ = e.Write(b[:n])
|
||||||
}
|
}
|
||||||
|
|
||||||
type encodeFunc func(*encoder, value)
|
type encodeFunc func(*encoder, value)
|
||||||
@ -530,7 +422,7 @@ func structEncodeFuncOf(typ reflect.Type, version int16, flexible bool) encodeFu
|
|||||||
se := &encoder{writer: buf}
|
se := &encoder{writer: buf}
|
||||||
f.encode(se, v.fieldByIndex(f.index))
|
f.encode(se, v.fieldByIndex(f.index))
|
||||||
e.writeUnsignedVarInt(uint64(buf.Len()))
|
e.writeUnsignedVarInt(uint64(buf.Len()))
|
||||||
e.Write(buf.Bytes())
|
_, _ = e.Write(buf.Bytes())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -346,7 +346,13 @@ func representProduceRequest(data map[string]interface{}) []interface{} {
|
|||||||
})
|
})
|
||||||
|
|
||||||
obj, err := oj.ParseString(string(partitionsJson))
|
obj, err := oj.ParseString(string(partitionsJson))
|
||||||
|
if err != nil {
|
||||||
|
return rep
|
||||||
|
}
|
||||||
recordBatchPath, err := jp.ParseString(`partitionData.records.recordBatch`)
|
recordBatchPath, err := jp.ParseString(`partitionData.records.recordBatch`)
|
||||||
|
if err != nil {
|
||||||
|
return rep
|
||||||
|
}
|
||||||
recordBatchresults := recordBatchPath.Get(obj)
|
recordBatchresults := recordBatchPath.Get(obj)
|
||||||
if len(recordBatchresults) > 0 {
|
if len(recordBatchresults) > 0 {
|
||||||
rep = append(rep, api.SectionData{
|
rep = append(rep, api.SectionData{
|
||||||
@ -357,6 +363,9 @@ func representProduceRequest(data map[string]interface{}) []interface{} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
recordsPath, err := jp.ParseString(`partitionData.records.recordBatch.record`)
|
recordsPath, err := jp.ParseString(`partitionData.records.recordBatch.record`)
|
||||||
|
if err != nil {
|
||||||
|
return rep
|
||||||
|
}
|
||||||
recordsResults := recordsPath.Get(obj)
|
recordsResults := recordsPath.Get(obj)
|
||||||
if len(recordsResults) > 0 {
|
if len(recordsResults) > 0 {
|
||||||
records := recordsResults[0].([]interface{})
|
records := recordsResults[0].([]interface{})
|
||||||
|
@ -81,10 +81,8 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
if len(summary) > 0 {
|
if len(summary) > 0 {
|
||||||
summary = summary[:len(summary)-2]
|
summary = summary[:len(summary)-2]
|
||||||
}
|
}
|
||||||
break
|
|
||||||
case ApiVersions:
|
case ApiVersions:
|
||||||
summary = reqDetails["clientID"].(string)
|
summary = reqDetails["clientID"].(string)
|
||||||
break
|
|
||||||
case Produce:
|
case Produce:
|
||||||
_topics := reqDetails["payload"].(map[string]interface{})["topicData"]
|
_topics := reqDetails["payload"].(map[string]interface{})["topicData"]
|
||||||
if _topics == nil {
|
if _topics == nil {
|
||||||
@ -97,7 +95,6 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
if len(summary) > 0 {
|
if len(summary) > 0 {
|
||||||
summary = summary[:len(summary)-2]
|
summary = summary[:len(summary)-2]
|
||||||
}
|
}
|
||||||
break
|
|
||||||
case Fetch:
|
case Fetch:
|
||||||
_topics := reqDetails["payload"].(map[string]interface{})["topics"]
|
_topics := reqDetails["payload"].(map[string]interface{})["topics"]
|
||||||
if _topics == nil {
|
if _topics == nil {
|
||||||
@ -110,7 +107,6 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
if len(summary) > 0 {
|
if len(summary) > 0 {
|
||||||
summary = summary[:len(summary)-2]
|
summary = summary[:len(summary)-2]
|
||||||
}
|
}
|
||||||
break
|
|
||||||
case ListOffsets:
|
case ListOffsets:
|
||||||
_topics := reqDetails["payload"].(map[string]interface{})["topics"]
|
_topics := reqDetails["payload"].(map[string]interface{})["topics"]
|
||||||
if _topics == nil {
|
if _topics == nil {
|
||||||
@ -123,7 +119,6 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
if len(summary) > 0 {
|
if len(summary) > 0 {
|
||||||
summary = summary[:len(summary)-2]
|
summary = summary[:len(summary)-2]
|
||||||
}
|
}
|
||||||
break
|
|
||||||
case CreateTopics:
|
case CreateTopics:
|
||||||
topics := reqDetails["payload"].(map[string]interface{})["topics"].([]interface{})
|
topics := reqDetails["payload"].(map[string]interface{})["topics"].([]interface{})
|
||||||
for _, topic := range topics {
|
for _, topic := range topics {
|
||||||
@ -132,13 +127,11 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
if len(summary) > 0 {
|
if len(summary) > 0 {
|
||||||
summary = summary[:len(summary)-2]
|
summary = summary[:len(summary)-2]
|
||||||
}
|
}
|
||||||
break
|
|
||||||
case DeleteTopics:
|
case DeleteTopics:
|
||||||
topicNames := reqDetails["topicNames"].([]string)
|
topicNames := reqDetails["topicNames"].([]string)
|
||||||
for _, name := range topicNames {
|
for _, name := range topicNames {
|
||||||
summary += fmt.Sprintf("%s, ", name)
|
summary += fmt.Sprintf("%s, ", name)
|
||||||
}
|
}
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
request["url"] = summary
|
request["url"] = summary
|
||||||
@ -173,7 +166,7 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
|
|
||||||
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
||||||
bodySize = 0
|
bodySize = 0
|
||||||
representation := make(map[string]interface{}, 0)
|
representation := make(map[string]interface{})
|
||||||
|
|
||||||
apiKey := ApiKey(request["apiKey"].(float64))
|
apiKey := ApiKey(request["apiKey"].(float64))
|
||||||
|
|
||||||
@ -183,31 +176,24 @@ func (d dissecting) Represent(request map[string]interface{}, response map[strin
|
|||||||
case Metadata:
|
case Metadata:
|
||||||
repRequest = representMetadataRequest(request)
|
repRequest = representMetadataRequest(request)
|
||||||
repResponse = representMetadataResponse(response)
|
repResponse = representMetadataResponse(response)
|
||||||
break
|
|
||||||
case ApiVersions:
|
case ApiVersions:
|
||||||
repRequest = representApiVersionsRequest(request)
|
repRequest = representApiVersionsRequest(request)
|
||||||
repResponse = representApiVersionsResponse(response)
|
repResponse = representApiVersionsResponse(response)
|
||||||
break
|
|
||||||
case Produce:
|
case Produce:
|
||||||
repRequest = representProduceRequest(request)
|
repRequest = representProduceRequest(request)
|
||||||
repResponse = representProduceResponse(response)
|
repResponse = representProduceResponse(response)
|
||||||
break
|
|
||||||
case Fetch:
|
case Fetch:
|
||||||
repRequest = representFetchRequest(request)
|
repRequest = representFetchRequest(request)
|
||||||
repResponse = representFetchResponse(response)
|
repResponse = representFetchResponse(response)
|
||||||
break
|
|
||||||
case ListOffsets:
|
case ListOffsets:
|
||||||
repRequest = representListOffsetsRequest(request)
|
repRequest = representListOffsetsRequest(request)
|
||||||
repResponse = representListOffsetsResponse(response)
|
repResponse = representListOffsetsResponse(response)
|
||||||
break
|
|
||||||
case CreateTopics:
|
case CreateTopics:
|
||||||
repRequest = representCreateTopicsRequest(request)
|
repRequest = representCreateTopicsRequest(request)
|
||||||
repResponse = representCreateTopicsResponse(response)
|
repResponse = representCreateTopicsResponse(response)
|
||||||
break
|
|
||||||
case DeleteTopics:
|
case DeleteTopics:
|
||||||
repRequest = representDeleteTopicsRequest(request)
|
repRequest = representDeleteTopicsRequest(request)
|
||||||
repResponse = representDeleteTopicsResponse(response)
|
repResponse = representDeleteTopicsResponse(response)
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
representation["request"] = repRequest
|
representation["request"] = repRequest
|
||||||
|
@ -26,9 +26,9 @@ func CreateResponseRequestMatcher() requestResponseMatcher {
|
|||||||
func (matcher *requestResponseMatcher) registerRequest(key string, request *Request) *RequestResponsePair {
|
func (matcher *requestResponseMatcher) registerRequest(key string, request *Request) *RequestResponsePair {
|
||||||
if response, found := matcher.openMessagesMap.LoadAndDelete(key); found {
|
if response, found := matcher.openMessagesMap.LoadAndDelete(key); found {
|
||||||
// Check for a situation that only occurs when a Kafka broker is initiating
|
// Check for a situation that only occurs when a Kafka broker is initiating
|
||||||
switch response.(type) {
|
switch v := response.(type) {
|
||||||
case *Response:
|
case *Response:
|
||||||
return matcher.preparePair(request, response.(*Response))
|
return matcher.preparePair(request, v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,10 +167,6 @@ type messageType struct {
|
|||||||
encode encodeFunc
|
encode encodeFunc
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *messageType) new() Message {
|
|
||||||
return reflect.New(t.gotype).Interface().(Message)
|
|
||||||
}
|
|
||||||
|
|
||||||
type apiType struct {
|
type apiType struct {
|
||||||
requests []messageType
|
requests []messageType
|
||||||
responses []messageType
|
responses []messageType
|
||||||
@ -389,16 +385,16 @@ func (b Broker) String() string {
|
|||||||
func (b Broker) Format(w fmt.State, v rune) {
|
func (b Broker) Format(w fmt.State, v rune) {
|
||||||
switch v {
|
switch v {
|
||||||
case 'd':
|
case 'd':
|
||||||
io.WriteString(w, itoa(b.ID))
|
_, _ = io.WriteString(w, itoa(b.ID))
|
||||||
case 's':
|
case 's':
|
||||||
io.WriteString(w, b.String())
|
_, _ = io.WriteString(w, b.String())
|
||||||
case 'v':
|
case 'v':
|
||||||
io.WriteString(w, itoa(b.ID))
|
_, _ = io.WriteString(w, itoa(b.ID))
|
||||||
io.WriteString(w, " ")
|
_, _ = io.WriteString(w, " ")
|
||||||
io.WriteString(w, b.String())
|
_, _ = io.WriteString(w, b.String())
|
||||||
if b.Rack != "" {
|
if b.Rack != "" {
|
||||||
io.WriteString(w, " ")
|
_, _ = io.WriteString(w, " ")
|
||||||
io.WriteString(w, b.Rack)
|
_, _ = io.WriteString(w, b.Rack)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -99,10 +99,8 @@ func (k apiKey) String() string {
|
|||||||
return strconv.Itoa(int(k))
|
return strconv.Itoa(int(k))
|
||||||
}
|
}
|
||||||
|
|
||||||
type apiVersion int16
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
v0 = 0
|
// v0 = 0
|
||||||
v1 = 1
|
v1 = 1
|
||||||
v2 = 2
|
v2 = 2
|
||||||
v3 = 3
|
v3 = 3
|
||||||
@ -113,6 +111,7 @@ const (
|
|||||||
v8 = 8
|
v8 = 8
|
||||||
v9 = 9
|
v9 = 9
|
||||||
v10 = 10
|
v10 = 10
|
||||||
|
v11 = 11
|
||||||
)
|
)
|
||||||
|
|
||||||
var apiKeyStrings = [...]string{
|
var apiKeyStrings = [...]string{
|
||||||
@ -166,35 +165,6 @@ var apiKeyStrings = [...]string{
|
|||||||
offsetDelete: "OffsetDelete",
|
offsetDelete: "OffsetDelete",
|
||||||
}
|
}
|
||||||
|
|
||||||
type requestHeader struct {
|
|
||||||
Size int32
|
|
||||||
ApiKey int16
|
|
||||||
ApiVersion int16
|
|
||||||
CorrelationID int32
|
|
||||||
ClientID string
|
|
||||||
}
|
|
||||||
|
|
||||||
func sizeofString(s string) int32 {
|
|
||||||
return 2 + int32(len(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h requestHeader) size() int32 {
|
|
||||||
return 4 + 2 + 2 + 4 + sizeofString(h.ClientID)
|
|
||||||
}
|
|
||||||
|
|
||||||
// func (h requestHeader) writeTo(wb *writeBuffer) {
|
|
||||||
// wb.writeInt32(h.Size)
|
|
||||||
// wb.writeInt16(h.ApiKey)
|
|
||||||
// wb.writeInt16(h.ApiVersion)
|
|
||||||
// wb.writeInt32(h.CorrelationID)
|
|
||||||
// wb.writeString(h.ClientID)
|
|
||||||
// }
|
|
||||||
|
|
||||||
type request interface {
|
|
||||||
size() int32
|
|
||||||
// writable
|
|
||||||
}
|
|
||||||
|
|
||||||
func makeInt8(b []byte) int8 {
|
func makeInt8(b []byte) int8 {
|
||||||
return int8(b[0])
|
return int8(b[0])
|
||||||
}
|
}
|
||||||
@ -210,10 +180,3 @@ func makeInt32(b []byte) int32 {
|
|||||||
func makeInt64(b []byte) int64 {
|
func makeInt64(b []byte) int64 {
|
||||||
return int64(binary.BigEndian.Uint64(b))
|
return int64(binary.BigEndian.Uint64(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectZeroSize(sz int, err error) error {
|
|
||||||
if err == nil && sz != 0 {
|
|
||||||
err = fmt.Errorf("reading a response left %d unread bytes", sz)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"reflect"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type readable interface {
|
type readable interface {
|
||||||
@ -42,59 +41,6 @@ func readInt64(r *bufio.Reader, sz int, v *int64) (int, error) {
|
|||||||
return peekRead(r, sz, 8, func(b []byte) { *v = makeInt64(b) })
|
return peekRead(r, sz, 8, func(b []byte) { *v = makeInt64(b) })
|
||||||
}
|
}
|
||||||
|
|
||||||
func readVarInt(r *bufio.Reader, sz int, v *int64) (remain int, err error) {
|
|
||||||
// Optimistically assume that most of the time, there will be data buffered
|
|
||||||
// in the reader. If this is not the case, the buffer will be refilled after
|
|
||||||
// consuming zero bytes from the input.
|
|
||||||
input, _ := r.Peek(r.Buffered())
|
|
||||||
x := uint64(0)
|
|
||||||
s := uint(0)
|
|
||||||
|
|
||||||
for {
|
|
||||||
if len(input) > sz {
|
|
||||||
input = input[:sz]
|
|
||||||
}
|
|
||||||
|
|
||||||
for i, b := range input {
|
|
||||||
if b < 0x80 {
|
|
||||||
x |= uint64(b) << s
|
|
||||||
*v = int64(x>>1) ^ -(int64(x) & 1)
|
|
||||||
n, err := r.Discard(i + 1)
|
|
||||||
return sz - n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
x |= uint64(b&0x7f) << s
|
|
||||||
s += 7
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make room in the input buffer to load more data from the underlying
|
|
||||||
// stream. The x and s variables are left untouched, ensuring that the
|
|
||||||
// varint decoding can continue on the next loop iteration.
|
|
||||||
n, _ := r.Discard(len(input))
|
|
||||||
sz -= n
|
|
||||||
if sz == 0 {
|
|
||||||
return 0, errShortRead
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fill the buffer: ask for one more byte, but in practice the reader
|
|
||||||
// will load way more from the underlying stream.
|
|
||||||
if _, err := r.Peek(1); err != nil {
|
|
||||||
if err == io.EOF {
|
|
||||||
err = errShortRead
|
|
||||||
}
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Grab as many bytes as possible from the buffer, then go on to the
|
|
||||||
// next loop iteration which is going to consume it.
|
|
||||||
input, _ = r.Peek(r.Buffered())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func readBool(r *bufio.Reader, sz int, v *bool) (int, error) {
|
|
||||||
return peekRead(r, sz, 1, func(b []byte) { *v = b[0] != 0 })
|
|
||||||
}
|
|
||||||
|
|
||||||
func readString(r *bufio.Reader, sz int, v *string) (int, error) {
|
func readString(r *bufio.Reader, sz int, v *string) (int, error) {
|
||||||
return readStringWith(r, sz, func(r *bufio.Reader, sz int, n int) (remain int, err error) {
|
return readStringWith(r, sz, func(r *bufio.Reader, sz int, n int) (remain int, err error) {
|
||||||
*v, remain, err = readNewString(r, sz, n)
|
*v, remain, err = readNewString(r, sz, n)
|
||||||
@ -179,102 +125,6 @@ func readArrayLen(r *bufio.Reader, sz int, n *int) (int, error) {
|
|||||||
return sz, nil
|
return sz, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func readArrayWith(r *bufio.Reader, sz int, cb func(*bufio.Reader, int) (int, error)) (int, error) {
|
|
||||||
var err error
|
|
||||||
var len int32
|
|
||||||
|
|
||||||
if sz, err = readInt32(r, sz, &len); err != nil {
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for n := int(len); n > 0; n-- {
|
|
||||||
if sz, err = cb(r, sz); err != nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func readStringArray(r *bufio.Reader, sz int, v *[]string) (remain int, err error) {
|
|
||||||
var content []string
|
|
||||||
fn := func(r *bufio.Reader, size int) (fnRemain int, fnErr error) {
|
|
||||||
var value string
|
|
||||||
if fnRemain, fnErr = readString(r, size, &value); fnErr != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
content = append(content, value)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if remain, err = readArrayWith(r, sz, fn); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
*v = content
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func readMapStringInt32(r *bufio.Reader, sz int, v *map[string][]int32) (remain int, err error) {
|
|
||||||
var len int32
|
|
||||||
if remain, err = readInt32(r, sz, &len); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
content := make(map[string][]int32, len)
|
|
||||||
for i := 0; i < int(len); i++ {
|
|
||||||
var key string
|
|
||||||
var values []int32
|
|
||||||
|
|
||||||
if remain, err = readString(r, remain, &key); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
fn := func(r *bufio.Reader, size int) (fnRemain int, fnErr error) {
|
|
||||||
var value int32
|
|
||||||
if fnRemain, fnErr = readInt32(r, size, &value); fnErr != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
values = append(values, value)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if remain, err = readArrayWith(r, remain, fn); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
content[key] = values
|
|
||||||
}
|
|
||||||
*v = content
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func read(r *bufio.Reader, sz int, a interface{}) (int, error) {
|
|
||||||
switch v := a.(type) {
|
|
||||||
case *int8:
|
|
||||||
return readInt8(r, sz, v)
|
|
||||||
case *int16:
|
|
||||||
return readInt16(r, sz, v)
|
|
||||||
case *int32:
|
|
||||||
return readInt32(r, sz, v)
|
|
||||||
case *int64:
|
|
||||||
return readInt64(r, sz, v)
|
|
||||||
case *bool:
|
|
||||||
return readBool(r, sz, v)
|
|
||||||
case *string:
|
|
||||||
return readString(r, sz, v)
|
|
||||||
case *[]byte:
|
|
||||||
return readBytes(r, sz, v)
|
|
||||||
}
|
|
||||||
switch v := reflect.ValueOf(a).Elem(); v.Kind() {
|
|
||||||
case reflect.Struct:
|
|
||||||
return readStruct(r, sz, v)
|
|
||||||
case reflect.Slice:
|
|
||||||
return readSlice(r, sz, v)
|
|
||||||
default:
|
|
||||||
panic(fmt.Sprintf("unsupported type: %T", a))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func ReadAll(r *bufio.Reader, sz int, ptrs ...interface{}) (int, error) {
|
func ReadAll(r *bufio.Reader, sz int, ptrs ...interface{}) (int, error) {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
@ -307,333 +157,3 @@ func readPtr(r *bufio.Reader, sz int, ptr interface{}) (int, error) {
|
|||||||
panic(fmt.Sprintf("unsupported type: %T", v))
|
panic(fmt.Sprintf("unsupported type: %T", v))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func readStruct(r *bufio.Reader, sz int, v reflect.Value) (int, error) {
|
|
||||||
var err error
|
|
||||||
for i, n := 0, v.NumField(); i != n; i++ {
|
|
||||||
if sz, err = read(r, sz, v.Field(i).Addr().Interface()); err != nil {
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return sz, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func readSlice(r *bufio.Reader, sz int, v reflect.Value) (int, error) {
|
|
||||||
var err error
|
|
||||||
var len int32
|
|
||||||
|
|
||||||
if sz, err = readInt32(r, sz, &len); err != nil {
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if n := int(len); n < 0 {
|
|
||||||
v.Set(reflect.Zero(v.Type()))
|
|
||||||
} else {
|
|
||||||
v.Set(reflect.MakeSlice(v.Type(), n, n))
|
|
||||||
|
|
||||||
for i := 0; i != n; i++ {
|
|
||||||
if sz, err = read(r, sz, v.Index(i).Addr().Interface()); err != nil {
|
|
||||||
return sz, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return sz, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func readFetchResponseHeaderV2(r *bufio.Reader, size int) (throttle int32, watermark int64, remain int, err error) {
|
|
||||||
var n int32
|
|
||||||
var p struct {
|
|
||||||
Partition int32
|
|
||||||
ErrorCode int16
|
|
||||||
HighwaterMarkOffset int64
|
|
||||||
MessageSetSize int32
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, size, &throttle); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka topic was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// We ignore the topic name because we've requests messages for a single
|
|
||||||
// topic, unless there's a bug in the kafka server we will have received
|
|
||||||
// the name of the topic that we requested.
|
|
||||||
if remain, err = discardString(r, remain); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka partition was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = read(r, remain, &p); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if p.ErrorCode != 0 {
|
|
||||||
err = Error(p.ErrorCode)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if remain != int(p.MessageSetSize) {
|
|
||||||
err = fmt.Errorf("the size of the message set in a fetch response doesn't match the number of remaining bytes (message set size = %d, remaining bytes = %d)", p.MessageSetSize, remain)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
watermark = p.HighwaterMarkOffset
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func readFetchResponseHeaderV5(r *bufio.Reader, size int) (throttle int32, watermark int64, remain int, err error) {
|
|
||||||
var n int32
|
|
||||||
type AbortedTransaction struct {
|
|
||||||
ProducerId int64
|
|
||||||
FirstOffset int64
|
|
||||||
}
|
|
||||||
var p struct {
|
|
||||||
Partition int32
|
|
||||||
ErrorCode int16
|
|
||||||
HighwaterMarkOffset int64
|
|
||||||
LastStableOffset int64
|
|
||||||
LogStartOffset int64
|
|
||||||
}
|
|
||||||
var messageSetSize int32
|
|
||||||
var abortedTransactions []AbortedTransaction
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, size, &throttle); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka topic was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// We ignore the topic name because we've requests messages for a single
|
|
||||||
// topic, unless there's a bug in the kafka server we will have received
|
|
||||||
// the name of the topic that we requested.
|
|
||||||
if remain, err = discardString(r, remain); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka partition was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = read(r, remain, &p); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var abortedTransactionLen int
|
|
||||||
if remain, err = readArrayLen(r, remain, &abortedTransactionLen); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if abortedTransactionLen == -1 {
|
|
||||||
abortedTransactions = nil
|
|
||||||
} else {
|
|
||||||
abortedTransactions = make([]AbortedTransaction, abortedTransactionLen)
|
|
||||||
for i := 0; i < abortedTransactionLen; i++ {
|
|
||||||
if remain, err = read(r, remain, &abortedTransactions[i]); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if p.ErrorCode != 0 {
|
|
||||||
err = Error(p.ErrorCode)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
remain, err = readInt32(r, remain, &messageSetSize)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if remain != int(messageSetSize) {
|
|
||||||
err = fmt.Errorf("the size of the message set in a fetch response doesn't match the number of remaining bytes (message set size = %d, remaining bytes = %d)", messageSetSize, remain)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
watermark = p.HighwaterMarkOffset
|
|
||||||
return
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func readFetchResponseHeaderV10(r *bufio.Reader, size int) (throttle int32, watermark int64, remain int, err error) {
|
|
||||||
var n int32
|
|
||||||
var errorCode int16
|
|
||||||
type AbortedTransaction struct {
|
|
||||||
ProducerId int64
|
|
||||||
FirstOffset int64
|
|
||||||
}
|
|
||||||
var p struct {
|
|
||||||
Partition int32
|
|
||||||
ErrorCode int16
|
|
||||||
HighwaterMarkOffset int64
|
|
||||||
LastStableOffset int64
|
|
||||||
LogStartOffset int64
|
|
||||||
}
|
|
||||||
var messageSetSize int32
|
|
||||||
var abortedTransactions []AbortedTransaction
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, size, &throttle); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt16(r, remain, &errorCode); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if errorCode != 0 {
|
|
||||||
err = Error(errorCode)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = discardInt32(r, remain); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka topic was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// We ignore the topic name because we've requests messages for a single
|
|
||||||
// topic, unless there's a bug in the kafka server we will have received
|
|
||||||
// the name of the topic that we requested.
|
|
||||||
if remain, err = discardString(r, remain); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt32(r, remain, &n); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if n != 1 {
|
|
||||||
err = fmt.Errorf("1 kafka partition was expected in the fetch response but the client received %d", n)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = read(r, remain, &p); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var abortedTransactionLen int
|
|
||||||
if remain, err = readArrayLen(r, remain, &abortedTransactionLen); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if abortedTransactionLen == -1 {
|
|
||||||
abortedTransactions = nil
|
|
||||||
} else {
|
|
||||||
abortedTransactions = make([]AbortedTransaction, abortedTransactionLen)
|
|
||||||
for i := 0; i < abortedTransactionLen; i++ {
|
|
||||||
if remain, err = read(r, remain, &abortedTransactions[i]); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if p.ErrorCode != 0 {
|
|
||||||
err = Error(p.ErrorCode)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
remain, err = readInt32(r, remain, &messageSetSize)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// This error should never trigger, unless there's a bug in the kafka client
|
|
||||||
// or server.
|
|
||||||
if remain != int(messageSetSize) {
|
|
||||||
err = fmt.Errorf("the size of the message set in a fetch response doesn't match the number of remaining bytes (message set size = %d, remaining bytes = %d)", messageSetSize, remain)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
watermark = p.HighwaterMarkOffset
|
|
||||||
return
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func readMessageHeader(r *bufio.Reader, sz int) (offset int64, attributes int8, timestamp int64, remain int, err error) {
|
|
||||||
var version int8
|
|
||||||
|
|
||||||
if remain, err = readInt64(r, sz, &offset); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// On discarding the message size and CRC:
|
|
||||||
// ---------------------------------------
|
|
||||||
//
|
|
||||||
// - Not sure why kafka gives the message size here, we already have the
|
|
||||||
// number of remaining bytes in the response and kafka should only truncate
|
|
||||||
// the trailing message.
|
|
||||||
//
|
|
||||||
// - TCP is already taking care of ensuring data integrity, no need to
|
|
||||||
// waste resources doing it a second time so we just skip the message CRC.
|
|
||||||
//
|
|
||||||
if remain, err = discardN(r, remain, 8); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt8(r, remain, &version); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if remain, err = readInt8(r, remain, &attributes); err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch version {
|
|
||||||
case 0:
|
|
||||||
case 1:
|
|
||||||
remain, err = readInt64(r, remain, ×tamp)
|
|
||||||
default:
|
|
||||||
err = fmt.Errorf("unsupported message version %d found in fetch response", version)
|
|
||||||
}
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
@ -110,25 +110,6 @@ type RecordSet struct {
|
|||||||
Records RecordReader
|
Records RecordReader
|
||||||
}
|
}
|
||||||
|
|
||||||
// bufferedReader is an interface implemented by types like bufio.Reader, which
|
|
||||||
// we use to optimize prefix reads by accessing the internal buffer directly
|
|
||||||
// through calls to Peek.
|
|
||||||
type bufferedReader interface {
|
|
||||||
Discard(int) (int, error)
|
|
||||||
Peek(int) ([]byte, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
// bytesBuffer is an interface implemented by types like bytes.Buffer, which we
|
|
||||||
// use to optimize prefix reads by accessing the internal buffer directly
|
|
||||||
// through calls to Bytes.
|
|
||||||
type bytesBuffer interface {
|
|
||||||
Bytes() []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
// magicByteOffset is the position of the magic byte in all versions of record
|
|
||||||
// sets in the kafka protocol.
|
|
||||||
const magicByteOffset = 16
|
|
||||||
|
|
||||||
// ReadFrom reads the representation of a record set from r into rs, returning
|
// ReadFrom reads the representation of a record set from r into rs, returning
|
||||||
// the number of bytes consumed from r, and an non-nil error if the record set
|
// the number of bytes consumed from r, and an non-nil error if the record set
|
||||||
// could not be read.
|
// could not be read.
|
||||||
@ -292,23 +273,7 @@ func (rs *RecordSet) WriteTo(w io.Writer) (int64, error) {
|
|||||||
return 0, nil
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeTime(t int64) time.Time {
|
|
||||||
return time.Unix(t/1000, (t%1000)*int64(time.Millisecond))
|
|
||||||
}
|
|
||||||
|
|
||||||
func timestamp(t time.Time) int64 {
|
|
||||||
if t.IsZero() {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
return t.UnixNano() / int64(time.Millisecond)
|
|
||||||
}
|
|
||||||
|
|
||||||
func packUint32(u uint32) (b [4]byte) {
|
func packUint32(u uint32) (b [4]byte) {
|
||||||
binary.BigEndian.PutUint32(b[:], u)
|
binary.BigEndian.PutUint32(b[:], u)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func packUint64(u uint64) (b [8]byte) {
|
|
||||||
binary.BigEndian.PutUint64(b[:], u)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
@ -34,10 +34,6 @@ func valueOf(x interface{}) value {
|
|||||||
return value{val: reflect.ValueOf(x).Elem()}
|
return value{val: reflect.ValueOf(x).Elem()}
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeValue(t reflect.Type) value {
|
|
||||||
return value{val: reflect.New(t).Elem()}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v value) bool() bool { return v.val.Bool() }
|
func (v value) bool() bool { return v.val.Bool() }
|
||||||
|
|
||||||
func (v value) int8() int8 { return int8(v.int64()) }
|
func (v value) int8() int8 { return int8(v.int64()) }
|
||||||
@ -54,7 +50,7 @@ func (v value) bytes() []byte { return v.val.Bytes() }
|
|||||||
|
|
||||||
func (v value) iface(t reflect.Type) interface{} { return v.val.Addr().Interface() }
|
func (v value) iface(t reflect.Type) interface{} { return v.val.Addr().Interface() }
|
||||||
|
|
||||||
func (v value) array(t reflect.Type) array { return array{val: v.val} }
|
func (v value) array(t reflect.Type) array { return array{val: v.val} } //nolint
|
||||||
|
|
||||||
func (v value) setBool(b bool) { v.val.SetBool(b) }
|
func (v value) setBool(b bool) { v.val.SetBool(b) }
|
||||||
|
|
||||||
|
@ -64,19 +64,19 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
case Metadata:
|
case Metadata:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var metadataRequest interface{}
|
var metadataRequest interface{}
|
||||||
if apiVersion >= 11 {
|
if apiVersion >= v11 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataRequestV11{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataRequestV11{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataRequest = &MetadataRequestV11{}
|
metadataRequest = &MetadataRequestV11{}
|
||||||
} else if apiVersion >= 10 {
|
} else if apiVersion >= v10 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataRequestV10{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataRequestV10{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataRequest = &MetadataRequestV10{}
|
metadataRequest = &MetadataRequestV10{}
|
||||||
} else if apiVersion >= 8 {
|
} else if apiVersion >= v8 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataRequestV8{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataRequestV8{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataRequest = &MetadataRequestV8{}
|
metadataRequest = &MetadataRequestV8{}
|
||||||
} else if apiVersion >= 4 {
|
} else if apiVersion >= v4 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataRequestV4{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataRequestV4{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataRequest = &MetadataRequestV4{}
|
metadataRequest = &MetadataRequestV4{}
|
||||||
@ -87,11 +87,10 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(metadataRequest))
|
mt.(messageType).decode(d, valueOf(metadataRequest))
|
||||||
payload = metadataRequest
|
payload = metadataRequest
|
||||||
break
|
|
||||||
case ApiVersions:
|
case ApiVersions:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var apiVersionsRequest interface{}
|
var apiVersionsRequest interface{}
|
||||||
if apiVersion >= 3 {
|
if apiVersion >= v3 {
|
||||||
types := makeTypes(reflect.TypeOf(&ApiVersionsRequestV3{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ApiVersionsRequestV3{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
apiVersionsRequest = &ApiVersionsRequestV3{}
|
apiVersionsRequest = &ApiVersionsRequestV3{}
|
||||||
@ -102,11 +101,10 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(apiVersionsRequest))
|
mt.(messageType).decode(d, valueOf(apiVersionsRequest))
|
||||||
payload = apiVersionsRequest
|
payload = apiVersionsRequest
|
||||||
break
|
|
||||||
case Produce:
|
case Produce:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var produceRequest interface{}
|
var produceRequest interface{}
|
||||||
if apiVersion >= 3 {
|
if apiVersion >= v3 {
|
||||||
types := makeTypes(reflect.TypeOf(&ProduceRequestV3{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ProduceRequestV3{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
produceRequest = &ProduceRequestV3{}
|
produceRequest = &ProduceRequestV3{}
|
||||||
@ -117,7 +115,6 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(produceRequest))
|
mt.(messageType).decode(d, valueOf(produceRequest))
|
||||||
payload = produceRequest
|
payload = produceRequest
|
||||||
break
|
|
||||||
case Fetch:
|
case Fetch:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var fetchRequest interface{}
|
var fetchRequest interface{}
|
||||||
@ -125,23 +122,23 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV11{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV11{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV11{}
|
fetchRequest = &FetchRequestV11{}
|
||||||
} else if apiVersion >= 9 {
|
} else if apiVersion >= v9 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV9{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV9{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV9{}
|
fetchRequest = &FetchRequestV9{}
|
||||||
} else if apiVersion >= 7 {
|
} else if apiVersion >= v7 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV7{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV7{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV7{}
|
fetchRequest = &FetchRequestV7{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV5{}
|
fetchRequest = &FetchRequestV5{}
|
||||||
} else if apiVersion >= 4 {
|
} else if apiVersion >= v4 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV4{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV4{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV4{}
|
fetchRequest = &FetchRequestV4{}
|
||||||
} else if apiVersion >= 3 {
|
} else if apiVersion >= v3 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchRequestV3{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchRequestV3{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchRequest = &FetchRequestV3{}
|
fetchRequest = &FetchRequestV3{}
|
||||||
@ -152,19 +149,18 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(fetchRequest))
|
mt.(messageType).decode(d, valueOf(fetchRequest))
|
||||||
payload = fetchRequest
|
payload = fetchRequest
|
||||||
break
|
|
||||||
case ListOffsets:
|
case ListOffsets:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var listOffsetsRequest interface{}
|
var listOffsetsRequest interface{}
|
||||||
if apiVersion >= 4 {
|
if apiVersion >= v4 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV4{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV4{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsRequest = &ListOffsetsRequestV4{}
|
listOffsetsRequest = &ListOffsetsRequestV4{}
|
||||||
} else if apiVersion >= 2 {
|
} else if apiVersion >= v2 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV2{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV2{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsRequest = &ListOffsetsRequestV2{}
|
listOffsetsRequest = &ListOffsetsRequestV2{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsRequestV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsRequest = &ListOffsetsRequestV1{}
|
listOffsetsRequest = &ListOffsetsRequestV1{}
|
||||||
@ -175,11 +171,10 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(listOffsetsRequest))
|
mt.(messageType).decode(d, valueOf(listOffsetsRequest))
|
||||||
payload = listOffsetsRequest
|
payload = listOffsetsRequest
|
||||||
break
|
|
||||||
case CreateTopics:
|
case CreateTopics:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var createTopicsRequest interface{}
|
var createTopicsRequest interface{}
|
||||||
if apiVersion >= 1 {
|
if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&CreateTopicsRequestV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&CreateTopicsRequestV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
createTopicsRequest = &CreateTopicsRequestV1{}
|
createTopicsRequest = &CreateTopicsRequestV1{}
|
||||||
@ -190,11 +185,10 @@ func ReadRequest(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer) (api
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(createTopicsRequest))
|
mt.(messageType).decode(d, valueOf(createTopicsRequest))
|
||||||
payload = createTopicsRequest
|
payload = createTopicsRequest
|
||||||
break
|
|
||||||
case DeleteTopics:
|
case DeleteTopics:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var deleteTopicsRequest interface{}
|
var deleteTopicsRequest interface{}
|
||||||
if apiVersion >= 6 {
|
if apiVersion >= v6 {
|
||||||
types := makeTypes(reflect.TypeOf(&DeleteTopicsRequestV6{}).Elem())
|
types := makeTypes(reflect.TypeOf(&DeleteTopicsRequestV6{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
deleteTopicsRequest = &DeleteTopicsRequestV6{}
|
deleteTopicsRequest = &DeleteTopicsRequestV6{}
|
||||||
@ -285,7 +279,7 @@ func WriteRequest(w io.Writer, apiVersion int16, correlationID int32, clientID s
|
|||||||
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
size := packUint32(uint32(b.Size()) - 4)
|
size := packUint32(uint32(b.Size()) - 4)
|
||||||
b.WriteAt(size[:], 0)
|
_, _ = b.WriteAt(size[:], 0)
|
||||||
_, err = b.WriteTo(w)
|
_, err = b.WriteTo(w)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,35 +62,35 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
case Metadata:
|
case Metadata:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var metadataResponse interface{}
|
var metadataResponse interface{}
|
||||||
if apiVersion >= 11 {
|
if apiVersion >= v11 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV11{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV11{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV11{}
|
metadataResponse = &MetadataResponseV11{}
|
||||||
} else if apiVersion >= 10 {
|
} else if apiVersion >= v10 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV10{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV10{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV10{}
|
metadataResponse = &MetadataResponseV10{}
|
||||||
} else if apiVersion >= 8 {
|
} else if apiVersion >= v8 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV8{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV8{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV8{}
|
metadataResponse = &MetadataResponseV8{}
|
||||||
} else if apiVersion >= 7 {
|
} else if apiVersion >= v7 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV7{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV7{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV7{}
|
metadataResponse = &MetadataResponseV7{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV5{}
|
metadataResponse = &MetadataResponseV5{}
|
||||||
} else if apiVersion >= 3 {
|
} else if apiVersion >= v3 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV3{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV3{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV3{}
|
metadataResponse = &MetadataResponseV3{}
|
||||||
} else if apiVersion >= 2 {
|
} else if apiVersion >= v2 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV2{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV2{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV2{}
|
metadataResponse = &MetadataResponseV2{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&MetadataResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&MetadataResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
metadataResponse = &MetadataResponseV1{}
|
metadataResponse = &MetadataResponseV1{}
|
||||||
@ -101,11 +101,10 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(metadataResponse))
|
mt.(messageType).decode(d, valueOf(metadataResponse))
|
||||||
reqResPair.Response.Payload = metadataResponse
|
reqResPair.Response.Payload = metadataResponse
|
||||||
break
|
|
||||||
case ApiVersions:
|
case ApiVersions:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var apiVersionsResponse interface{}
|
var apiVersionsResponse interface{}
|
||||||
if apiVersion >= 1 {
|
if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&ApiVersionsResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ApiVersionsResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
apiVersionsResponse = &ApiVersionsResponseV1{}
|
apiVersionsResponse = &ApiVersionsResponseV1{}
|
||||||
@ -116,23 +115,22 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(apiVersionsResponse))
|
mt.(messageType).decode(d, valueOf(apiVersionsResponse))
|
||||||
reqResPair.Response.Payload = apiVersionsResponse
|
reqResPair.Response.Payload = apiVersionsResponse
|
||||||
break
|
|
||||||
case Produce:
|
case Produce:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var produceResponse interface{}
|
var produceResponse interface{}
|
||||||
if apiVersion >= 8 {
|
if apiVersion >= v8 {
|
||||||
types := makeTypes(reflect.TypeOf(&ProduceResponseV8{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ProduceResponseV8{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
produceResponse = &ProduceResponseV8{}
|
produceResponse = &ProduceResponseV8{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&ProduceResponseV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ProduceResponseV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
produceResponse = &ProduceResponseV5{}
|
produceResponse = &ProduceResponseV5{}
|
||||||
} else if apiVersion >= 2 {
|
} else if apiVersion >= v2 {
|
||||||
types := makeTypes(reflect.TypeOf(&ProduceResponseV2{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ProduceResponseV2{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
produceResponse = &ProduceResponseV2{}
|
produceResponse = &ProduceResponseV2{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&ProduceResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ProduceResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
produceResponse = &ProduceResponseV1{}
|
produceResponse = &ProduceResponseV1{}
|
||||||
@ -143,27 +141,26 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(produceResponse))
|
mt.(messageType).decode(d, valueOf(produceResponse))
|
||||||
reqResPair.Response.Payload = produceResponse
|
reqResPair.Response.Payload = produceResponse
|
||||||
break
|
|
||||||
case Fetch:
|
case Fetch:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var fetchResponse interface{}
|
var fetchResponse interface{}
|
||||||
if apiVersion >= 11 {
|
if apiVersion >= v11 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchResponseV11{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchResponseV11{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchResponse = &FetchResponseV11{}
|
fetchResponse = &FetchResponseV11{}
|
||||||
} else if apiVersion >= 7 {
|
} else if apiVersion >= v7 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchResponseV7{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchResponseV7{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchResponse = &FetchResponseV7{}
|
fetchResponse = &FetchResponseV7{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchResponseV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchResponseV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchResponse = &FetchResponseV5{}
|
fetchResponse = &FetchResponseV5{}
|
||||||
} else if apiVersion >= 4 {
|
} else if apiVersion >= v4 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchResponseV4{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchResponseV4{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchResponse = &FetchResponseV4{}
|
fetchResponse = &FetchResponseV4{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&FetchResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&FetchResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
fetchResponse = &FetchResponseV1{}
|
fetchResponse = &FetchResponseV1{}
|
||||||
@ -174,19 +171,18 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(fetchResponse))
|
mt.(messageType).decode(d, valueOf(fetchResponse))
|
||||||
reqResPair.Response.Payload = fetchResponse
|
reqResPair.Response.Payload = fetchResponse
|
||||||
break
|
|
||||||
case ListOffsets:
|
case ListOffsets:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var listOffsetsResponse interface{}
|
var listOffsetsResponse interface{}
|
||||||
if apiVersion >= 4 {
|
if apiVersion >= v4 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV4{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV4{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsResponse = &ListOffsetsResponseV4{}
|
listOffsetsResponse = &ListOffsetsResponseV4{}
|
||||||
} else if apiVersion >= 2 {
|
} else if apiVersion >= v2 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV2{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV2{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsResponse = &ListOffsetsResponseV2{}
|
listOffsetsResponse = &ListOffsetsResponseV2{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&ListOffsetsResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
listOffsetsResponse = &ListOffsetsResponseV1{}
|
listOffsetsResponse = &ListOffsetsResponseV1{}
|
||||||
@ -200,19 +196,19 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
case CreateTopics:
|
case CreateTopics:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var createTopicsResponse interface{}
|
var createTopicsResponse interface{}
|
||||||
if apiVersion >= 7 {
|
if apiVersion >= v7 {
|
||||||
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV0{}).Elem())
|
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV0{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
createTopicsResponse = &CreateTopicsResponseV0{}
|
createTopicsResponse = &CreateTopicsResponseV0{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
createTopicsResponse = &CreateTopicsResponseV5{}
|
createTopicsResponse = &CreateTopicsResponseV5{}
|
||||||
} else if apiVersion >= 2 {
|
} else if apiVersion >= v2 {
|
||||||
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV2{}).Elem())
|
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV2{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
createTopicsResponse = &CreateTopicsResponseV2{}
|
createTopicsResponse = &CreateTopicsResponseV2{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&CreateTopicsResponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
createTopicsResponse = &CreateTopicsResponseV1{}
|
createTopicsResponse = &CreateTopicsResponseV1{}
|
||||||
@ -223,19 +219,18 @@ func ReadResponse(r io.Reader, tcpID *api.TcpID, superTimer *api.SuperTimer, emi
|
|||||||
}
|
}
|
||||||
mt.(messageType).decode(d, valueOf(createTopicsResponse))
|
mt.(messageType).decode(d, valueOf(createTopicsResponse))
|
||||||
reqResPair.Response.Payload = createTopicsResponse
|
reqResPair.Response.Payload = createTopicsResponse
|
||||||
break
|
|
||||||
case DeleteTopics:
|
case DeleteTopics:
|
||||||
var mt interface{}
|
var mt interface{}
|
||||||
var deleteTopicsResponse interface{}
|
var deleteTopicsResponse interface{}
|
||||||
if apiVersion >= 6 {
|
if apiVersion >= v6 {
|
||||||
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV6{}).Elem())
|
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV6{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
deleteTopicsResponse = &DeleteTopicsReponseV6{}
|
deleteTopicsResponse = &DeleteTopicsReponseV6{}
|
||||||
} else if apiVersion >= 5 {
|
} else if apiVersion >= v5 {
|
||||||
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV5{}).Elem())
|
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV5{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
deleteTopicsResponse = &DeleteTopicsReponseV5{}
|
deleteTopicsResponse = &DeleteTopicsReponseV5{}
|
||||||
} else if apiVersion >= 1 {
|
} else if apiVersion >= v1 {
|
||||||
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV1{}).Elem())
|
types := makeTypes(reflect.TypeOf(&DeleteTopicsReponseV1{}).Elem())
|
||||||
mt = types[0]
|
mt = types[0]
|
||||||
deleteTopicsResponse = &DeleteTopicsReponseV1{}
|
deleteTopicsResponse = &DeleteTopicsReponseV1{}
|
||||||
@ -337,7 +332,7 @@ func WriteResponse(w io.Writer, apiVersion int16, correlationID int32, msg Messa
|
|||||||
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
size := packUint32(uint32(b.Size()) - 4)
|
size := packUint32(uint32(b.Size()) - 4)
|
||||||
b.WriteAt(size[:], 0)
|
_, _ = b.WriteAt(size[:], 0)
|
||||||
_, err = b.WriteTo(w)
|
_, err = b.WriteTo(w)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -296,8 +296,6 @@ type MessageV0 struct {
|
|||||||
Set *MessageSet `json:"set"` // the message set a message might wrap
|
Set *MessageSet `json:"set"` // the message set a message might wrap
|
||||||
Version int8 `json:"version"` // v1 requires Kafka 0.10
|
Version int8 `json:"version"` // v1 requires Kafka 0.10
|
||||||
Timestamp time.Time `json:"timestamp"` // the timestamp of the message (version 1+ only)
|
Timestamp time.Time `json:"timestamp"` // the timestamp of the message (version 1+ only)
|
||||||
|
|
||||||
compressedSize int // used for computing the compression ratio metrics
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MessageBlock represents a part of request with message
|
// MessageBlock represents a part of request with message
|
||||||
|
@ -52,9 +52,13 @@ func (d dissecting) Dissect(b *bufio.Reader, isClient bool, tcpID *api.TcpID, co
|
|||||||
}
|
}
|
||||||
|
|
||||||
if isClient {
|
if isClient {
|
||||||
handleClientStream(tcpID, counterPair, superTimer, emitter, redisPacket)
|
err = handleClientStream(tcpID, counterPair, superTimer, emitter, redisPacket)
|
||||||
} else {
|
} else {
|
||||||
handleServerStream(tcpID, counterPair, superTimer, emitter, redisPacket)
|
err = handleServerStream(tcpID, counterPair, superTimer, emitter, redisPacket)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -108,7 +112,7 @@ func (d dissecting) Analyze(item *api.OutputChannelItem, resolvedSource string,
|
|||||||
|
|
||||||
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
func (d dissecting) Represent(request map[string]interface{}, response map[string]interface{}) (object []byte, bodySize int64, err error) {
|
||||||
bodySize = 0
|
bodySize = 0
|
||||||
representation := make(map[string]interface{}, 0)
|
representation := make(map[string]interface{})
|
||||||
repRequest := representGeneric(request, `request.`)
|
repRequest := representGeneric(request, `request.`)
|
||||||
repResponse := representGeneric(response, `response.`)
|
repResponse := representGeneric(response, `response.`)
|
||||||
representation["request"] = repRequest
|
representation["request"] = repRequest
|
||||||
|
@ -4,11 +4,9 @@ import (
|
|||||||
"bufio"
|
"bufio"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -18,90 +16,12 @@ const (
|
|||||||
busyPrefix = "BUSY "
|
busyPrefix = "BUSY "
|
||||||
noscriptPrefix = "NOSCRIPT "
|
noscriptPrefix = "NOSCRIPT "
|
||||||
|
|
||||||
defaultHost = "localhost"
|
|
||||||
defaultPort = 6379
|
|
||||||
defaultSentinelPort = 26379
|
|
||||||
defaultTimeout = 5 * time.Second
|
|
||||||
defaultDatabase = 2 * time.Second
|
|
||||||
|
|
||||||
dollarByte = '$'
|
dollarByte = '$'
|
||||||
asteriskByte = '*'
|
asteriskByte = '*'
|
||||||
plusByte = '+'
|
plusByte = '+'
|
||||||
minusByte = '-'
|
minusByte = '-'
|
||||||
colonByte = ':'
|
colonByte = ':'
|
||||||
notApplicableByte = '0'
|
notApplicableByte = '0'
|
||||||
|
|
||||||
sentinelMasters = "masters"
|
|
||||||
sentinelGetMasterAddrByName = "get-master-addr-by-name"
|
|
||||||
sentinelReset = "reset"
|
|
||||||
sentinelSlaves = "slaves"
|
|
||||||
sentinelFailOver = "failover"
|
|
||||||
sentinelMonitor = "monitor"
|
|
||||||
sentinelRemove = "remove"
|
|
||||||
sentinelSet = "set"
|
|
||||||
|
|
||||||
clusterNodes = "nodes"
|
|
||||||
clusterMeet = "meet"
|
|
||||||
clusterReset = "reset"
|
|
||||||
clusterAddSlots = "addslots"
|
|
||||||
clusterDelSlots = "delslots"
|
|
||||||
clusterInfo = "info"
|
|
||||||
clusterGetKeysInSlot = "getkeysinslot"
|
|
||||||
clusterSetSlot = "setslot"
|
|
||||||
clusterSetSlotNode = "node"
|
|
||||||
clusterSetSlotMigrating = "migrating"
|
|
||||||
clusterSetSlotImporting = "importing"
|
|
||||||
clusterSetSlotStable = "stable"
|
|
||||||
clusterForget = "forget"
|
|
||||||
clusterFlushSlot = "flushslots"
|
|
||||||
clusterKeySlot = "keyslot"
|
|
||||||
clusterCountKeyInSlot = "countkeysinslot"
|
|
||||||
clusterSaveConfig = "saveconfig"
|
|
||||||
clusterReplicate = "replicate"
|
|
||||||
clusterSlaves = "slaves"
|
|
||||||
clusterFailOver = "failover"
|
|
||||||
clusterSlots = "slots"
|
|
||||||
pubSubChannels = "channels"
|
|
||||||
pubSubNumSub = "numsub"
|
|
||||||
pubSubNumPat = "numpat"
|
|
||||||
)
|
|
||||||
|
|
||||||
//intToByteArr convert int to byte array
|
|
||||||
func intToByteArr(a int) []byte {
|
|
||||||
buf := make([]byte, 0)
|
|
||||||
return strconv.AppendInt(buf, int64(a), 10)
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
bytesTrue = intToByteArr(1)
|
|
||||||
bytesFalse = intToByteArr(0)
|
|
||||||
bytesTilde = []byte("~")
|
|
||||||
|
|
||||||
positiveInfinityBytes = []byte("+inf")
|
|
||||||
negativeInfinityBytes = []byte("-inf")
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
sizeTable = []int{9, 99, 999, 9999, 99999, 999999, 9999999, 99999999,
|
|
||||||
999999999, math.MaxInt32}
|
|
||||||
|
|
||||||
digitTens = []byte{'0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '1',
|
|
||||||
'1', '1', '1', '1', '1', '1', '1', '1', '1', '2', '2', '2', '2', '2', '2', '2', '2', '2',
|
|
||||||
'2', '3', '3', '3', '3', '3', '3', '3', '3', '3', '3', '4', '4', '4', '4', '4', '4', '4',
|
|
||||||
'4', '4', '4', '5', '5', '5', '5', '5', '5', '5', '5', '5', '5', '6', '6', '6', '6', '6',
|
|
||||||
'6', '6', '6', '6', '6', '7', '7', '7', '7', '7', '7', '7', '7', '7', '7', '8', '8', '8',
|
|
||||||
'8', '8', '8', '8', '8', '8', '8', '9', '9', '9', '9', '9', '9', '9', '9', '9', '9'}
|
|
||||||
|
|
||||||
digitOnes = []byte{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0',
|
|
||||||
'1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8',
|
|
||||||
'9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6',
|
|
||||||
'7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4',
|
|
||||||
'5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2',
|
|
||||||
'3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9'}
|
|
||||||
|
|
||||||
digits = []byte{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a',
|
|
||||||
'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's',
|
|
||||||
't', 'u', 'v', 'w', 'x', 'y', 'z'}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// receive message from redis
|
// receive message from redis
|
||||||
@ -197,12 +117,6 @@ func (r *RedisInputStream) readLineBytes() ([]byte, error) {
|
|||||||
line := make([]byte, N)
|
line := make([]byte, N)
|
||||||
j := 0
|
j := 0
|
||||||
for i := r.count; i <= N; i++ {
|
for i := r.count; i <= N; i++ {
|
||||||
if i >= len(buf) {
|
|
||||||
return nil, errors.New("Redis buffer index mismatch.")
|
|
||||||
}
|
|
||||||
if i >= len(line) {
|
|
||||||
return nil, errors.New("Redis line index mismatch.")
|
|
||||||
}
|
|
||||||
line[j] = buf[i]
|
line[j] = buf[i]
|
||||||
j++
|
j++
|
||||||
}
|
}
|
||||||
@ -298,9 +212,9 @@ func (p *RedisProtocol) Read() (packet *RedisPacket, err error) {
|
|||||||
packet = &RedisPacket{}
|
packet = &RedisPacket{}
|
||||||
packet.Type = r
|
packet.Type = r
|
||||||
|
|
||||||
switch x.(type) {
|
switch v := x.(type) {
|
||||||
case []interface{}:
|
case []interface{}:
|
||||||
array := x.([]interface{})
|
array := v
|
||||||
if len(array) > 0 {
|
if len(array) > 0 {
|
||||||
switch array[0].(type) {
|
switch array[0].(type) {
|
||||||
case []uint8:
|
case []uint8:
|
||||||
@ -324,11 +238,11 @@ func (p *RedisProtocol) Read() (packet *RedisPacket, err error) {
|
|||||||
if len(array) > 3 {
|
if len(array) > 3 {
|
||||||
packet.Value = fmt.Sprintf("[%s", packet.Value)
|
packet.Value = fmt.Sprintf("[%s", packet.Value)
|
||||||
for _, item := range array[3:] {
|
for _, item := range array[3:] {
|
||||||
switch item.(type) {
|
switch j := item.(type) {
|
||||||
case []uint8:
|
case []uint8:
|
||||||
packet.Value = fmt.Sprintf("%s, %s", packet.Value, item.([]uint8))
|
packet.Value = fmt.Sprintf("%s, %s", packet.Value, j)
|
||||||
case int64:
|
case int64:
|
||||||
packet.Value = fmt.Sprintf("%s, %d", packet.Value, item.(int64))
|
packet.Value = fmt.Sprintf("%s, %d", packet.Value, j)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
packet.Value = strings.TrimSuffix(packet.Value, ", ")
|
packet.Value = strings.TrimSuffix(packet.Value, ", ")
|
||||||
@ -341,20 +255,20 @@ func (p *RedisProtocol) Read() (packet *RedisPacket, err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
case []uint8:
|
case []uint8:
|
||||||
val := string(x.([]uint8))
|
val := string(v)
|
||||||
if packet.Type == types[plusByte] {
|
if packet.Type == types[plusByte] {
|
||||||
packet.Keyword = RedisKeyword(strings.ToUpper(val))
|
packet.Keyword = RedisKeyword(strings.ToUpper(val))
|
||||||
if !isValidRedisKeyword(keywords, packet.Keyword) {
|
if !isValidRedisKeyword(keywords, packet.Keyword) {
|
||||||
err = errors.New(fmt.Sprintf("Unrecognized keyword: %s", string(packet.Command)))
|
err = fmt.Errorf("Unrecognized keyword: %s", string(packet.Command))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
packet.Value = val
|
packet.Value = val
|
||||||
}
|
}
|
||||||
case string:
|
case string:
|
||||||
packet.Value = x.(string)
|
packet.Value = v
|
||||||
case int64:
|
case int64:
|
||||||
packet.Value = fmt.Sprintf("%d", x.(int64))
|
packet.Value = fmt.Sprintf("%d", v)
|
||||||
default:
|
default:
|
||||||
msg := fmt.Sprintf("Unrecognized Redis data type: %v", reflect.TypeOf(x))
|
msg := fmt.Sprintf("Unrecognized Redis data type: %v", reflect.TypeOf(x))
|
||||||
err = errors.New(msg)
|
err = errors.New(msg)
|
||||||
@ -363,7 +277,7 @@ func (p *RedisProtocol) Read() (packet *RedisPacket, err error) {
|
|||||||
|
|
||||||
if packet.Command != "" {
|
if packet.Command != "" {
|
||||||
if !isValidRedisCommand(commands, packet.Command) {
|
if !isValidRedisCommand(commands, packet.Command) {
|
||||||
err = errors.New(fmt.Sprintf("Unrecognized command: %s", string(packet.Command)))
|
err = fmt.Errorf("Unrecognized command: %s", string(packet.Command))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -493,7 +407,7 @@ func (p *RedisProtocol) processError() (interface{}, error) {
|
|||||||
func (p *RedisProtocol) parseTargetHostAndSlot(clusterRedirectResponse string) (host string, po int, slot int, err error) {
|
func (p *RedisProtocol) parseTargetHostAndSlot(clusterRedirectResponse string) (host string, po int, slot int, err error) {
|
||||||
arr := strings.Split(clusterRedirectResponse, " ")
|
arr := strings.Split(clusterRedirectResponse, " ")
|
||||||
host, port := p.extractParts(arr[2])
|
host, port := p.extractParts(arr[2])
|
||||||
slot, err = strconv.Atoi(arr[1])
|
slot, _ = strconv.Atoi(arr[1])
|
||||||
po, err = strconv.Atoi(port)
|
po, err = strconv.Atoi(port)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -96,6 +96,9 @@ func (h *tcpReader) run(wg *sync.WaitGroup) {
|
|||||||
b := bufio.NewReader(h)
|
b := bufio.NewReader(h)
|
||||||
err := h.extension.Dissector.Dissect(b, h.isClient, h.tcpID, h.counterPair, h.superTimer, h.parent.superIdentifier, h.emitter, filteringOptions)
|
err := h.extension.Dissector.Dissect(b, h.isClient, h.tcpID, h.counterPair, h.superTimer, h.parent.superIdentifier, h.emitter, filteringOptions)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
io.Copy(ioutil.Discard, b)
|
_, err = io.Copy(ioutil.Discard, b)
|
||||||
|
if err != nil {
|
||||||
|
logger.Log.Errorf("%v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -47,6 +47,7 @@ const EntryTitle: React.FC<any> = ({protocol, data, bodySize, elapsedTime}) => {
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
style={{opacity: 0.5}}
|
style={{opacity: 0.5}}
|
||||||
|
id="entryDetailedTitleBodySize"
|
||||||
>
|
>
|
||||||
{formatSize(bodySize)}
|
{formatSize(bodySize)}
|
||||||
</div>
|
</div>
|
||||||
@ -58,6 +59,7 @@ const EntryTitle: React.FC<any> = ({protocol, data, bodySize, elapsedTime}) => {
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
style={{opacity: 0.5}}
|
style={{opacity: 0.5}}
|
||||||
|
id="entryDetailedTitleElapsedTime"
|
||||||
>
|
>
|
||||||
{Math.round(elapsedTime)}ms
|
{Math.round(elapsedTime)}ms
|
||||||
</div>
|
</div>
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user