mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 12:27:37 +03:00
json: Add badoption templates
This commit is contained in:
parent
0acb36c118
commit
c324d4143d
7 changed files with 478 additions and 0 deletions
60
common/json/badjson/merge_objects.go
Normal file
60
common/json/badjson/merge_objects.go
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
package badjson
|
||||||
|
|
||||||
|
import (
|
||||||
|
E "github.com/sagernet/sing/common/exceptions"
|
||||||
|
"github.com/sagernet/sing/common/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
func MarshallObjects(objects ...any) ([]byte, error) {
|
||||||
|
if len(objects) == 1 {
|
||||||
|
return json.Marshal(objects[0])
|
||||||
|
}
|
||||||
|
var content JSONObject
|
||||||
|
for _, object := range objects {
|
||||||
|
objectMap, err := newJSONObject(object)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
content.PutAll(objectMap)
|
||||||
|
}
|
||||||
|
return content.MarshalJSON()
|
||||||
|
}
|
||||||
|
|
||||||
|
func UnmarshallExcluded(inputContent []byte, parentObject any, object any) error {
|
||||||
|
parentContent, err := newJSONObject(parentObject)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var content JSONObject
|
||||||
|
err = content.UnmarshalJSON(inputContent)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, key := range parentContent.Keys() {
|
||||||
|
content.Remove(key)
|
||||||
|
}
|
||||||
|
if object == nil {
|
||||||
|
if content.IsEmpty() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return E.New("unexpected key: ", content.Keys()[0])
|
||||||
|
}
|
||||||
|
inputContent, err = content.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return json.UnmarshalDisallowUnknownFields(inputContent, object)
|
||||||
|
}
|
||||||
|
|
||||||
|
func newJSONObject(object any) (*JSONObject, error) {
|
||||||
|
inputContent, err := json.Marshal(object)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var content JSONObject
|
||||||
|
err = content.UnmarshalJSON(inputContent)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &content, nil
|
||||||
|
}
|
32
common/json/badoption/duration.go
Normal file
32
common/json/badoption/duration.go
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
package badoption
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common/json"
|
||||||
|
"github.com/sagernet/sing/common/json/badoption/internal/my_time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Duration time.Duration
|
||||||
|
|
||||||
|
func (d Duration) Build() time.Duration {
|
||||||
|
return time.Duration(d)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d Duration) MarshalJSON() ([]byte, error) {
|
||||||
|
return json.Marshal((time.Duration)(d).String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Duration) UnmarshalJSON(bytes []byte) error {
|
||||||
|
var value string
|
||||||
|
err := json.Unmarshal(bytes, &value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
duration, err := my_time.ParseDuration(value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*d = Duration(duration)
|
||||||
|
return nil
|
||||||
|
}
|
15
common/json/badoption/http.go
Normal file
15
common/json/badoption/http.go
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
package badoption
|
||||||
|
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
|
type HTTPHeader map[string]Listable[string]
|
||||||
|
|
||||||
|
func (h HTTPHeader) Build() http.Header {
|
||||||
|
header := make(http.Header)
|
||||||
|
for name, values := range h {
|
||||||
|
for _, value := range values {
|
||||||
|
header.Add(name, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return header
|
||||||
|
}
|
226
common/json/badoption/internal/my_time/format.go
Normal file
226
common/json/badoption/internal/my_time/format.go
Normal file
|
@ -0,0 +1,226 @@
|
||||||
|
package my_time
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Copyright 2010 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
const durationDay = 24 * time.Hour
|
||||||
|
|
||||||
|
var unitMap = map[string]uint64{
|
||||||
|
"ns": uint64(time.Nanosecond),
|
||||||
|
"us": uint64(time.Microsecond),
|
||||||
|
"µs": uint64(time.Microsecond), // U+00B5 = micro symbol
|
||||||
|
"μs": uint64(time.Microsecond), // U+03BC = Greek letter mu
|
||||||
|
"ms": uint64(time.Millisecond),
|
||||||
|
"s": uint64(time.Second),
|
||||||
|
"m": uint64(time.Minute),
|
||||||
|
"h": uint64(time.Hour),
|
||||||
|
"d": uint64(durationDay),
|
||||||
|
}
|
||||||
|
|
||||||
|
// ParseDuration parses a duration string.
|
||||||
|
// A duration string is a possibly signed sequence of
|
||||||
|
// decimal numbers, each with optional fraction and a unit suffix,
|
||||||
|
// such as "300ms", "-1.5h" or "2h45m".
|
||||||
|
// Valid time units are "ns", "us" (or "µs"), "ms", "s", "m", "h".
|
||||||
|
func ParseDuration(s string) (time.Duration, error) {
|
||||||
|
// [-+]?([0-9]*(\.[0-9]*)?[a-z]+)+
|
||||||
|
orig := s
|
||||||
|
var d uint64
|
||||||
|
neg := false
|
||||||
|
|
||||||
|
// Consume [-+]?
|
||||||
|
if s != "" {
|
||||||
|
c := s[0]
|
||||||
|
if c == '-' || c == '+' {
|
||||||
|
neg = c == '-'
|
||||||
|
s = s[1:]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Special case: if all that is left is "0", this is zero.
|
||||||
|
if s == "0" {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
if s == "" {
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
for s != "" {
|
||||||
|
var (
|
||||||
|
v, f uint64 // integers before, after decimal point
|
||||||
|
scale float64 = 1 // value = v + f/scale
|
||||||
|
)
|
||||||
|
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// The next character must be [0-9.]
|
||||||
|
if !(s[0] == '.' || '0' <= s[0] && s[0] <= '9') {
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
// Consume [0-9]*
|
||||||
|
pl := len(s)
|
||||||
|
v, s, err = leadingInt(s)
|
||||||
|
if err != nil {
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
pre := pl != len(s) // whether we consumed anything before a period
|
||||||
|
|
||||||
|
// Consume (\.[0-9]*)?
|
||||||
|
post := false
|
||||||
|
if s != "" && s[0] == '.' {
|
||||||
|
s = s[1:]
|
||||||
|
pl := len(s)
|
||||||
|
f, scale, s = leadingFraction(s)
|
||||||
|
post = pl != len(s)
|
||||||
|
}
|
||||||
|
if !pre && !post {
|
||||||
|
// no digits (e.g. ".s" or "-.s")
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Consume unit.
|
||||||
|
i := 0
|
||||||
|
for ; i < len(s); i++ {
|
||||||
|
c := s[i]
|
||||||
|
if c == '.' || '0' <= c && c <= '9' {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if i == 0 {
|
||||||
|
return 0, errors.New("time: missing unit in duration " + quote(orig))
|
||||||
|
}
|
||||||
|
u := s[:i]
|
||||||
|
s = s[i:]
|
||||||
|
unit, ok := unitMap[u]
|
||||||
|
if !ok {
|
||||||
|
return 0, errors.New("time: unknown unit " + quote(u) + " in duration " + quote(orig))
|
||||||
|
}
|
||||||
|
if v > 1<<63/unit {
|
||||||
|
// overflow
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
v *= unit
|
||||||
|
if f > 0 {
|
||||||
|
// float64 is needed to be nanosecond accurate for fractions of hours.
|
||||||
|
// v >= 0 && (f*unit/scale) <= 3.6e+12 (ns/h, h is the largest unit)
|
||||||
|
v += uint64(float64(f) * (float64(unit) / scale))
|
||||||
|
if v > 1<<63 {
|
||||||
|
// overflow
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
d += v
|
||||||
|
if d > 1<<63 {
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if neg {
|
||||||
|
return -time.Duration(d), nil
|
||||||
|
}
|
||||||
|
if d > 1<<63-1 {
|
||||||
|
return 0, errors.New("time: invalid duration " + quote(orig))
|
||||||
|
}
|
||||||
|
return time.Duration(d), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var errLeadingInt = errors.New("time: bad [0-9]*") // never printed
|
||||||
|
|
||||||
|
// leadingInt consumes the leading [0-9]* from s.
|
||||||
|
func leadingInt[bytes []byte | string](s bytes) (x uint64, rem bytes, err error) {
|
||||||
|
i := 0
|
||||||
|
for ; i < len(s); i++ {
|
||||||
|
c := s[i]
|
||||||
|
if c < '0' || c > '9' {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if x > 1<<63/10 {
|
||||||
|
// overflow
|
||||||
|
return 0, rem, errLeadingInt
|
||||||
|
}
|
||||||
|
x = x*10 + uint64(c) - '0'
|
||||||
|
if x > 1<<63 {
|
||||||
|
// overflow
|
||||||
|
return 0, rem, errLeadingInt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return x, s[i:], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// leadingFraction consumes the leading [0-9]* from s.
|
||||||
|
// It is used only for fractions, so does not return an error on overflow,
|
||||||
|
// it just stops accumulating precision.
|
||||||
|
func leadingFraction(s string) (x uint64, scale float64, rem string) {
|
||||||
|
i := 0
|
||||||
|
scale = 1
|
||||||
|
overflow := false
|
||||||
|
for ; i < len(s); i++ {
|
||||||
|
c := s[i]
|
||||||
|
if c < '0' || c > '9' {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if overflow {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if x > (1<<63-1)/10 {
|
||||||
|
// It's possible for overflow to give a positive number, so take care.
|
||||||
|
overflow = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
y := x*10 + uint64(c) - '0'
|
||||||
|
if y > 1<<63 {
|
||||||
|
overflow = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
x = y
|
||||||
|
scale *= 10
|
||||||
|
}
|
||||||
|
return x, scale, s[i:]
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are borrowed from unicode/utf8 and strconv and replicate behavior in
|
||||||
|
// that package, since we can't take a dependency on either.
|
||||||
|
const (
|
||||||
|
lowerhex = "0123456789abcdef"
|
||||||
|
runeSelf = 0x80
|
||||||
|
runeError = '\uFFFD'
|
||||||
|
)
|
||||||
|
|
||||||
|
func quote(s string) string {
|
||||||
|
buf := make([]byte, 1, len(s)+2) // slice will be at least len(s) + quotes
|
||||||
|
buf[0] = '"'
|
||||||
|
for i, c := range s {
|
||||||
|
if c >= runeSelf || c < ' ' {
|
||||||
|
// This means you are asking us to parse a time.Duration or
|
||||||
|
// time.Location with unprintable or non-ASCII characters in it.
|
||||||
|
// We don't expect to hit this case very often. We could try to
|
||||||
|
// reproduce strconv.Quote's behavior with full fidelity but
|
||||||
|
// given how rarely we expect to hit these edge cases, speed and
|
||||||
|
// conciseness are better.
|
||||||
|
var width int
|
||||||
|
if c == runeError {
|
||||||
|
width = 1
|
||||||
|
if i+2 < len(s) && s[i:i+3] == string(runeError) {
|
||||||
|
width = 3
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
width = len(string(c))
|
||||||
|
}
|
||||||
|
for j := 0; j < width; j++ {
|
||||||
|
buf = append(buf, `\x`...)
|
||||||
|
buf = append(buf, lowerhex[s[i+j]>>4])
|
||||||
|
buf = append(buf, lowerhex[s[i+j]&0xF])
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if c == '"' || c == '\\' {
|
||||||
|
buf = append(buf, '\\')
|
||||||
|
}
|
||||||
|
buf = append(buf, string(c)...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
buf = append(buf, '"')
|
||||||
|
return string(buf)
|
||||||
|
}
|
30
common/json/badoption/listable.go
Normal file
30
common/json/badoption/listable.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package badoption
|
||||||
|
|
||||||
|
import (
|
||||||
|
E "github.com/sagernet/sing/common/exceptions"
|
||||||
|
"github.com/sagernet/sing/common/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Listable[T any] []T
|
||||||
|
|
||||||
|
func (l Listable[T]) MarshalJSON() ([]byte, error) {
|
||||||
|
arrayList := []T(l)
|
||||||
|
if len(arrayList) == 1 {
|
||||||
|
return json.Marshal(arrayList[0])
|
||||||
|
}
|
||||||
|
return json.Marshal(arrayList)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Listable[T]) UnmarshalJSON(content []byte) error {
|
||||||
|
err := json.UnmarshalDisallowUnknownFields(content, (*[]T)(l))
|
||||||
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
var singleItem T
|
||||||
|
newError := json.UnmarshalDisallowUnknownFields(content, &singleItem)
|
||||||
|
if newError != nil {
|
||||||
|
return E.Errors(err, newError)
|
||||||
|
}
|
||||||
|
*l = []T{singleItem}
|
||||||
|
return nil
|
||||||
|
}
|
84
common/json/badoption/netip.go
Normal file
84
common/json/badoption/netip.go
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
package badoption
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/netip"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Addr netip.Addr
|
||||||
|
|
||||||
|
func (a *Addr) Build(defaultAddr netip.Addr) netip.Addr {
|
||||||
|
if a == nil {
|
||||||
|
return defaultAddr
|
||||||
|
}
|
||||||
|
return netip.Addr(*a)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Addr) MarshalJSON() ([]byte, error) {
|
||||||
|
return json.Marshal(netip.Addr(*a).String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Addr) UnmarshalJSON(content []byte) error {
|
||||||
|
var value string
|
||||||
|
err := json.Unmarshal(content, &value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
addr, err := netip.ParseAddr(value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*a = Addr(addr)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type Prefix netip.Prefix
|
||||||
|
|
||||||
|
func (p *Prefix) MarshalJSON() ([]byte, error) {
|
||||||
|
return json.Marshal(netip.Prefix(*p).String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Prefix) UnmarshalJSON(content []byte) error {
|
||||||
|
var value string
|
||||||
|
err := json.Unmarshal(content, &value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
prefix, err := netip.ParsePrefix(value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*p = Prefix(prefix)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type Prefixable netip.Prefix
|
||||||
|
|
||||||
|
func (p *Prefixable) MarshalJSON() ([]byte, error) {
|
||||||
|
prefix := netip.Prefix(*p)
|
||||||
|
if prefix.Bits() == prefix.Addr().BitLen() {
|
||||||
|
return json.Marshal(prefix.Addr().String())
|
||||||
|
} else {
|
||||||
|
return json.Marshal(prefix.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Prefixable) UnmarshalJSON(content []byte) error {
|
||||||
|
var value string
|
||||||
|
err := json.Unmarshal(content, &value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
prefix, prefixErr := netip.ParsePrefix(value)
|
||||||
|
if prefixErr == nil {
|
||||||
|
*p = Prefixable(prefix)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
addr, addrErr := netip.ParseAddr(value)
|
||||||
|
if addrErr == nil {
|
||||||
|
*p = Prefixable(netip.PrefixFrom(addr, addr.BitLen()))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return prefixErr
|
||||||
|
}
|
31
common/json/badoption/regexp.go
Normal file
31
common/json/badoption/regexp.go
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
package badoption
|
||||||
|
|
||||||
|
import (
|
||||||
|
"regexp"
|
||||||
|
|
||||||
|
"github.com/sagernet/sing/common/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Regexp regexp.Regexp
|
||||||
|
|
||||||
|
func (r *Regexp) Build() *regexp.Regexp {
|
||||||
|
return (*regexp.Regexp)(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regexp) MarshalJSON() ([]byte, error) {
|
||||||
|
return json.Marshal((*regexp.Regexp)(r).String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regexp) UnmarshalJSON(content []byte) error {
|
||||||
|
var stringValue string
|
||||||
|
err := json.Unmarshal(content, &stringValue)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
regex, err := regexp.Compile(stringValue)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*r = Regexp(*regex)
|
||||||
|
return nil
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue