blob: c9eb70241f5bc45821a8e85655de8ec66ee6e060 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package datareq
import (
"fmt"
"net/url"
"strconv"
"strings"
"github.com/apache/trafficcontrol/lib/go-tc"
"github.com/apache/trafficcontrol/traffic_monitor/cache"
)
// CacheStatFilter fulfills the cache.Filter interface, for filtering stats. See the `NewCacheStatFilter` documentation for details on which query parameters are used to filter.
type CacheStatFilter struct {
historyCount uint64
statsToUse map[string]struct{}
interfaceStatsToUse map[string]struct{}
wildcard bool
cacheType tc.CacheType
hosts map[tc.CacheName]struct{}
cacheTypes map[tc.CacheName]tc.CacheType
}
// UseCache returns whether the given cache is in the filter.
func (f *CacheStatFilter) UseCache(name tc.CacheName) bool {
if _, inHosts := f.hosts[name]; len(f.hosts) != 0 && !inHosts {
return false
}
if f.cacheType != tc.CacheTypeInvalid && f.cacheTypes[name] != f.cacheType {
return false
}
return true
}
// UseStat returns whether the given stat is in the filter.
func (f *CacheStatFilter) UseStat(statName string) bool {
if len(f.statsToUse) == 0 {
return true
}
if !f.wildcard {
_, ok := f.statsToUse[statName]
return ok
}
for statToUse := range f.statsToUse {
if strings.Contains(statName, statToUse) {
return true
}
}
return false
}
func (f *CacheStatFilter) UseInterfaceStat(statName string) bool {
if len(f.interfaceStatsToUse) == 0 {
return true
}
if !f.wildcard {
_, ok := f.interfaceStatsToUse[statName]
return ok
}
for stat := range f.interfaceStatsToUse {
if strings.Contains(statName, stat) {
return true
}
}
return false
}
// WithinStatHistoryMax returns whether the given history index is less than the
// max history of this filter.
func (f *CacheStatFilter) WithinStatHistoryMax(n uint64) bool {
if f.historyCount == 0 {
return true
}
if n <= f.historyCount {
return true
}
return false
}
// NewCacheStatFilter takes the HTTP query parameters and creates a CacheStatFilter which fulfills the `cache.Filter` interface, filtering according to the query parameters passed.
// Query parameters used are `hc`, `stats`, `wildcard`, `type`, and `hosts`.
// If `hc` is 0, all history is returned. If `hc` is empty, 1 history is returned.
// If `stats` is empty, all stats are returned.
// If `wildcard` is empty, `stats` is considered exact.
// If `type` is empty, all cache types are returned.
func NewCacheStatFilter(path string, params url.Values, cacheTypes map[tc.CacheName]tc.CacheType) (cache.Filter, error) {
validParams := map[string]struct{}{
"hc": struct{}{},
"stats": struct{}{},
"interfaceStats": struct{}{},
"wildcard": struct{}{},
"type": struct{}{},
"hosts": struct{}{},
"cache": struct{}{},
}
if len(params) > len(validParams) {
return nil, fmt.Errorf("invalid query parameters")
}
for param := range params {
if _, ok := validParams[param]; !ok {
return nil, fmt.Errorf("invalid query parameter '%v'", param)
}
}
var historyCount uint64 = 1
if paramHc, exists := params["hc"]; exists && len(paramHc) > 0 {
v, err := strconv.ParseUint(paramHc[0], 10, 64)
if err == nil {
historyCount = v
}
}
statsToUse := map[string]struct{}{}
if paramStats, exists := params["stats"]; exists && len(paramStats) > 0 {
commaStats := strings.Split(paramStats[0], ",")
for _, stat := range commaStats {
statsToUse[stat] = struct{}{}
}
}
interfaceStatsToUse := map[string]struct{}{}
if paramStats, exists := params["interfaceStats"]; exists {
for _, paramStat := range paramStats {
for _, stat := range strings.Split(paramStat, ",") {
interfaceStatsToUse[stat] = struct{}{}
}
}
}
wildcard := false
if paramWildcard, exists := params["wildcard"]; exists && len(paramWildcard) > 0 {
wildcard, _ = strconv.ParseBool(paramWildcard[0]) // ignore errors, error => false
}
cacheType := tc.CacheTypeInvalid
if paramType, exists := params["type"]; exists && len(paramType) > 0 {
cacheType = tc.CacheTypeFromString(paramType[0])
if cacheType == tc.CacheTypeInvalid {
return nil, fmt.Errorf("invalid query parameter type '%v' - valid types are: {edge, mid}", paramType[0])
}
}
hosts := map[tc.CacheName]struct{}{}
if paramHosts, exists := params["hosts"]; exists && len(paramHosts) > 0 {
commaHosts := strings.Split(paramHosts[0], ",")
for _, host := range commaHosts {
hosts[tc.CacheName(host)] = struct{}{}
}
}
if paramHosts, exists := params["cache"]; exists && len(paramHosts) > 0 {
commaHosts := strings.Split(paramHosts[0], ",")
for _, host := range commaHosts {
hosts[tc.CacheName(host)] = struct{}{}
}
}
pathArgument := getPathArgument(path)
if pathArgument != "" {
hosts[tc.CacheName(pathArgument)] = struct{}{}
}
// parameters without values are considered hosts, e.g. `?my-cache-0`
for maybeHost, val := range params {
if len(val) == 0 || (len(val) == 1 && val[0] == "") {
hosts[tc.CacheName(maybeHost)] = struct{}{}
}
}
return &CacheStatFilter{
historyCount: historyCount,
statsToUse: statsToUse,
wildcard: wildcard,
cacheType: cacheType,
hosts: hosts,
cacheTypes: cacheTypes,
}, nil
}