netdata/netdata

View on GitHub
src/go/collectors/go.d.plugin/modules/tengine/apiclient.go

Summary

Maintainability
B
4 hrs
Test Coverage
// SPDX-License-Identifier: GPL-3.0-or-later

package tengine

import (
    "bufio"
    "fmt"
    "io"
    "net/http"
    "strconv"
    "strings"

    "github.com/netdata/netdata/go/go.d.plugin/pkg/web"
)

const (
    bytesIn               = "bytes_in"
    bytesOut              = "bytes_out"
    connTotal             = "conn_total"
    reqTotal              = "req_total"
    http2xx               = "http_2xx"
    http3xx               = "http_3xx"
    http4xx               = "http_4xx"
    http5xx               = "http_5xx"
    httpOtherStatus       = "http_other_status"
    rt                    = "rt"
    upsReq                = "ups_req"
    upsRT                 = "ups_rt"
    upsTries              = "ups_tries"
    http200               = "http_200"
    http206               = "http_206"
    http302               = "http_302"
    http304               = "http_304"
    http403               = "http_403"
    http404               = "http_404"
    http416               = "http_416"
    http499               = "http_499"
    http500               = "http_500"
    http502               = "http_502"
    http503               = "http_503"
    http504               = "http_504"
    http508               = "http_508"
    httpOtherDetailStatus = "http_other_detail_status"
    httpUps4xx            = "http_ups_4xx"
    httpUps5xx            = "http_ups_5xx"
)

var defaultLineFormat = []string{
    bytesIn,
    bytesOut,
    connTotal,
    reqTotal,
    http2xx,
    http3xx,
    http4xx,
    http5xx,
    httpOtherStatus,
    rt,
    upsReq,
    upsRT,
    upsTries,
    http200,
    http206,
    http302,
    http304,
    http403,
    http404,
    http416,
    http499,
    http500,
    http502,
    http503,
    http504,
    http508,
    httpOtherDetailStatus,
    httpUps4xx,
    httpUps5xx,
}

func newAPIClient(client *http.Client, request web.Request) *apiClient {
    return &apiClient{httpClient: client, request: request}
}

type apiClient struct {
    httpClient *http.Client
    request    web.Request
}

func (a apiClient) getStatus() (*tengineStatus, error) {
    req, err := web.NewHTTPRequest(a.request)
    if err != nil {
        return nil, fmt.Errorf("error on creating request : %v", err)
    }

    resp, err := a.doRequestOK(req)
    defer closeBody(resp)
    if err != nil {
        return nil, err
    }

    status, err := parseStatus(resp.Body)
    if err != nil {
        return nil, fmt.Errorf("error on parsing response : %v", err)
    }

    return status, nil
}

func (a apiClient) doRequestOK(req *http.Request) (*http.Response, error) {
    resp, err := a.httpClient.Do(req)
    if err != nil {
        return nil, fmt.Errorf("error on request : %v", err)
    }
    if resp.StatusCode != http.StatusOK {
        return resp, fmt.Errorf("%s returned HTTP code %d", req.URL, resp.StatusCode)
    }
    return resp, nil
}

func closeBody(resp *http.Response) {
    if resp != nil && resp.Body != nil {
        _, _ = io.Copy(io.Discard, resp.Body)
        _ = resp.Body.Close()
    }
}

func parseStatus(r io.Reader) (*tengineStatus, error) {
    var status tengineStatus

    s := bufio.NewScanner(r)
    for s.Scan() {
        m, err := parseStatusLine(s.Text(), defaultLineFormat)
        if err != nil {
            return nil, err
        }
        status = append(status, *m)
    }

    return &status, nil
}

func parseStatusLine(line string, lineFormat []string) (*metric, error) {
    parts := strings.Split(line, ",")

    // NOTE: only default line format is supported
    // TODO: custom line format?
    // www.example.com,127.0.0.1:80,162,6242,1,1,1,0,0,0,0,10,1,10,1....
    i := findFirstInt(parts)
    if i == -1 {
        return nil, fmt.Errorf("invalid line : %s", line)
    }
    if len(parts[i:]) != len(lineFormat) {
        return nil, fmt.Errorf("invalid line length, got %d, expected %d, line : %s",
            len(parts[i:]), len(lineFormat), line)
    }

    // skip "$host,$server_addr:$server_port"
    parts = parts[i:]

    var m metric
    for i, key := range lineFormat {
        value := mustParseInt(parts[i])
        switch key {
        default:
            return nil, fmt.Errorf("unknown line format key: %s", key)
        case bytesIn:
            m.BytesIn = value
        case bytesOut:
            m.BytesOut = value
        case connTotal:
            m.ConnTotal = value
        case reqTotal:
            m.ReqTotal = value
        case http2xx:
            m.HTTP2xx = value
        case http3xx:
            m.HTTP3xx = value
        case http4xx:
            m.HTTP4xx = value
        case http5xx:
            m.HTTP5xx = value
        case httpOtherStatus:
            m.HTTPOtherStatus = value
        case rt:
            m.RT = value
        case upsReq:
            m.UpsReq = value
        case upsRT:
            m.UpsRT = value
        case upsTries:
            m.UpsTries = value
        case http200:
            m.HTTP200 = value
        case http206:
            m.HTTP206 = value
        case http302:
            m.HTTP302 = value
        case http304:
            m.HTTP304 = value
        case http403:
            m.HTTP403 = value
        case http404:
            m.HTTP404 = value
        case http416:
            m.HTTP416 = value
        case http499:
            m.HTTP499 = value
        case http500:
            m.HTTP500 = value
        case http502:
            m.HTTP502 = value
        case http503:
            m.HTTP503 = value
        case http504:
            m.HTTP504 = value
        case http508:
            m.HTTP508 = value
        case httpOtherDetailStatus:
            m.HTTPOtherDetailStatus = value
        case httpUps4xx:
            m.HTTPUps4xx = value
        case httpUps5xx:
            m.HTTPUps5xx = value
        }
    }
    return &m, nil
}

func findFirstInt(s []string) int {
    for i, v := range s {
        _, err := strconv.ParseInt(v, 10, 64)
        if err != nil {
            continue
        }
        return i
    }
    return -1
}

func mustParseInt(value string) *int64 {
    v, err := strconv.ParseInt(value, 10, 64)
    if err != nil {
        panic(err)
    }

    return &v
}