2023-01-27 01:10:27 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2023-01-28 12:56:06 +07:00
|
|
|
* Copyright (c) since 2023, v2rayA Organization <team@v2raya.org>
|
2023-01-27 01:10:27 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package config_parser
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type ItemType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
ItemType_RoutingRule ItemType = iota
|
|
|
|
ItemType_Param
|
|
|
|
ItemType_Section
|
|
|
|
)
|
|
|
|
|
|
|
|
func (t ItemType) String() string {
|
|
|
|
switch t {
|
|
|
|
case ItemType_RoutingRule:
|
|
|
|
return "RoutingRule"
|
|
|
|
case ItemType_Param:
|
|
|
|
return "Param"
|
|
|
|
case ItemType_Section:
|
|
|
|
return "Section"
|
|
|
|
default:
|
|
|
|
return "<Unknown>"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewRoutingRuleItem(rule *RoutingRule) *Item {
|
|
|
|
return &Item{
|
|
|
|
Type: ItemType_RoutingRule,
|
|
|
|
Value: rule,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewParamItem(param *Param) *Item {
|
|
|
|
return &Item{
|
|
|
|
Type: ItemType_Param,
|
|
|
|
Value: param,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewSectionItem(section *Section) *Item {
|
|
|
|
return &Item{
|
|
|
|
Type: ItemType_Param,
|
|
|
|
Value: section,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type Item struct {
|
|
|
|
Type ItemType
|
|
|
|
Value interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (i *Item) String() string {
|
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteString("type: " + i.Type.String() + "\n")
|
|
|
|
var content string
|
|
|
|
switch val := i.Value.(type) {
|
|
|
|
case *RoutingRule:
|
|
|
|
content = val.String(false)
|
|
|
|
case *Param:
|
2023-01-28 00:50:21 +07:00
|
|
|
content = val.String(false)
|
2023-01-27 01:10:27 +07:00
|
|
|
case *Section:
|
|
|
|
content = val.String()
|
|
|
|
default:
|
|
|
|
return "<Unknown>\n"
|
|
|
|
}
|
|
|
|
lines := strings.Split(content, "\n")
|
|
|
|
for i := range lines {
|
|
|
|
lines[i] = "\t" + lines[i]
|
|
|
|
}
|
|
|
|
builder.WriteString(strings.Join(lines, "\n"))
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type Section struct {
|
|
|
|
Name string
|
|
|
|
Items []*Item
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Section) String() string {
|
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteString("section: " + s.Name + "\n")
|
|
|
|
var strItemList []string
|
|
|
|
for _, item := range s.Items {
|
|
|
|
lines := strings.Split(item.String(), "\n")
|
|
|
|
for i := range lines {
|
|
|
|
lines[i] = "\t" + lines[i]
|
|
|
|
}
|
|
|
|
strItemList = append(strItemList, strings.Join(lines, "\n"))
|
|
|
|
}
|
|
|
|
builder.WriteString(strings.Join(strItemList, "\n"))
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type Param struct {
|
2023-01-28 00:50:21 +07:00
|
|
|
// Key may be empty.
|
|
|
|
Key string
|
|
|
|
|
|
|
|
// Either Val or AndFunctions is empty.
|
2023-01-27 01:10:27 +07:00
|
|
|
Val string
|
|
|
|
AndFunctions []*Function
|
|
|
|
}
|
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
func (p *Param) String(compact bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
if p.Key == "" {
|
|
|
|
return p.Val
|
|
|
|
}
|
|
|
|
if p.AndFunctions != nil {
|
|
|
|
a := paramAndFunctions{
|
|
|
|
Key: p.Key,
|
|
|
|
AndFunctions: p.AndFunctions,
|
|
|
|
}
|
2023-01-28 00:50:21 +07:00
|
|
|
return a.String(compact)
|
|
|
|
}
|
|
|
|
if compact {
|
|
|
|
return p.Key + ":" + p.Val
|
|
|
|
} else {
|
|
|
|
return p.Key + ": " + p.Val
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type Function struct {
|
|
|
|
Name string
|
|
|
|
Params []*Param
|
|
|
|
}
|
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
func (f *Function) String(compact bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteString(f.Name + "(")
|
|
|
|
var strParamList []string
|
|
|
|
for _, p := range f.Params {
|
2023-01-28 00:50:21 +07:00
|
|
|
strParamList = append(strParamList, p.String(compact))
|
|
|
|
}
|
|
|
|
if compact {
|
|
|
|
builder.WriteString(strings.Join(strParamList, ","))
|
|
|
|
} else {
|
|
|
|
builder.WriteString(strings.Join(strParamList, ", "))
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
builder.WriteString(")")
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type paramAndFunctions struct {
|
|
|
|
Key string
|
|
|
|
AndFunctions []*Function
|
|
|
|
}
|
|
|
|
|
2023-01-28 00:50:21 +07:00
|
|
|
func (p *paramAndFunctions) String(compact bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
2023-01-28 00:50:21 +07:00
|
|
|
if compact {
|
|
|
|
builder.WriteString(p.Key + ":")
|
|
|
|
} else {
|
|
|
|
builder.WriteString(p.Key + ": ")
|
|
|
|
}
|
2023-01-27 01:10:27 +07:00
|
|
|
var strFunctionList []string
|
|
|
|
for _, f := range p.AndFunctions {
|
2023-01-28 00:50:21 +07:00
|
|
|
strFunctionList = append(strFunctionList, f.String(compact))
|
|
|
|
}
|
|
|
|
if compact {
|
|
|
|
builder.WriteString(strings.Join(strFunctionList, "&&"))
|
|
|
|
} else {
|
|
|
|
builder.WriteString(strings.Join(strFunctionList, " && "))
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type RoutingRule struct {
|
|
|
|
AndFunctions []*Function
|
|
|
|
Outbound string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *RoutingRule) String(calcN bool) string {
|
|
|
|
var builder strings.Builder
|
|
|
|
var n int
|
|
|
|
for _, f := range r.AndFunctions {
|
|
|
|
if builder.Len() != 0 {
|
|
|
|
builder.WriteString(" && ")
|
|
|
|
}
|
|
|
|
var paramBuilder strings.Builder
|
|
|
|
n += len(f.Params)
|
|
|
|
for _, p := range f.Params {
|
|
|
|
if paramBuilder.Len() != 0 {
|
|
|
|
paramBuilder.WriteString(", ")
|
|
|
|
}
|
|
|
|
if p.Key != "" {
|
|
|
|
paramBuilder.WriteString(p.Key + ": " + p.Val)
|
|
|
|
} else {
|
|
|
|
paramBuilder.WriteString(p.Val)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
builder.WriteString(fmt.Sprintf("%v(%v)", f.Name, paramBuilder.String()))
|
|
|
|
}
|
|
|
|
builder.WriteString(" -> " + r.Outbound)
|
|
|
|
if calcN {
|
|
|
|
builder.WriteString(" [n = " + strconv.Itoa(n) + "]")
|
|
|
|
}
|
|
|
|
return builder.String()
|
|
|
|
}
|