|  | @@ -0,0 +1,167 @@
 | 
											
												
													
														|  | 
 |  | +package main
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +import (
 | 
											
												
													
														|  | 
 |  | +	"bytes"
 | 
											
												
													
														|  | 
 |  | +	"os"
 | 
											
												
													
														|  | 
 |  | +	"path/filepath"
 | 
											
												
													
														|  | 
 |  | +	"strings"
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	"github.com/sagernet/sing-box/common/json"
 | 
											
												
													
														|  | 
 |  | +	C "github.com/sagernet/sing-box/constant"
 | 
											
												
													
														|  | 
 |  | +	"github.com/sagernet/sing-box/log"
 | 
											
												
													
														|  | 
 |  | +	"github.com/sagernet/sing-box/option"
 | 
											
												
													
														|  | 
 |  | +	E "github.com/sagernet/sing/common/exceptions"
 | 
											
												
													
														|  | 
 |  | +	"github.com/sagernet/sing/common/rw"
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	"github.com/spf13/cobra"
 | 
											
												
													
														|  | 
 |  | +)
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +var commandMerge = &cobra.Command{
 | 
											
												
													
														|  | 
 |  | +	Use:   "merge [output]",
 | 
											
												
													
														|  | 
 |  | +	Short: "Merge configurations",
 | 
											
												
													
														|  | 
 |  | +	Run: func(cmd *cobra.Command, args []string) {
 | 
											
												
													
														|  | 
 |  | +		err := merge(args[0])
 | 
											
												
													
														|  | 
 |  | +		if err != nil {
 | 
											
												
													
														|  | 
 |  | +			log.Fatal(err)
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	},
 | 
											
												
													
														|  | 
 |  | +	Args: cobra.ExactArgs(1),
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func init() {
 | 
											
												
													
														|  | 
 |  | +	mainCommand.AddCommand(commandMerge)
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func merge(outputPath string) error {
 | 
											
												
													
														|  | 
 |  | +	mergedOptions, err := readConfigAndMerge()
 | 
											
												
													
														|  | 
 |  | +	if err != nil {
 | 
											
												
													
														|  | 
 |  | +		return err
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	err = mergePathResources(&mergedOptions)
 | 
											
												
													
														|  | 
 |  | +	if err != nil {
 | 
											
												
													
														|  | 
 |  | +		return err
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	buffer := new(bytes.Buffer)
 | 
											
												
													
														|  | 
 |  | +	encoder := json.NewEncoder(buffer)
 | 
											
												
													
														|  | 
 |  | +	encoder.SetIndent("", "  ")
 | 
											
												
													
														|  | 
 |  | +	err = encoder.Encode(mergedOptions)
 | 
											
												
													
														|  | 
 |  | +	if err != nil {
 | 
											
												
													
														|  | 
 |  | +		return E.Cause(err, "encode config")
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if existsContent, err := os.ReadFile(outputPath); err != nil {
 | 
											
												
													
														|  | 
 |  | +		if string(existsContent) == buffer.String() {
 | 
											
												
													
														|  | 
 |  | +			return nil
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	err = rw.WriteFile(outputPath, buffer.Bytes())
 | 
											
												
													
														|  | 
 |  | +	if err != nil {
 | 
											
												
													
														|  | 
 |  | +		return err
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	outputPath, _ = filepath.Abs(outputPath)
 | 
											
												
													
														|  | 
 |  | +	os.Stderr.WriteString(outputPath + "\n")
 | 
											
												
													
														|  | 
 |  | +	return nil
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func mergePathResources(options *option.Options) error {
 | 
											
												
													
														|  | 
 |  | +	for index, inbound := range options.Inbounds {
 | 
											
												
													
														|  | 
 |  | +		switch inbound.Type {
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHTTP:
 | 
											
												
													
														|  | 
 |  | +			inbound.HTTPOptions.TLS = mergeTLSInboundOptions(inbound.HTTPOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeMixed:
 | 
											
												
													
														|  | 
 |  | +			inbound.MixedOptions.TLS = mergeTLSInboundOptions(inbound.MixedOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeVMess:
 | 
											
												
													
														|  | 
 |  | +			inbound.VMessOptions.TLS = mergeTLSInboundOptions(inbound.VMessOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeTrojan:
 | 
											
												
													
														|  | 
 |  | +			inbound.TrojanOptions.TLS = mergeTLSInboundOptions(inbound.TrojanOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeNaive:
 | 
											
												
													
														|  | 
 |  | +			inbound.NaiveOptions.TLS = mergeTLSInboundOptions(inbound.NaiveOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHysteria:
 | 
											
												
													
														|  | 
 |  | +			inbound.HysteriaOptions.TLS = mergeTLSInboundOptions(inbound.HysteriaOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeVLESS:
 | 
											
												
													
														|  | 
 |  | +			inbound.VLESSOptions.TLS = mergeTLSInboundOptions(inbound.VLESSOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeTUIC:
 | 
											
												
													
														|  | 
 |  | +			inbound.TUICOptions.TLS = mergeTLSInboundOptions(inbound.TUICOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHysteria2:
 | 
											
												
													
														|  | 
 |  | +			inbound.Hysteria2Options.TLS = mergeTLSInboundOptions(inbound.Hysteria2Options.TLS)
 | 
											
												
													
														|  | 
 |  | +		default:
 | 
											
												
													
														|  | 
 |  | +			continue
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +		options.Inbounds[index] = inbound
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	for index, outbound := range options.Outbounds {
 | 
											
												
													
														|  | 
 |  | +		switch outbound.Type {
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHTTP:
 | 
											
												
													
														|  | 
 |  | +			outbound.HTTPOptions.TLS = mergeTLSOutboundOptions(outbound.HTTPOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeVMess:
 | 
											
												
													
														|  | 
 |  | +			outbound.VMessOptions.TLS = mergeTLSOutboundOptions(outbound.VMessOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeTrojan:
 | 
											
												
													
														|  | 
 |  | +			outbound.TrojanOptions.TLS = mergeTLSOutboundOptions(outbound.TrojanOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHysteria:
 | 
											
												
													
														|  | 
 |  | +			outbound.HysteriaOptions.TLS = mergeTLSOutboundOptions(outbound.HysteriaOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeSSH:
 | 
											
												
													
														|  | 
 |  | +			outbound.SSHOptions = mergeSSHOutboundOptions(outbound.SSHOptions)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeVLESS:
 | 
											
												
													
														|  | 
 |  | +			outbound.VLESSOptions.TLS = mergeTLSOutboundOptions(outbound.VLESSOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeTUIC:
 | 
											
												
													
														|  | 
 |  | +			outbound.TUICOptions.TLS = mergeTLSOutboundOptions(outbound.TUICOptions.TLS)
 | 
											
												
													
														|  | 
 |  | +		case C.TypeHysteria2:
 | 
											
												
													
														|  | 
 |  | +			outbound.Hysteria2Options.TLS = mergeTLSOutboundOptions(outbound.Hysteria2Options.TLS)
 | 
											
												
													
														|  | 
 |  | +		default:
 | 
											
												
													
														|  | 
 |  | +			continue
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +		options.Outbounds[index] = outbound
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	return nil
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func mergeTLSInboundOptions(options *option.InboundTLSOptions) *option.InboundTLSOptions {
 | 
											
												
													
														|  | 
 |  | +	if options == nil {
 | 
											
												
													
														|  | 
 |  | +		return nil
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if options.CertificatePath != "" {
 | 
											
												
													
														|  | 
 |  | +		if content, err := os.ReadFile(options.CertificatePath); err == nil {
 | 
											
												
													
														|  | 
 |  | +			options.Certificate = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if options.KeyPath != "" {
 | 
											
												
													
														|  | 
 |  | +		if content, err := os.ReadFile(options.KeyPath); err == nil {
 | 
											
												
													
														|  | 
 |  | +			options.Key = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if options.ECH != nil {
 | 
											
												
													
														|  | 
 |  | +		if options.ECH.KeyPath != "" {
 | 
											
												
													
														|  | 
 |  | +			if content, err := os.ReadFile(options.ECH.KeyPath); err == nil {
 | 
											
												
													
														|  | 
 |  | +				options.ECH.Key = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +			}
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	return options
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func mergeTLSOutboundOptions(options *option.OutboundTLSOptions) *option.OutboundTLSOptions {
 | 
											
												
													
														|  | 
 |  | +	if options == nil {
 | 
											
												
													
														|  | 
 |  | +		return nil
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if options.CertificatePath != "" {
 | 
											
												
													
														|  | 
 |  | +		if content, err := os.ReadFile(options.CertificatePath); err == nil {
 | 
											
												
													
														|  | 
 |  | +			options.Certificate = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	if options.ECH != nil {
 | 
											
												
													
														|  | 
 |  | +		if options.ECH.ConfigPath != "" {
 | 
											
												
													
														|  | 
 |  | +			if content, err := os.ReadFile(options.ECH.ConfigPath); err == nil {
 | 
											
												
													
														|  | 
 |  | +				options.ECH.Config = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +			}
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	return options
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +func mergeSSHOutboundOptions(options option.SSHOutboundOptions) option.SSHOutboundOptions {
 | 
											
												
													
														|  | 
 |  | +	if options.PrivateKeyPath != "" {
 | 
											
												
													
														|  | 
 |  | +		if content, err := os.ReadFile(options.PrivateKeyPath); err == nil {
 | 
											
												
													
														|  | 
 |  | +			options.PrivateKey = strings.Split(string(content), "\n")
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +	return options
 | 
											
												
													
														|  | 
 |  | +}
 |