| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | package xray | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/fs" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"os/exec" | 
					
						
							|  |  |  | 	"runtime" | 
					
						
							| 
									
										
										
										
											2025-09-14 18:04:12 +00:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2023-06-15 21:40:49 +00:00
										 |  |  | 	"syscall" | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2023-08-08 18:51:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-19 08:05:43 +00:00
										 |  |  | 	"github.com/mhsanaei/3x-ui/v2/config" | 
					
						
							|  |  |  | 	"github.com/mhsanaei/3x-ui/v2/logger" | 
					
						
							|  |  |  | 	"github.com/mhsanaei/3x-ui/v2/util/common" | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetBinaryName returns the Xray binary filename for the current OS and architecture.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func GetBinaryName() string { | 
					
						
							|  |  |  | 	return fmt.Sprintf("xray-%s-%s", runtime.GOOS, runtime.GOARCH) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetBinaryPath returns the full path to the Xray binary executable.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func GetBinaryPath() string { | 
					
						
							| 
									
										
										
										
											2023-04-13 19:40:01 +00:00
										 |  |  | 	return config.GetBinFolderPath() + "/" + GetBinaryName() | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetConfigPath returns the path to the Xray configuration file in the binary folder.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func GetConfigPath() string { | 
					
						
							| 
									
										
										
										
											2023-04-13 19:40:01 +00:00
										 |  |  | 	return config.GetBinFolderPath() + "/config.json" | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetGeositePath returns the path to the geosite data file used by Xray.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func GetGeositePath() string { | 
					
						
							| 
									
										
										
										
											2023-04-13 19:40:01 +00:00
										 |  |  | 	return config.GetBinFolderPath() + "/geosite.dat" | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetGeoipPath returns the path to the geoip data file used by Xray.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func GetGeoipPath() string { | 
					
						
							| 
									
										
										
										
											2023-04-13 19:40:01 +00:00
										 |  |  | 	return config.GetBinFolderPath() + "/geoip.dat" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetIPLimitLogPath returns the path to the IP limit log file.
 | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | func GetIPLimitLogPath() string { | 
					
						
							|  |  |  | 	return config.GetLogFolder() + "/3xipl.log" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetIPLimitBannedLogPath returns the path to the banned IP log file.
 | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | func GetIPLimitBannedLogPath() string { | 
					
						
							|  |  |  | 	return config.GetLogFolder() + "/3xipl-banned.log" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetIPLimitBannedPrevLogPath returns the path to the previous banned IP log file.
 | 
					
						
							| 
									
										
										
										
											2024-01-01 15:07:56 +00:00
										 |  |  | func GetIPLimitBannedPrevLogPath() string { | 
					
						
							|  |  |  | 	return config.GetLogFolder() + "/3xipl-banned.prev.log" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetAccessPersistentLogPath returns the path to the persistent access log file.
 | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | func GetAccessPersistentLogPath() string { | 
					
						
							| 
									
										
										
										
											2024-01-01 15:07:56 +00:00
										 |  |  | 	return config.GetLogFolder() + "/3xipl-ap.log" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetAccessPersistentPrevLogPath returns the path to the previous persistent access log file.
 | 
					
						
							| 
									
										
										
										
											2024-01-01 15:07:56 +00:00
										 |  |  | func GetAccessPersistentPrevLogPath() string { | 
					
						
							|  |  |  | 	return config.GetLogFolder() + "/3xipl-ap.prev.log" | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetAccessLogPath reads the Xray config and returns the access log file path.
 | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | func GetAccessLogPath() (string, error) { | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 	config, err := os.ReadFile(GetConfigPath()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-08 21:08:00 +00:00
										 |  |  | 		logger.Warningf("Failed to read configuration file: %s", err) | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 19:13:51 +00:00
										 |  |  | 	jsonConfig := map[string]any{} | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 	err = json.Unmarshal([]byte(config), &jsonConfig) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-08 21:08:00 +00:00
										 |  |  | 		logger.Warningf("Failed to parse JSON configuration: %s", err) | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | 		return "", err | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if jsonConfig["log"] != nil { | 
					
						
							| 
									
										
										
										
											2025-03-12 19:13:51 +00:00
										 |  |  | 		jsonLog := jsonConfig["log"].(map[string]any) | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 		if jsonLog["access"] != nil { | 
					
						
							|  |  |  | 			accessLogPath := jsonLog["access"].(string) | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | 			return accessLogPath, nil | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | 	return "", err | 
					
						
							| 
									
										
										
										
											2023-07-01 12:26:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // stopProcess calls Stop on the given Process instance.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func stopProcess(p *Process) { | 
					
						
							|  |  |  | 	p.Stop() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // Process wraps an Xray process instance and provides management methods.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | type Process struct { | 
					
						
							|  |  |  | 	*process | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // NewProcess creates a new Xray process and sets up cleanup on garbage collection.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func NewProcess(xrayConfig *Config) *Process { | 
					
						
							|  |  |  | 	p := &Process{newProcess(xrayConfig)} | 
					
						
							|  |  |  | 	runtime.SetFinalizer(p, stopProcess) | 
					
						
							|  |  |  | 	return p | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type process struct { | 
					
						
							|  |  |  | 	cmd *exec.Cmd | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	version string | 
					
						
							|  |  |  | 	apiPort int | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-04 18:13:21 +00:00
										 |  |  | 	onlineClients []string | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | 	config    *Config | 
					
						
							| 
									
										
										
										
											2023-12-10 12:07:50 +00:00
										 |  |  | 	logWriter *LogWriter | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | 	exitErr   error | 
					
						
							|  |  |  | 	startTime time.Time | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // newProcess creates a new internal process struct for Xray.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func newProcess(config *Config) *process { | 
					
						
							|  |  |  | 	return &process{ | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | 		version:   "Unknown", | 
					
						
							|  |  |  | 		config:    config, | 
					
						
							| 
									
										
										
										
											2023-12-10 12:07:50 +00:00
										 |  |  | 		logWriter: NewLogWriter(), | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | 		startTime: time.Now(), | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // IsRunning returns true if the Xray process is currently running.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) IsRunning() bool { | 
					
						
							|  |  |  | 	if p.cmd == nil || p.cmd.Process == nil { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if p.cmd.ProcessState == nil { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetErr returns the last error encountered by the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) GetErr() error { | 
					
						
							|  |  |  | 	return p.exitErr | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetResult returns the last log line or error from the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) GetResult() string { | 
					
						
							| 
									
										
										
										
											2023-12-10 12:07:50 +00:00
										 |  |  | 	if len(p.logWriter.lastLine) == 0 && p.exitErr != nil { | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 		return p.exitErr.Error() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-12-10 12:07:50 +00:00
										 |  |  | 	return p.logWriter.lastLine | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetVersion returns the version string of the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) GetVersion() string { | 
					
						
							|  |  |  | 	return p.version | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetAPIPort returns the API port used by the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *Process) GetAPIPort() int { | 
					
						
							|  |  |  | 	return p.apiPort | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetConfig returns the configuration used by the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *Process) GetConfig() *Config { | 
					
						
							|  |  |  | 	return p.config | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetOnlineClients returns the list of online clients for the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-12-04 18:13:21 +00:00
										 |  |  | func (p *Process) GetOnlineClients() []string { | 
					
						
							|  |  |  | 	return p.onlineClients | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // SetOnlineClients sets the list of online clients for the Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-12-04 18:13:21 +00:00
										 |  |  | func (p *Process) SetOnlineClients(users []string) { | 
					
						
							|  |  |  | 	p.onlineClients = users | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // GetUptime returns the uptime of the Xray process in seconds.
 | 
					
						
							| 
									
										
										
										
											2023-08-08 21:07:05 +00:00
										 |  |  | func (p *Process) GetUptime() uint64 { | 
					
						
							|  |  |  | 	return uint64(time.Since(p.startTime).Seconds()) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // refreshAPIPort updates the API port from the inbound configs.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) refreshAPIPort() { | 
					
						
							|  |  |  | 	for _, inbound := range p.config.InboundConfigs { | 
					
						
							|  |  |  | 		if inbound.Tag == "api" { | 
					
						
							|  |  |  | 			p.apiPort = inbound.Port | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // refreshVersion updates the version string by running the Xray binary with -version.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) refreshVersion() { | 
					
						
							|  |  |  | 	cmd := exec.Command(GetBinaryPath(), "-version") | 
					
						
							|  |  |  | 	data, err := cmd.Output() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		p.version = "Unknown" | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		datas := bytes.Split(data, []byte(" ")) | 
					
						
							|  |  |  | 		if len(datas) <= 1 { | 
					
						
							|  |  |  | 			p.version = "Unknown" | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			p.version = string(datas[1]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // Start launches the Xray process with the current configuration.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) Start() (err error) { | 
					
						
							|  |  |  | 	if p.IsRunning() { | 
					
						
							|  |  |  | 		return errors.New("xray is already running") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-04-01 07:50:34 +00:00
										 |  |  | 			logger.Error("Failure in running xray-core process: ", err) | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 			p.exitErr = err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	data, err := json.MarshalIndent(p.config, "", "  ") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-04-01 07:50:34 +00:00
										 |  |  | 		return common.NewErrorf("Failed to generate XRAY configuration files: %v", err) | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-03-10 21:31:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-13 07:54:41 +00:00
										 |  |  | 	err = os.MkdirAll(config.GetLogFolder(), 0o770) | 
					
						
							| 
									
										
										
										
											2024-03-05 13:39:20 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-08 21:08:00 +00:00
										 |  |  | 		logger.Warningf("Failed to create log folder: %s", err) | 
					
						
							| 
									
										
										
										
											2024-03-05 13:39:20 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 	configPath := GetConfigPath() | 
					
						
							|  |  |  | 	err = os.WriteFile(configPath, data, fs.ModePerm) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-11 20:55:21 +00:00
										 |  |  | 		return common.NewErrorf("Failed to write configuration file: %v", err) | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-27 07:36:27 +00:00
										 |  |  | 	cmd := exec.Command(GetBinaryPath(), "-c", configPath) | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 	p.cmd = cmd | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-10 12:07:50 +00:00
										 |  |  | 	cmd.Stdout = p.logWriter | 
					
						
							|  |  |  | 	cmd.Stderr = p.logWriter | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	go func() { | 
					
						
							|  |  |  | 		err := cmd.Run() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2025-09-14 18:04:12 +00:00
										 |  |  | 			// On Windows, killing the process results in "exit status 1" which isn't an error for us
 | 
					
						
							|  |  |  | 			if runtime.GOOS == "windows" { | 
					
						
							|  |  |  | 				errStr := strings.ToLower(err.Error()) | 
					
						
							|  |  |  | 				if strings.Contains(errStr, "exit status 1") { | 
					
						
							|  |  |  | 					// Suppress noisy log on graceful stop
 | 
					
						
							|  |  |  | 					p.exitErr = err | 
					
						
							|  |  |  | 					return | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-07-08 21:08:00 +00:00
										 |  |  | 			logger.Error("Failure in running xray-core:", err) | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | 			p.exitErr = err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p.refreshVersion() | 
					
						
							|  |  |  | 	p.refreshAPIPort() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // Stop terminates the running Xray process.
 | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | func (p *process) Stop() error { | 
					
						
							|  |  |  | 	if !p.IsRunning() { | 
					
						
							|  |  |  | 		return errors.New("xray is not running") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-09-14 18:04:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-03 22:45:50 +00:00
										 |  |  | 	if runtime.GOOS == "windows" { | 
					
						
							|  |  |  | 		return p.cmd.Process.Kill() | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return p.cmd.Process.Signal(syscall.SIGTERM) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-02-09 19:18:06 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-11-21 13:55:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-20 07:35:50 +00:00
										 |  |  | // writeCrashReport writes a crash report to the binary folder with a timestamped filename.
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:36:02 +00:00
										 |  |  | func writeCrashReport(m []byte) error { | 
					
						
							| 
									
										
										
										
											2024-11-21 13:55:11 +00:00
										 |  |  | 	crashReportPath := config.GetBinFolderPath() + "/core_crash_" + time.Now().Format("20060102_150405") + ".log" | 
					
						
							| 
									
										
										
										
											2024-12-03 22:07:24 +00:00
										 |  |  | 	return os.WriteFile(crashReportPath, m, os.ModePerm) | 
					
						
							| 
									
										
										
										
											2024-11-21 13:55:11 +00:00
										 |  |  | } |