1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-02 23:47:07 -05:00
v2fly/common/protofilter/filter.go

134 lines
3.2 KiB
Go
Raw Normal View History

package protofilter
import (
2021-09-05 10:02:51 -04:00
"context"
2021-10-28 06:34:19 -04:00
"io"
"net"
"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/reflect/protoreflect"
"github.com/v2fly/v2ray-core/v5/common/environment/envctx"
"github.com/v2fly/v2ray-core/v5/common/environment/filesystemcap"
"github.com/v2fly/v2ray-core/v5/common/protoext"
)
//go:generate go run github.com/v2fly/v2ray-core/v5/common/errors/errorgen
2021-09-05 10:02:51 -04:00
func FilterProtoConfig(ctx context.Context, config proto.Message) error {
messageProtoReflect := config.ProtoReflect()
2021-09-05 10:02:51 -04:00
return filterMessage(ctx, messageProtoReflect)
}
2021-09-05 10:02:51 -04:00
func filterMessage(ctx context.Context, message protoreflect.Message) error {
var err error
type fileRead struct {
filename string
field string
}
var fileReadingQueue []fileRead
2021-09-18 16:04:30 -04:00
type pendingWrite struct {
field protoreflect.FieldDescriptor
value protoreflect.Value
}
var pendingWriteQueue []pendingWrite
message.Range(func(descriptor protoreflect.FieldDescriptor, value protoreflect.Value) bool {
v2extension, ferr := protoext.GetFieldOptions(descriptor)
2021-09-05 11:29:12 -04:00
if ferr == nil {
if v2extension.Forbidden {
if value.Bool() {
err = newError("a forbidden value is set ", descriptor.FullName())
return false
}
}
if v2extension.ConvertTimeReadFileInto != "" {
fileReadingQueue = append(fileReadingQueue, fileRead{
filename: value.String(),
field: v2extension.ConvertTimeReadFileInto,
})
}
2021-09-18 16:04:30 -04:00
if v2extension.ConvertTimeParseIp != "" {
ipValue := net.ParseIP(value.String())
2021-09-18 16:04:30 -04:00
target := message.Descriptor().Fields().ByTextName(v2extension.ConvertTimeParseIp)
pendingWriteQueue = append(pendingWriteQueue, pendingWrite{
field: target,
value: protoreflect.ValueOf(ipValue),
})
}
}
switch descriptor.Kind() {
case protoreflect.MessageKind:
if descriptor.IsMap() {
2021-09-05 10:02:51 -04:00
err = filterMap(ctx, value.Map())
break
}
if descriptor.IsList() {
2021-09-05 10:02:51 -04:00
err = filterList(ctx, value.List())
break
}
2021-09-05 10:02:51 -04:00
err = filterMessage(ctx, value.Message())
}
return true
})
if err != nil {
return err
}
fsenvironment := envctx.EnvironmentFromContext(ctx)
fsifce := fsenvironment.(filesystemcap.FileSystemCapabilitySet)
for _, v := range fileReadingQueue {
field := message.Descriptor().Fields().ByTextName(v.field)
if v.filename == "" {
continue
}
if len(message.Get(field).Bytes()) > 0 {
continue
}
file, err := fsifce.OpenFileForRead()(v.filename)
if err != nil {
return newError("unable to open file").Base(err)
}
fileContent, err := io.ReadAll(file)
if err != nil {
return newError("unable to read file").Base(err)
}
file.Close()
message.Set(field, protoreflect.ValueOf(fileContent))
}
2021-09-18 16:04:30 -04:00
for _, v := range pendingWriteQueue {
message.Set(v.field, v.value)
}
return nil
}
2021-09-05 10:02:51 -04:00
func filterMap(ctx context.Context, mapValue protoreflect.Map) error {
var err error
mapValue.Range(func(key protoreflect.MapKey, value protoreflect.Value) bool {
2021-09-05 10:02:51 -04:00
err = filterMessage(ctx, value.Message())
return err == nil
})
return err
}
2021-09-05 10:02:51 -04:00
func filterList(ctx context.Context, listValue protoreflect.List) error {
var err error
size := listValue.Len()
for i := 0; i < size; i++ {
2021-09-05 10:02:51 -04:00
err = filterMessage(ctx, listValue.Get(i).Message())
if err != nil {
return err
}
}
return nil
}