|
| 1 | +package network |
| 2 | + |
| 3 | +import ( |
| 4 | + "context" |
| 5 | + "fmt" |
| 6 | + "net/http" |
| 7 | + "time" |
| 8 | + |
| 9 | + "github.com/loft-sh/log" |
| 10 | + "tailscale.com/client/tailscale" |
| 11 | + "tailscale.com/tsnet" |
| 12 | +) |
| 13 | + |
| 14 | +// HeartbeatService sends periodic heartbeats when there are active connections. |
| 15 | +type HeartbeatService struct { |
| 16 | + tsServer *tsnet.Server |
| 17 | + lc *tailscale.LocalClient |
| 18 | + config *WorkspaceServerConfig |
| 19 | + projectName string |
| 20 | + workspaceName string |
| 21 | + log log.Logger |
| 22 | + tracker *ConnTracker |
| 23 | +} |
| 24 | + |
| 25 | +// NewHeartbeatService creates a new HeartbeatService. |
| 26 | +func NewHeartbeatService(config *WorkspaceServerConfig, tsServer *tsnet.Server, lc *tailscale.LocalClient, projectName, workspaceName string, tracker *ConnTracker, log log.Logger) *HeartbeatService { |
| 27 | + return &HeartbeatService{ |
| 28 | + tsServer: tsServer, |
| 29 | + lc: lc, |
| 30 | + config: config, |
| 31 | + projectName: projectName, |
| 32 | + workspaceName: workspaceName, |
| 33 | + log: log, |
| 34 | + tracker: tracker, |
| 35 | + } |
| 36 | +} |
| 37 | + |
| 38 | +// Start begins the heartbeat loop. |
| 39 | +func (s *HeartbeatService) Start(ctx context.Context) { |
| 40 | + transport := &http.Transport{DialContext: s.tsServer.Dial} |
| 41 | + client := &http.Client{Transport: transport, Timeout: 10 * time.Second} |
| 42 | + ticker := time.NewTicker(10 * time.Second) |
| 43 | + defer ticker.Stop() |
| 44 | + for { |
| 45 | + select { |
| 46 | + case <-ctx.Done(): |
| 47 | + return |
| 48 | + case <-ticker.C: |
| 49 | + if s.tracker.Count() > 0 { |
| 50 | + if err := s.sendHeartbeat(ctx, client); err != nil { |
| 51 | + s.log.Errorf("HeartbeatService: failed to send heartbeat: %v", err) |
| 52 | + } |
| 53 | + } |
| 54 | + } |
| 55 | + } |
| 56 | +} |
| 57 | + |
| 58 | +func (s *HeartbeatService) sendHeartbeat(ctx context.Context, client *http.Client) error { |
| 59 | + hbCtx, cancel := context.WithTimeout(ctx, 10*time.Second) |
| 60 | + defer cancel() |
| 61 | + discoveredRunner, err := discoverRunner(hbCtx, s.lc, s.log) |
| 62 | + if err != nil { |
| 63 | + return fmt.Errorf("failed to discover runner: %w", err) |
| 64 | + } |
| 65 | + heartbeatURL := fmt.Sprintf("http://%s.ts.loft/devpod/%s/%s/heartbeat", discoveredRunner, s.projectName, s.workspaceName) |
| 66 | + s.log.Infof("HeartbeatService: sending heartbeat to %s, active connections: %d", heartbeatURL, s.tracker.Count()) |
| 67 | + req, err := http.NewRequestWithContext(hbCtx, "GET", heartbeatURL, nil) |
| 68 | + if err != nil { |
| 69 | + return fmt.Errorf("failed to create request for %s: %w", heartbeatURL, err) |
| 70 | + } |
| 71 | + req.Header.Set("Authorization", "Bearer "+s.config.AccessKey) |
| 72 | + resp, err := client.Do(req) |
| 73 | + if err != nil { |
| 74 | + return fmt.Errorf("request to %s failed: %w", heartbeatURL, err) |
| 75 | + } |
| 76 | + defer resp.Body.Close() |
| 77 | + if resp.StatusCode != http.StatusOK { |
| 78 | + return fmt.Errorf("received response from %s - Status: %d", heartbeatURL, resp.StatusCode) |
| 79 | + } |
| 80 | + s.log.Infof("HeartbeatService: received response from %s - Status: %d", heartbeatURL, resp.StatusCode) |
| 81 | + return nil |
| 82 | +} |
0 commit comments