|
6 | 6 | "fmt" |
7 | 7 | "net" |
8 | 8 | "net/http" |
| 9 | + "strings" |
9 | 10 | "time" |
10 | 11 |
|
11 | 12 | "github.com/brevdev/cloud/internal/ssh" |
@@ -141,6 +142,144 @@ func ValidateDockerFirewallBlocksPort(ctx context.Context, client CloudInstanceR |
141 | 142 | return nil |
142 | 143 | } |
143 | 144 |
|
| 145 | +func ValidateDockerFirewallAllowsEgress(ctx context.Context, client CloudInstanceReader, instance *Instance, privateKey string) error { |
| 146 | + var err error |
| 147 | + instance, err = WaitForInstanceLifecycleStatus(ctx, client, instance, LifecycleStatusRunning, PendingToRunningTimeout) |
| 148 | + if err != nil { |
| 149 | + return fmt.Errorf("failed to wait for instance running: %w", err) |
| 150 | + } |
| 151 | + |
| 152 | + publicIP := instance.PublicIP |
| 153 | + if publicIP == "" { |
| 154 | + return fmt.Errorf("public IP is not available for instance %s", instance.CloudID) |
| 155 | + } |
| 156 | + |
| 157 | + sshClient, err := ssh.ConnectToHost(ctx, ssh.ConnectionConfig{ |
| 158 | + User: instance.SSHUser, |
| 159 | + HostPort: fmt.Sprintf("%s:%d", publicIP, instance.SSHPort), |
| 160 | + PrivKey: privateKey, |
| 161 | + }) |
| 162 | + if err != nil { |
| 163 | + return fmt.Errorf("failed to SSH into instance: %w", err) |
| 164 | + } |
| 165 | + defer func() { _ = sshClient.Close() }() |
| 166 | + |
| 167 | + dockerCmd, err := setupDockerCommand(ctx, sshClient, instance.CloudID) |
| 168 | + if err != nil { |
| 169 | + return err |
| 170 | + } |
| 171 | + |
| 172 | + // Pull the alpine image |
| 173 | + cmd := fmt.Sprintf( |
| 174 | + "%s pull alpine", |
| 175 | + dockerCmd, |
| 176 | + ) |
| 177 | + _, stderr, err := sshClient.RunCommand(ctx, cmd) |
| 178 | + if err != nil { |
| 179 | + return fmt.Errorf("failed to pull alpine image: %w, stderr: %s", err, stderr) |
| 180 | + } |
| 181 | + |
| 182 | + // Start a Docker container to ping Google's DNS server |
| 183 | + cmd = fmt.Sprintf( |
| 184 | + "%s run --rm alpine ping -c 3 8.8.8.8", |
| 185 | + dockerCmd, |
| 186 | + ) |
| 187 | + stdout, stderr, err := sshClient.RunCommand(ctx, cmd) |
| 188 | + if err != nil { |
| 189 | + return fmt.Errorf("failed to connect to Google's DNS server: %w, stderr: %s", err, stderr) |
| 190 | + } |
| 191 | + if !strings.Contains(stdout, "3 packets transmitted, 3 packets received") { |
| 192 | + return fmt.Errorf("expected successful ping, got: %s", stdout) |
| 193 | + } |
| 194 | + |
| 195 | + return nil |
| 196 | +} |
| 197 | + |
| 198 | +func ValidateDockerFirewallAllowsContainerToContainerCommunication(ctx context.Context, client CloudInstanceReader, instance *Instance, privateKey string) error { |
| 199 | + var err error |
| 200 | + instance, err = WaitForInstanceLifecycleStatus(ctx, client, instance, LifecycleStatusRunning, PendingToRunningTimeout) |
| 201 | + if err != nil { |
| 202 | + return fmt.Errorf("failed to wait for instance running: %w", err) |
| 203 | + } |
| 204 | + |
| 205 | + publicIP := instance.PublicIP |
| 206 | + if publicIP == "" { |
| 207 | + return fmt.Errorf("public IP is not available for instance %s", instance.CloudID) |
| 208 | + } |
| 209 | + |
| 210 | + sshClient, err := ssh.ConnectToHost(ctx, ssh.ConnectionConfig{ |
| 211 | + User: instance.SSHUser, |
| 212 | + HostPort: fmt.Sprintf("%s:%d", publicIP, instance.SSHPort), |
| 213 | + PrivKey: privateKey, |
| 214 | + }) |
| 215 | + if err != nil { |
| 216 | + return fmt.Errorf("failed to SSH into instance: %w", err) |
| 217 | + } |
| 218 | + defer func() { _ = sshClient.Close() }() |
| 219 | + |
| 220 | + dockerCmd, err := setupDockerCommand(ctx, sshClient, instance.CloudID) |
| 221 | + if err != nil { |
| 222 | + return err |
| 223 | + } |
| 224 | + |
| 225 | + // Create a docker network |
| 226 | + networkName := "firewall-test-network" |
| 227 | + cmd := fmt.Sprintf( |
| 228 | + "%s network create %s", |
| 229 | + dockerCmd, networkName, |
| 230 | + ) |
| 231 | + _, stderr, err := sshClient.RunCommand(ctx, cmd) |
| 232 | + if err != nil { |
| 233 | + return fmt.Errorf("failed to create docker network: %w, stderr: %s", err, stderr) |
| 234 | + } |
| 235 | + |
| 236 | + // Pull the alpine image |
| 237 | + cmd = fmt.Sprintf( |
| 238 | + "%s pull alpine", |
| 239 | + dockerCmd, |
| 240 | + ) |
| 241 | + _, stderr, err = sshClient.RunCommand(ctx, cmd) |
| 242 | + if err != nil { |
| 243 | + return fmt.Errorf("failed to pull alpine image: %w, stderr: %s", err, stderr) |
| 244 | + } |
| 245 | + |
| 246 | + // Pull the nginx image |
| 247 | + cmd = fmt.Sprintf( |
| 248 | + "%s pull nginx:alpine", |
| 249 | + dockerCmd, |
| 250 | + ) |
| 251 | + _, stderr, err = sshClient.RunCommand(ctx, cmd) |
| 252 | + if err != nil { |
| 253 | + return fmt.Errorf("failed to pull nginx image: %w, stderr: %s", err, stderr) |
| 254 | + } |
| 255 | + |
| 256 | + // Start a Docker container in the background |
| 257 | + containerName := "firewall-test-container-to-container" |
| 258 | + cmd = fmt.Sprintf( |
| 259 | + "%s run -d --name %s --network %s nginx:alpine", |
| 260 | + dockerCmd, containerName, networkName, |
| 261 | + ) |
| 262 | + _, stderr, err = sshClient.RunCommand(ctx, cmd) |
| 263 | + if err != nil { |
| 264 | + return fmt.Errorf("failed to start docker container: %w, stderr: %s", err, stderr) |
| 265 | + } |
| 266 | + |
| 267 | + // Start a second Docker container to connect to the first container |
| 268 | + cmd = fmt.Sprintf( |
| 269 | + "%s run --network %s --rm alpine wget -q -O- http://%s", |
| 270 | + dockerCmd, networkName, containerName, |
| 271 | + ) |
| 272 | + stdout, stderr, err := sshClient.RunCommand(ctx, cmd) |
| 273 | + if err != nil { |
| 274 | + return fmt.Errorf("failed to connect to nginx container: %w, stderr: %s", err, stderr) |
| 275 | + } |
| 276 | + |
| 277 | + if !strings.Contains(stdout, "Welcome to nginx") { |
| 278 | + return fmt.Errorf("expected successful wget, got: %s", stdout) |
| 279 | + } |
| 280 | + return nil |
| 281 | +} |
| 282 | + |
144 | 283 | // setupDockerCommand ensures Docker is available and returns the command to use (always with sudo) |
145 | 284 | func setupDockerCommand(ctx context.Context, sshClient *ssh.Client, instanceID CloudProviderInstanceID) (string, error) { |
146 | 285 | // Check if Docker is available |
|
0 commit comments