mirror of
https://github.com/kubeshark/kubeshark.git
synced 2025-08-25 11:59:35 +00:00
206 lines
4.3 KiB
Go
206 lines
4.3 KiB
Go
package oas
|
|
|
|
import (
|
|
"github.com/chanced/openapi"
|
|
"github.com/up9inc/mizu/shared/logger"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
type NodePath = []string
|
|
|
|
type Node struct {
|
|
constant *string
|
|
param *openapi.ParameterObj
|
|
ops *openapi.PathObj
|
|
parent *Node
|
|
children []*Node
|
|
}
|
|
|
|
func (n *Node) getOrSet(path NodePath, pathObjToSet *openapi.PathObj) (node *Node) {
|
|
if pathObjToSet == nil {
|
|
panic("Invalid function call")
|
|
}
|
|
|
|
pathChunk := path[0]
|
|
chunkIsParam := strings.HasPrefix(pathChunk, "{") && strings.HasSuffix(pathChunk, "}")
|
|
chunkIsGibberish := IsGibberish(pathChunk) && !IsVersionString(pathChunk)
|
|
|
|
var paramObj *openapi.ParameterObj
|
|
if chunkIsParam && pathObjToSet != nil {
|
|
paramObj = findParamByName(pathObjToSet.Parameters, openapi.InPath, pathChunk[1:len(pathChunk)-1])
|
|
}
|
|
|
|
if paramObj == nil {
|
|
node = n.searchInConstants(pathChunk)
|
|
}
|
|
|
|
if node == nil {
|
|
node = n.searchInParams(paramObj, chunkIsGibberish)
|
|
}
|
|
|
|
// still no node found, should create it
|
|
if node == nil {
|
|
node = new(Node)
|
|
node.parent = n
|
|
n.children = append(n.children, node)
|
|
|
|
if paramObj != nil {
|
|
node.param = paramObj
|
|
} else if chunkIsGibberish {
|
|
initParams(&pathObjToSet.Parameters)
|
|
|
|
newParam := n.createParam()
|
|
node.param = newParam
|
|
|
|
appended := append(*pathObjToSet.Parameters, newParam)
|
|
pathObjToSet.Parameters = &appended
|
|
} else {
|
|
node.constant = &pathChunk
|
|
}
|
|
}
|
|
|
|
// add example if it's a param
|
|
if node.param != nil && !chunkIsParam {
|
|
exmp := &node.param.Examples
|
|
err := fillParamExample(&exmp, pathChunk)
|
|
if err != nil {
|
|
logger.Log.Warningf("Failed to add example to a parameter: %s", err)
|
|
}
|
|
}
|
|
|
|
// TODO: eat up trailing slash, in a smart way: node.ops!=nil && path[1]==""
|
|
if len(path) > 1 {
|
|
return node.getOrSet(path[1:], pathObjToSet)
|
|
} else if node.ops == nil {
|
|
node.ops = pathObjToSet
|
|
}
|
|
|
|
return node
|
|
}
|
|
|
|
func (n *Node) createParam() *openapi.ParameterObj {
|
|
name := "param"
|
|
|
|
// REST assumption, not always correct
|
|
if strings.HasSuffix(*n.constant, "es") && len(*n.constant) > 4 {
|
|
name = *n.constant
|
|
name = name[:len(name)-2] + "Id"
|
|
} else if strings.HasSuffix(*n.constant, "s") && len(*n.constant) > 3 {
|
|
name = *n.constant
|
|
name = name[:len(name)-1] + "Id"
|
|
}
|
|
|
|
newParam := createSimpleParam(name, "path", "string")
|
|
x := n.countParentParams()
|
|
if x > 1 {
|
|
newParam.Name = newParam.Name + strconv.Itoa(x)
|
|
}
|
|
|
|
return newParam
|
|
}
|
|
|
|
func (n *Node) searchInParams(paramObj *openapi.ParameterObj, chunkIsGibberish bool) *Node {
|
|
// look among params
|
|
if paramObj != nil || chunkIsGibberish {
|
|
for _, subnode := range n.children {
|
|
if subnode.constant != nil {
|
|
continue
|
|
}
|
|
|
|
// TODO: check the regex pattern of param? for exceptions etc
|
|
|
|
if paramObj != nil {
|
|
// TODO: mergeParam(subnode.param, paramObj)
|
|
return subnode
|
|
} else {
|
|
return subnode
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (n *Node) searchInConstants(pathChunk string) *Node {
|
|
// look among constants
|
|
for _, subnode := range n.children {
|
|
if subnode.constant == nil {
|
|
continue
|
|
}
|
|
|
|
if *subnode.constant == pathChunk {
|
|
return subnode
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (n *Node) compact() {
|
|
// TODO
|
|
}
|
|
|
|
func (n *Node) listPaths() *openapi.Paths {
|
|
paths := &openapi.Paths{Items: map[openapi.PathValue]*openapi.PathObj{}}
|
|
|
|
var strChunk string
|
|
if n.constant != nil {
|
|
strChunk = *n.constant
|
|
} else if n.param != nil {
|
|
strChunk = "{" + n.param.Name + "}"
|
|
} else {
|
|
// this is the root node
|
|
}
|
|
|
|
// add self
|
|
if n.ops != nil {
|
|
paths.Items[openapi.PathValue(strChunk)] = n.ops
|
|
}
|
|
|
|
// recurse into children
|
|
for _, child := range n.children {
|
|
subPaths := child.listPaths()
|
|
for path, pathObj := range subPaths.Items {
|
|
var concat string
|
|
if n.parent == nil {
|
|
concat = string(path)
|
|
} else {
|
|
concat = strChunk + "/" + string(path)
|
|
}
|
|
paths.Items[openapi.PathValue(concat)] = pathObj
|
|
}
|
|
}
|
|
|
|
return paths
|
|
}
|
|
|
|
type PathAndOp struct {
|
|
path string
|
|
op *openapi.Operation
|
|
}
|
|
|
|
func (n *Node) listOps() []PathAndOp {
|
|
res := make([]PathAndOp, 0)
|
|
for path, pathObj := range n.listPaths().Items {
|
|
for _, op := range getOps(pathObj) {
|
|
res = append(res, PathAndOp{path: string(path), op: op})
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
func (n *Node) countParentParams() int {
|
|
res := 0
|
|
node := n
|
|
for {
|
|
if node.param != nil {
|
|
res++
|
|
}
|
|
|
|
if node.parent == nil {
|
|
break
|
|
}
|
|
node = node.parent
|
|
}
|
|
return res
|
|
}
|