2023-01-27 01:10:27 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2023-02-18 17:27:28 +07:00
|
|
|
* Copyright (c) 2022-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{}
|
|
|
|
}
|
|
|
|
|
2023-02-25 21:53:18 +07:00
|
|
|
func (i *Item) String(compact bool, quoteVal bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteString("type: " + i.Type.String() + "\n")
|
|
|
|
var content string
|
|
|
|
switch val := i.Value.(type) {
|
|
|
|
case *RoutingRule:
|
2023-02-25 21:53:18 +07:00
|
|
|
content = val.String(false, compact, quoteVal)
|
2023-01-27 01:10:27 +07:00
|
|
|
case *Param:
|
2023-02-25 21:53:18 +07:00
|
|
|
content = val.String(false, quoteVal)
|
2023-01-27 01:10:27 +07:00
|
|
|
case *Section:
|
2023-02-25 21:53:18 +07:00
|
|
|
content = val.String(compact, quoteVal)
|
2023-01-27 01:10:27 +07:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2023-02-25 21:53:18 +07:00
|
|
|
func (s *Section) String(compact bool, quoteVal bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteString("section: " + s.Name + "\n")
|
|
|
|
var strItemList []string
|
|
|
|
for _, item := range s.Items {
|
2023-02-25 21:53:18 +07:00
|
|
|
lines := strings.Split(item.String(compact, quoteVal), "\n")
|
2023-01-27 01:10:27 +07:00
|
|
|
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-02-25 21:53:18 +07:00
|
|
|
func (p *Param) String(compact bool, quoteVal bool) string {
|
|
|
|
var quote func(string) string
|
|
|
|
if quoteVal {
|
|
|
|
quote = strconv.Quote
|
|
|
|
} else {
|
|
|
|
quote = func(s string) string { return s }
|
|
|
|
}
|
2023-01-27 01:10:27 +07:00
|
|
|
if p.Key == "" {
|
2023-02-25 21:53:18 +07:00
|
|
|
return quote(p.Val)
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
if p.AndFunctions != nil {
|
|
|
|
a := paramAndFunctions{
|
|
|
|
Key: p.Key,
|
|
|
|
AndFunctions: p.AndFunctions,
|
|
|
|
}
|
2023-02-25 21:53:18 +07:00
|
|
|
return a.String(compact, quoteVal)
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
|
|
|
if compact {
|
2023-02-25 21:53:18 +07:00
|
|
|
return p.Key + ":" + quote(p.Val)
|
2023-01-28 00:50:21 +07:00
|
|
|
} else {
|
2023-02-25 21:53:18 +07:00
|
|
|
return p.Key + ": " + quote(p.Val)
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type Function struct {
|
|
|
|
Name string
|
2023-01-29 06:31:52 +07:00
|
|
|
Not bool
|
2023-01-27 01:10:27 +07:00
|
|
|
Params []*Param
|
|
|
|
}
|
|
|
|
|
2023-02-25 21:53:18 +07:00
|
|
|
func (f *Function) String(compact bool, quoteVal bool, omitEmpty bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
2023-01-29 10:19:58 +07:00
|
|
|
if f.Not {
|
|
|
|
builder.WriteString("!")
|
|
|
|
}
|
2023-02-25 21:53:18 +07:00
|
|
|
builder.WriteString(f.Name)
|
|
|
|
if !(omitEmpty && len(f.Params) == 0) {
|
|
|
|
builder.WriteString("(")
|
|
|
|
var strParamList []string
|
|
|
|
for _, p := range f.Params {
|
|
|
|
strParamList = append(strParamList, p.String(compact, quoteVal))
|
|
|
|
}
|
|
|
|
if compact {
|
|
|
|
builder.WriteString(strings.Join(strParamList, ","))
|
|
|
|
} else {
|
|
|
|
builder.WriteString(strings.Join(strParamList, ", "))
|
|
|
|
}
|
|
|
|
builder.WriteString(")")
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type paramAndFunctions struct {
|
|
|
|
Key string
|
|
|
|
AndFunctions []*Function
|
|
|
|
}
|
|
|
|
|
2023-02-25 21:53:18 +07:00
|
|
|
func (p *paramAndFunctions) String(compact bool, quoteVal 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-02-25 21:53:18 +07:00
|
|
|
strFunctionList = append(strFunctionList, f.String(compact, quoteVal, false))
|
2023-01-28 00:50:21 +07:00
|
|
|
}
|
|
|
|
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
|
2023-02-20 17:06:54 +07:00
|
|
|
Outbound Function
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
|
2023-02-25 21:53:18 +07:00
|
|
|
func (r *RoutingRule) String(replaceParamWithN bool, compact bool, quoteVal bool) string {
|
2023-01-27 01:10:27 +07:00
|
|
|
var builder strings.Builder
|
|
|
|
var n int
|
2023-01-29 10:19:58 +07:00
|
|
|
for i, f := range r.AndFunctions {
|
|
|
|
if i != 0 {
|
2023-02-25 21:53:18 +07:00
|
|
|
if compact {
|
|
|
|
builder.WriteString("&&")
|
|
|
|
} else {
|
|
|
|
builder.WriteString(" && ")
|
|
|
|
}
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
var paramBuilder strings.Builder
|
|
|
|
n += len(f.Params)
|
2023-02-25 01:38:21 +07:00
|
|
|
if replaceParamWithN {
|
2023-01-29 10:19:58 +07:00
|
|
|
paramBuilder.WriteString("[n = " + strconv.Itoa(n) + "]")
|
|
|
|
} else {
|
|
|
|
for j, param := range f.Params {
|
|
|
|
if j != 0 {
|
2023-02-25 21:53:18 +07:00
|
|
|
if compact {
|
|
|
|
paramBuilder.WriteString(",")
|
|
|
|
} else {
|
|
|
|
paramBuilder.WriteString(", ")
|
|
|
|
}
|
2023-01-29 10:19:58 +07:00
|
|
|
}
|
2023-02-25 21:53:18 +07:00
|
|
|
paramBuilder.WriteString(param.String(compact, quoteVal))
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
|
|
|
}
|
2023-01-29 10:19:58 +07:00
|
|
|
symNot := ""
|
|
|
|
if f.Not {
|
|
|
|
symNot = "!"
|
|
|
|
}
|
|
|
|
builder.WriteString(fmt.Sprintf("%v%v(%v)", symNot, f.Name, paramBuilder.String()))
|
2023-01-27 01:10:27 +07:00
|
|
|
}
|
2023-02-25 21:53:18 +07:00
|
|
|
if compact {
|
|
|
|
builder.WriteString("->" + r.Outbound.String(compact, quoteVal, true))
|
|
|
|
} else {
|
|
|
|
builder.WriteString(" -> " + r.Outbound.String(compact, quoteVal, true))
|
|
|
|
}
|
2023-01-27 01:10:27 +07:00
|
|
|
return builder.String()
|
|
|
|
}
|