summaryrefslogtreecommitdiff
path: root/integrationtests/dgrep_server_helpers.go
blob: c65a74e58a99257ced5123eb6f89191dd050495c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
package integrationtests

import (
	"context"
	"fmt"
	"os"
	"strings"
	"testing"
	"time"
)

// testDGrepWithServer tests dgrep command with a running server
func testDGrepWithServer(t *testing.T, args []string, outFile, expectedFile string) error {
	port := getUniquePortNumber()
	bindAddress := "localhost"
	
	ctx, cancel := context.WithCancel(context.Background())
	defer cancel()

	// Start server
	serverCh, _, _, err := startCommand(ctx, t,
		"", "../dserver",
		"--cfg", "none",
		"--logger", "stdout",
		"--logLevel", "error",
		"--bindAddress", bindAddress,
		"--port", fmt.Sprintf("%d", port),
	)
	if err != nil {
		return fmt.Errorf("failed to start server: %v", err)
	}

	// Give server time to start
	time.Sleep(1 * time.Second)

	// Prepare dgrep args with server connection
	dgrepArgs := append([]string{
		"--servers", fmt.Sprintf("%s:%d", bindAddress, port),
		"--trustAllHosts",
		"--noColor",
	}, args...)

	// Start dgrep client
	clientCh, _, _, err := startCommand(ctx, t,
		"", "../dgrep", dgrepArgs...)
	if err != nil {
		cancel()
		return fmt.Errorf("failed to start dgrep client: %v", err)
	}

	// Collect all output
	var output []string
	timeout := time.After(15 * time.Second)
	linesReceived := 0
	
	for {
		select {
		case line := <-serverCh:
			// Only log important server errors, not routine messages
			if strings.Contains(line, "ERROR|") && 
			   !strings.Contains(line, "use of closed network connection") {
				t.Logf("server error: %s", line)
			}
		case line := <-clientCh:
			// Only log client errors if needed
			// Process empty lines too - they are valid content
			// Don't skip empty lines as they may be meaningful
			
			if strings.HasPrefix(line, "REMOTE|") {
				// Extract the actual content from DTail protocol format
				// Format: REMOTE|hostname|priority|lineno|sourceID|hostname|filename|linenum|content
				parts := strings.Split(line, "|")
				if len(parts) >= 8 {
					content := strings.Join(parts[7:], "|")
					// Remove line number prefix if present (from DTail format)
					if strings.Contains(content, " ") {
						contentParts := strings.SplitN(content, " ", 2)
						if len(contentParts) == 2 {
							output = append(output, contentParts[1])
						} else {
							output = append(output, content)
						}
					} else {
						output = append(output, content)
					}
					linesReceived++
				}
			} else if strings.HasPrefix(line, "CLIENT|") {
				// Client status messages - ignore
				continue
			} else {
				// Direct content line from dgrep (not wrapped in protocol)
				// Include empty lines as they are meaningful content
				output = append(output, line)
				linesReceived++
			}
		case <-timeout:
			// Timeout reached, finish collecting
			goto done
		case <-ctx.Done():
			goto done
		}
		
		// If we received some output and haven't seen new lines for a bit, we're probably done
		if linesReceived > 0 {
			select {
			case <-time.After(500 * time.Millisecond):
				goto done
			default:
				continue
			}
		}
	}

done:
	cancel()
	
	// Write collected output to file
	if len(output) > 0 {
		fd, err := os.Create(outFile)
		if err != nil {
			return fmt.Errorf("failed to create output file: %v", err)
		}
		defer fd.Close()
		
		// Check if the expected file ends with a newline to match format
		expectedData, err := os.ReadFile(expectedFile)
		if err != nil {
			return fmt.Errorf("failed to read expected file: %v", err)
		}
		
		endsWithNewline := len(expectedData) > 0 && expectedData[len(expectedData)-1] == '\n'
		
		for i, line := range output {
			if i == len(output)-1 && !endsWithNewline {
				// Last line and original doesn't end with newline
				fd.WriteString(line)
			} else {
				fd.WriteString(line + "\n")
			}
		}
	}

	// Compare results
	if err := compareFiles(t, outFile, expectedFile); err != nil {
		return err
	}

	os.Remove(outFile)
	return nil
}