- Pass registration token from URL query parameter to install script generation - Update RenderInstallScriptFromBuild to accept registration token - Add RegistrationToken field to template data structure This lays groundwork for fixing agent registration - install scripts will be able to call the registration API with the provided token.
194 lines
5.2 KiB
Go
194 lines
5.2 KiB
Go
package services
|
|
|
|
import (
|
|
"bytes"
|
|
"embed"
|
|
"fmt"
|
|
"log"
|
|
"strings"
|
|
"text/template"
|
|
|
|
"github.com/Fimeg/RedFlag/aggregator-server/internal/models"
|
|
"github.com/google/uuid"
|
|
)
|
|
|
|
//go:embed templates/install/scripts/*.tmpl
|
|
var installScriptTemplates embed.FS
|
|
|
|
// InstallTemplateService renders installation scripts from templates
|
|
type InstallTemplateService struct{}
|
|
|
|
// NewInstallTemplateService creates a new template service
|
|
func NewInstallTemplateService() *InstallTemplateService {
|
|
return &InstallTemplateService{}
|
|
}
|
|
|
|
// RenderInstallScript renders an installation script for the specified platform
|
|
func (s *InstallTemplateService) RenderInstallScript(agent *models.Agent, binaryURL, configURL string) (string, error) {
|
|
// Define template data
|
|
data := struct {
|
|
AgentID string
|
|
BinaryURL string
|
|
ConfigURL string
|
|
Platform string
|
|
Architecture string
|
|
Version string
|
|
}{
|
|
AgentID: agent.ID.String(),
|
|
BinaryURL: binaryURL,
|
|
ConfigURL: configURL,
|
|
Platform: agent.OSType,
|
|
Architecture: agent.OSArchitecture,
|
|
Version: agent.CurrentVersion,
|
|
}
|
|
|
|
// Choose template based on platform
|
|
var templateName string
|
|
if strings.Contains(agent.OSType, "windows") {
|
|
templateName = "templates/install/scripts/windows.ps1.tmpl"
|
|
} else {
|
|
templateName = "templates/install/scripts/linux.sh.tmpl"
|
|
}
|
|
|
|
// Load and parse template
|
|
tmpl, err := template.ParseFS(installScriptTemplates, templateName)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to load template: %w", err)
|
|
}
|
|
|
|
// Render template
|
|
var buf bytes.Buffer
|
|
if err := tmpl.Execute(&buf, data); err != nil {
|
|
return "", fmt.Errorf("failed to render template: %w", err)
|
|
}
|
|
|
|
return buf.String(), nil
|
|
}
|
|
|
|
// RenderInstallScriptFromBuild renders script using build response
|
|
func (s *InstallTemplateService) RenderInstallScriptFromBuild(
|
|
agentIDParam string,
|
|
platform string,
|
|
architecture string,
|
|
version string,
|
|
serverURL string,
|
|
registrationToken string,
|
|
) (string, error) {
|
|
// Extract or generate agent ID
|
|
agentID := s.extractOrGenerateAgentID(agentIDParam)
|
|
|
|
// Build correct URLs in Go, not templates
|
|
binaryURL := fmt.Sprintf("%s/api/v1/downloads/%s-%s?version=%s", serverURL, platform, architecture, version)
|
|
configURL := fmt.Sprintf("%s/api/v1/downloads/config/%s", serverURL, agentID)
|
|
|
|
data := struct {
|
|
AgentID string
|
|
BinaryURL string
|
|
ConfigURL string
|
|
Platform string
|
|
Architecture string
|
|
Version string
|
|
ServerURL string
|
|
RegistrationToken string
|
|
}{
|
|
AgentID: agentID,
|
|
BinaryURL: binaryURL,
|
|
ConfigURL: configURL,
|
|
Platform: platform,
|
|
Architecture: architecture,
|
|
Version: version,
|
|
ServerURL: serverURL,
|
|
RegistrationToken: registrationToken,
|
|
}
|
|
|
|
templateName := "templates/install/scripts/linux.sh.tmpl"
|
|
if strings.Contains(platform, "windows") {
|
|
templateName = "templates/install/scripts/windows.ps1.tmpl"
|
|
}
|
|
|
|
tmpl, err := template.ParseFS(installScriptTemplates, templateName)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
if err := tmpl.Execute(&buf, data); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return buf.String(), nil
|
|
}
|
|
|
|
// BuildAgentConfigWithAgentID builds config for an existing agent (for upgrades)
|
|
func (s *InstallTemplateService) BuildAgentConfigWithAgentID(
|
|
agentID string,
|
|
platform string,
|
|
architecture string,
|
|
version string,
|
|
serverURL string,
|
|
) (string, error) {
|
|
// Validate agent ID
|
|
if _, err := uuid.Parse(agentID); err != nil {
|
|
return "", fmt.Errorf("invalid agent ID: %w", err)
|
|
}
|
|
|
|
// Build correct URLs using existing agent ID
|
|
binaryURL := fmt.Sprintf("%s/api/v1/downloads/%s-%s?version=%s", serverURL, platform, architecture, version)
|
|
configURL := fmt.Sprintf("%s/api/v1/downloads/config/%s", serverURL, agentID)
|
|
|
|
data := struct {
|
|
AgentID string
|
|
BinaryURL string
|
|
ConfigURL string
|
|
Platform string
|
|
Architecture string
|
|
Version string
|
|
ServerURL string
|
|
}{
|
|
AgentID: agentID,
|
|
BinaryURL: binaryURL,
|
|
ConfigURL: configURL,
|
|
Platform: platform,
|
|
Architecture: architecture,
|
|
Version: version,
|
|
ServerURL: serverURL,
|
|
}
|
|
|
|
templateName := "templates/install/scripts/linux.sh.tmpl"
|
|
if strings.Contains(platform, "windows") {
|
|
templateName = "templates/install/scripts/windows.ps1.tmpl"
|
|
}
|
|
|
|
tmpl, err := template.ParseFS(installScriptTemplates, templateName)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
if err := tmpl.Execute(&buf, data); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return buf.String(), nil
|
|
}
|
|
|
|
// extractOrGenerateAgentID extracts or generates a valid agent ID
|
|
func (s *InstallTemplateService) extractOrGenerateAgentID(param string) string {
|
|
log.Printf("[DEBUG] extractOrGenerateAgentID received param: %s", param)
|
|
|
|
// If we got a real agent ID (UUID format), validate and use it
|
|
if param != "" && param != "<AGENT_ID>" {
|
|
// Validate it's a UUID
|
|
if _, err := uuid.Parse(param); err == nil {
|
|
log.Printf("[DEBUG] Using passed UUID: %s", param)
|
|
return param
|
|
}
|
|
log.Printf("[DEBUG] Invalid UUID format, generating new one")
|
|
}
|
|
|
|
// Placeholder case - generate new UUID for fresh installation
|
|
newID := uuid.New().String()
|
|
log.Printf("[DEBUG] Generated new UUID: %s", newID)
|
|
return newID
|
|
}
|