366 lines
8.0 KiB
Go
366 lines
8.0 KiB
Go
package global
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"github.com/hiddify/libcore/config"
|
|
"github.com/sagernet/sing-box/experimental/libbox"
|
|
"github.com/sagernet/sing-box/log"
|
|
"github.com/sagernet/sing-box/option"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
var box *libbox.BoxService
|
|
var configOptions *config.ConfigOptions
|
|
var activeConfigPath *string
|
|
var logFactory *log.Factory
|
|
|
|
func setup(baseDir string, workingDir string, tempDir string, statusPort int64, debug bool) error {
|
|
Setup(baseDir, workingDir, tempDir)
|
|
statusPropagationPort = statusPort
|
|
|
|
var defaultWriter io.Writer
|
|
if !debug {
|
|
defaultWriter = io.Discard
|
|
}
|
|
factory, err := log.New(
|
|
log.Options{
|
|
DefaultWriter: defaultWriter,
|
|
BaseTime: time.Now(),
|
|
Observable: false,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
logFactory = &factory
|
|
return nil
|
|
}
|
|
|
|
func parse(path string, tempPath string, debug bool) error {
|
|
config, err := config.ParseConfig(tempPath, debug)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = os.WriteFile(path, config, 0777)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func changeConfigOptions(configOptionsJson string) error {
|
|
configOptions = &config.ConfigOptions{}
|
|
err := json.Unmarshal([]byte(configOptionsJson), configOptions)
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func generateConfig(path string) (string, error) {
|
|
config, err := generateConfigFromFile(path, *configOptions)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return config, nil
|
|
}
|
|
|
|
func generateConfigFromFile(path string, configOpt config.ConfigOptions) (string, error) {
|
|
os.Chdir(filepath.Dir(path))
|
|
content, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
options, err := parseConfig(string(content))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
config, err := config.BuildConfigJson(configOpt, options)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return config, nil
|
|
}
|
|
|
|
func start(configPath string, disableMemoryLimit bool) error {
|
|
if status != Stopped {
|
|
return nil
|
|
}
|
|
propagateStatus(Starting)
|
|
|
|
activeConfigPath = &configPath
|
|
|
|
libbox.SetMemoryLimit(!disableMemoryLimit)
|
|
err := startService(false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func startService(delayStart bool) error {
|
|
content, err := os.ReadFile(*activeConfigPath)
|
|
if err != nil {
|
|
return stopAndAlert(EmptyConfiguration, err)
|
|
}
|
|
options, err := parseConfig(string(content))
|
|
if err != nil {
|
|
return stopAndAlert(EmptyConfiguration, err)
|
|
}
|
|
os.Chdir(filepath.Dir(*activeConfigPath))
|
|
var patchedOptions *option.Options
|
|
patchedOptions, err = config.BuildConfig(*configOptions, options)
|
|
if err != nil {
|
|
return fmt.Errorf("error building config: %w", err)
|
|
}
|
|
|
|
config.SaveCurrentConfig(sWorkingPath, *patchedOptions)
|
|
|
|
err = startCommandServer(*logFactory)
|
|
if err != nil {
|
|
return stopAndAlert(StartCommandServer, err)
|
|
}
|
|
|
|
instance, err := NewService(*patchedOptions)
|
|
if err != nil {
|
|
return stopAndAlert(CreateService, err)
|
|
}
|
|
|
|
if delayStart {
|
|
time.Sleep(250 * time.Millisecond)
|
|
}
|
|
|
|
err = instance.Start()
|
|
if err != nil {
|
|
return stopAndAlert(StartService, err)
|
|
}
|
|
box = instance
|
|
commandServer.SetService(box)
|
|
|
|
propagateStatus(Started)
|
|
return nil
|
|
}
|
|
|
|
func stop() error {
|
|
if status != Started {
|
|
return nil
|
|
}
|
|
if box == nil {
|
|
return errors.New("instance not found")
|
|
}
|
|
propagateStatus(Stopping)
|
|
|
|
commandServer.SetService(nil)
|
|
err := box.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
box = nil
|
|
|
|
err = commandServer.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
commandServer = nil
|
|
propagateStatus(Stopped)
|
|
|
|
return nil
|
|
}
|
|
|
|
func restart(configPath string, disableMemoryLimit bool) error {
|
|
log.Debug("[Service] Restarting")
|
|
|
|
if status != Started {
|
|
return nil
|
|
}
|
|
if box == nil {
|
|
return errors.New("instance not found")
|
|
}
|
|
|
|
err := stop()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
propagateStatus(Starting)
|
|
|
|
time.Sleep(250 * time.Millisecond)
|
|
|
|
activeConfigPath = &configPath
|
|
libbox.SetMemoryLimit(!disableMemoryLimit)
|
|
gErr := startService(false)
|
|
if gErr != nil {
|
|
return gErr
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func startCommandClient(command int, port int64) error {
|
|
err := StartCommand(int32(command), port, *logFactory)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func stopCommandClient(command int) error {
|
|
err := StopCommand(int32(command))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func selectOutbound(groupTag string, outboundTag string) error {
|
|
err := libbox.NewStandaloneCommandClient().SelectOutbound(groupTag, outboundTag)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func urlTest(groupTag string) error {
|
|
err := libbox.NewStandaloneCommandClient().URLTest(groupTag)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func StartServiceC(delayStart bool, content string) error {
|
|
options, err := parseConfig(content)
|
|
if err != nil {
|
|
return stopAndAlert(EmptyConfiguration, err)
|
|
}
|
|
configOptions = &config.ConfigOptions{}
|
|
patchedOptions, err := config.BuildConfig(*configOptions, options)
|
|
|
|
options = *patchedOptions
|
|
|
|
err = config.SaveCurrentConfig(sWorkingPath, options)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = startCommandServer(*logFactory)
|
|
if err != nil {
|
|
return stopAndAlert(StartCommandServer, err)
|
|
}
|
|
|
|
instance, err := NewService(options)
|
|
if err != nil {
|
|
return stopAndAlert(CreateService, err)
|
|
}
|
|
|
|
if delayStart {
|
|
time.Sleep(250 * time.Millisecond)
|
|
}
|
|
|
|
err = instance.Start()
|
|
if err != nil {
|
|
return stopAndAlert(StartService, err)
|
|
}
|
|
box = instance
|
|
commandServer.SetService(box)
|
|
|
|
propagateStatus(Started)
|
|
return nil
|
|
}
|
|
|
|
func StopService() error {
|
|
if status != Started {
|
|
return nil
|
|
}
|
|
if box == nil {
|
|
return errors.New("instance not found")
|
|
}
|
|
|
|
propagateStatus(Stopping)
|
|
commandServer.SetService(nil)
|
|
err := box.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
box = nil
|
|
|
|
err = commandServer.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
commandServer = nil
|
|
propagateStatus(Stopped)
|
|
|
|
return nil
|
|
}
|
|
|
|
func SetupC(baseDir string, workDir string, tempDir string, debug bool) error {
|
|
err := os.MkdirAll("./bin", 600)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = os.MkdirAll("./work", 600)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = os.MkdirAll("./temp", 600)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
Setup(baseDir, workDir, tempDir)
|
|
var defaultWriter io.Writer
|
|
if !debug {
|
|
defaultWriter = io.Discard
|
|
}
|
|
factory, err := log.New(
|
|
log.Options{
|
|
DefaultWriter: defaultWriter,
|
|
BaseTime: time.Now(),
|
|
Observable: false,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
logFactory = &factory
|
|
return nil
|
|
}
|
|
|
|
func MakeConfig(Ipv6 bool, ServerPort int, StrictRoute bool, EndpointIndependentNat bool, Stack string) string {
|
|
var ipv6 string
|
|
if Ipv6 {
|
|
ipv6 = " \"inet6_address\": \"fdfe:dcba:9876::1/126\",\n"
|
|
} else {
|
|
ipv6 = ""
|
|
}
|
|
base := "{\n \"inbounds\": [\n {\n \"type\": \"tun\",\n \"tag\": \"tun-in\",\n \"interface_name\": \"tun0\",\n \"inet4_address\": \"172.19.0.1/30\",\n" + ipv6 + " \"mtu\": 9000,\n \"auto_route\": true,\n \"strict_route\": " + fmt.Sprintf("%t", StrictRoute) + ",\n \"endpoint_independent_nat\": " + fmt.Sprintf("%t", EndpointIndependentNat) + ",\n \"stack\": \"" + Stack + "\"\n }],\n \"outbounds\": [\n {\n \"type\": \"socks\",\n \"tag\": \"socks-out\",\n \"server\": \"127.0.0.1\",\n \"server_port\": " + fmt.Sprintf("%d", ServerPort) + ",\n \"version\": \"5\"\n }\n ]\n}\n"
|
|
return base
|
|
}
|
|
|
|
func WriteParameters(Ipv6 bool, ServerPort int, StrictRoute bool, EndpointIndependentNat bool, Stack string) error {
|
|
parameters := fmt.Sprintf("%t,%d,%t,%t,%s", Ipv6, ServerPort, StrictRoute, EndpointIndependentNat, Stack)
|
|
err := os.WriteFile("bin/parameters.config", []byte(parameters), 600)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
func ReadParameters() (bool, int, bool, bool, string, error) {
|
|
Data, err := os.ReadFile("bin/parameters.config")
|
|
if err != nil {
|
|
return false, 0, false, false, "", err
|
|
}
|
|
DataSlice := strings.Split(string(Data), ",")
|
|
Ipv6, _ := strconv.ParseBool(DataSlice[0])
|
|
ServerPort, _ := strconv.Atoi(DataSlice[1])
|
|
StrictRoute, _ := strconv.ParseBool(DataSlice[2])
|
|
EndpointIndependentNat, _ := strconv.ParseBool(DataSlice[3])
|
|
stack := DataSlice[4]
|
|
return Ipv6, ServerPort, StrictRoute, EndpointIndependentNat, stack, nil
|
|
}
|