mirror of
https://github.com/SajadMRjl/find-me-internet.git
synced 2026-07-02 15:09:00 +00:00
132 lines
2.9 KiB
Go
132 lines
2.9 KiB
Go
package tester
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"net"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"find-me-internet/internal/model"
|
|
)
|
|
|
|
type Runner struct {
|
|
BinPath string
|
|
TestURL string
|
|
Timeout time.Duration
|
|
}
|
|
|
|
func NewRunner(binPath, testURL string, timeout time.Duration) *Runner {
|
|
return &Runner{BinPath: binPath, TestURL: testURL, Timeout: timeout}
|
|
}
|
|
|
|
func (r *Runner) Test(p *model.Proxy) error {
|
|
log := slog.With("target", p.Address, "sni", p.SNI)
|
|
|
|
// 1. Port Allocation
|
|
port, err := getFreePort()
|
|
if err != nil {
|
|
log.Error("local_port_allocation_failed", "error", err)
|
|
return err
|
|
}
|
|
|
|
// 2. Config Generation
|
|
configData, err := GenerateConfig(p, port)
|
|
if err != nil {
|
|
log.Error("config_generation_failed", "error", err)
|
|
return err
|
|
}
|
|
|
|
configName := filepath.Join(os.TempDir(), fmt.Sprintf("sb_%d_%s.json", port, p.Address))
|
|
if err := os.WriteFile(configName, configData, 0644); err != nil {
|
|
return err
|
|
}
|
|
defer os.Remove(configName)
|
|
|
|
// 3. Process Execution
|
|
ctx, cancel := context.WithTimeout(context.Background(), r.Timeout+2*time.Second)
|
|
defer cancel()
|
|
|
|
cmd := exec.CommandContext(ctx, r.BinPath, "run", "-c", configName)
|
|
if err := cmd.Start(); err != nil {
|
|
log.Error("singbox_process_start_failed", "error", err)
|
|
return err
|
|
}
|
|
|
|
defer func() {
|
|
if cmd.Process != nil {
|
|
cmd.Process.Kill()
|
|
}
|
|
}()
|
|
|
|
// 4. Wait for Binding
|
|
if !waitForPort(port, 2*time.Second) {
|
|
log.Debug("singbox_bind_timeout", "local_port", port)
|
|
return fmt.Errorf("process_bind_timeout")
|
|
}
|
|
|
|
// 5. HTTP Probe
|
|
startProbe := time.Now()
|
|
latency, err := r.measureLatency(port)
|
|
if err != nil {
|
|
log.Debug("http_probe_failed",
|
|
"duration", time.Since(startProbe),
|
|
"error", err,
|
|
)
|
|
return err
|
|
}
|
|
|
|
p.Latency = latency
|
|
return nil
|
|
}
|
|
|
|
func (r *Runner) measureLatency(port int) (time.Duration, error) {
|
|
proxyUrl, _ := url.Parse(fmt.Sprintf("http://127.0.0.1:%d", port))
|
|
client := &http.Client{
|
|
Transport: &http.Transport{
|
|
Proxy: http.ProxyURL(proxyUrl),
|
|
DisableKeepAlives: true,
|
|
},
|
|
Timeout: r.Timeout,
|
|
}
|
|
|
|
start := time.Now()
|
|
resp, err := client.Get(r.TestURL)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode < 200 || resp.StatusCode > 299 {
|
|
return 0, fmt.Errorf("unexpected_status_code_%d", resp.StatusCode)
|
|
}
|
|
|
|
return time.Since(start), nil
|
|
}
|
|
|
|
func getFreePort() (int, error) {
|
|
addr, err := net.ResolveTCPAddr("tcp", "localhost:0")
|
|
if err != nil { return 0, err }
|
|
l, err := net.ListenTCP("tcp", addr)
|
|
if err != nil { return 0, err }
|
|
defer l.Close()
|
|
return l.Addr().(*net.TCPAddr).Port, nil
|
|
}
|
|
|
|
func waitForPort(port int, timeout time.Duration) bool {
|
|
deadline := time.Now().Add(timeout)
|
|
for time.Now().Before(deadline) {
|
|
conn, err := net.DialTimeout("tcp", fmt.Sprintf("127.0.0.1:%d", port), 100*time.Millisecond)
|
|
if err == nil {
|
|
conn.Close()
|
|
return true
|
|
}
|
|
time.Sleep(50 * time.Millisecond)
|
|
}
|
|
return false
|
|
} |