diff --git a/cmd/finch/main.go b/cmd/finch/main.go index dfef4cbd9..1c95ca958 100644 --- a/cmd/finch/main.go +++ b/cmd/finch/main.go @@ -96,6 +96,7 @@ var newApp = func(logger flog.Logger, fp path.Finch, fs afero.Fs, fc *config.Fin support.NewBundleConfig(fp, system.NewStdLib().Env("HOME")), fp, ecc, + lcc, wrapper.NewLimaWrapper(), ) diff --git a/cmd/finch/support_bundle.go b/cmd/finch/support_bundle.go index 0ccd78735..39062ff0c 100644 --- a/cmd/finch/support_bundle.go +++ b/cmd/finch/support_bundle.go @@ -35,7 +35,8 @@ func newSupportBundleGenerateCommand(logger flog.Logger, builder support.BundleB } supportBundleGenerateCommand.Flags().StringArray("include", []string{}, - "additional files to include in the support bundle, specified by absolute or relative path") + //nolint:lll // usage string + `additional files to include in the support bundle, specified by absolute or relative path. to include a file from the VM, prefix the file path with "vm:"`) supportBundleGenerateCommand.Flags().StringArray("exclude", []string{}, //nolint:lll // usage string "files to exclude from the support bundle. if you specify a base name, all files matching that base name will be excluded. if you specify an absolute or relative path, only exact matches will be excluded") diff --git a/pkg/command/command.go b/pkg/command/command.go index d726d585c..478aeedd4 100644 --- a/pkg/command/command.go +++ b/pkg/command/command.go @@ -26,6 +26,8 @@ type Command interface { SetStderr(io.Writer) Run() error + Start() error + Wait() error Output() ([]byte, error) CombinedOutput() ([]byte, error) } diff --git a/pkg/mocks/command_command.go b/pkg/mocks/command_command.go index e1d621e7b..c9006d599 100644 --- a/pkg/mocks/command_command.go +++ b/pkg/mocks/command_command.go @@ -128,3 +128,31 @@ func (mr *CommandMockRecorder) SetStdout(arg0 interface{}) *gomock.Call { mr.mock.ctrl.T.Helper() return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "SetStdout", reflect.TypeOf((*Command)(nil).SetStdout), arg0) } + +// Start mocks base method. +func (m *Command) Start() error { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "Start") + ret0, _ := ret[0].(error) + return ret0 +} + +// Start indicates an expected call of Start. +func (mr *CommandMockRecorder) Start() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Start", reflect.TypeOf((*Command)(nil).Start)) +} + +// Wait mocks base method. +func (m *Command) Wait() error { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "Wait") + ret0, _ := ret[0].(error) + return ret0 +} + +// Wait indicates an expected call of Wait. +func (mr *CommandMockRecorder) Wait() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Wait", reflect.TypeOf((*Command)(nil).Wait)) +} diff --git a/pkg/support/support.go b/pkg/support/support.go index 4d5b54022..09d9986aa 100644 --- a/pkg/support/support.go +++ b/pkg/support/support.go @@ -6,6 +6,7 @@ package support import ( "archive/zip" + "bufio" "bytes" "errors" "fmt" @@ -51,6 +52,7 @@ type bundleBuilder struct { config BundleConfig finch fpath.Finch ecc command.Creator + lcc command.LimaCmdCreator lima wrapper.LimaWrapper } @@ -61,6 +63,7 @@ func NewBundleBuilder( config BundleConfig, finch fpath.Finch, ecc command.Creator, + lcc command.LimaCmdCreator, lima wrapper.LimaWrapper, ) BundleBuilder { return &bundleBuilder{ @@ -69,6 +72,7 @@ func NewBundleBuilder( config: config, finch: finch, ecc: ecc, + lcc: lcc, lima: lima, } } @@ -108,7 +112,8 @@ func (bb *bundleBuilder) GenerateSupportBundle(additionalFiles []string, exclude bb.logger.Infof("Excluding %s...", file) continue } - err := bb.copyInFile(writer, file, path.Join(zipPrefix, logPrefix)) + bb.logger.Debugf("Copying %s...", file) + err = bb.copyFileFromVMOrLocal(writer, file, path.Join(zipPrefix, logPrefix)) if err != nil { bb.logger.Warnf("Could not copy in %q. Error: %s", file, err) } @@ -120,7 +125,8 @@ func (bb *bundleBuilder) GenerateSupportBundle(additionalFiles []string, exclude bb.logger.Infof("Excluding %s...", file) continue } - err := bb.copyInFile(writer, file, path.Join(zipPrefix, configPrefix)) + bb.logger.Debugf("Copying %s...", file) + err = bb.copyFileFromVMOrLocal(writer, file, path.Join(zipPrefix, configPrefix)) if err != nil { bb.logger.Warnf("Could not copy in %q. Error: %s", file, err) } @@ -132,7 +138,8 @@ func (bb *bundleBuilder) GenerateSupportBundle(additionalFiles []string, exclude bb.logger.Infof("Excluding %s...", file) continue } - err := bb.copyInFile(writer, file, path.Join(zipPrefix, additionalPrefix)) + bb.logger.Debugf("Copying %s...", file) + err = bb.copyFileFromVMOrLocal(writer, file, path.Join(zipPrefix, additionalPrefix)) if err != nil { bb.logger.Warnf("Could not add additional file %s. Error: %s", file, err) } @@ -146,30 +153,28 @@ func (bb *bundleBuilder) GenerateSupportBundle(additionalFiles []string, exclude return zipFileName, nil } -func (bb *bundleBuilder) copyInFile(writer *zip.Writer, fileName string, prefix string) error { - f, err := bb.fs.Open(fileName) - if err != nil { - return err - } - - bb.logger.Debugf("Copying %s...", fileName) +type bufReader interface { + ReadBytes(delim byte) ([]byte, error) +} - var buf bytes.Buffer - _, err = buf.ReadFrom(f) - if err != nil { - return err +func (bb *bundleBuilder) copyFileFromVMOrLocal(writer *zip.Writer, filename, zipPath string) error { + if isFileFromVM(filename) { + return bb.streamFileFromVM(writer, filename, zipPath) } + return bb.copyInFile(writer, filename, zipPath) +} - var redacted []byte +func (bb *bundleBuilder) copyAndRedactFile(writer io.Writer, reader bufReader) error { var bufErr error for bufErr == nil { var line []byte - line, bufErr = buf.ReadBytes('\n') + line, bufErr = reader.ReadBytes('\n') if bufErr != nil && !errors.Is(bufErr, io.EOF) { + bb.logger.Error(bufErr.Error()) continue } - line, err = redactFinchInstall(line, bb.finch) + line, err := redactFinchInstall(line, bb.finch) if err != nil { return err } @@ -187,7 +192,20 @@ func (bb *bundleBuilder) copyInFile(writer *zip.Writer, fileName string, prefix line = redactPorts(line) line = redactSSHKeys(line) - redacted = append(redacted, line...) + _, err = writer.Write(line) + if err != nil { + return err + } + } + + return nil +} + +func (bb *bundleBuilder) copyInFile(writer *zip.Writer, fileName string, prefix string) error { + // check filename validity? + f, err := bb.fs.Open(fileName) + if err != nil { + return err } baseName := path.Base(fileName) @@ -196,12 +214,56 @@ func (bb *bundleBuilder) copyInFile(writer *zip.Writer, fileName string, prefix return err } - _, err = zipCopy.Write(redacted) + buf := new(bytes.Buffer) + _, err = buf.ReadFrom(f) if err != nil { return err } - return nil + return bb.copyAndRedactFile(zipCopy, buf) +} + +func (bb *bundleBuilder) streamFileFromVM(writer *zip.Writer, filename, prefix string) error { + pipeReader, pipeWriter := io.Pipe() + errBuf := new(bytes.Buffer) + + _, filePathInVM, _ := strings.Cut(filename, ":") + cmd := bb.lcc.CreateWithoutStdio("shell", "finch", "sudo", "cat", filePathInVM) + cmd.SetStdout(pipeWriter) + cmd.SetStderr(errBuf) + + err := cmd.Start() + if err != nil { + return err + } + + waitStatus := make(chan error) + go func() { + err := cmd.Wait() + if err != nil { + errorMsg, readErr := io.ReadAll(errBuf) + if readErr == nil && len(errorMsg) > 0 { + err = errors.New(string(errorMsg)) + } + } + _ = pipeWriter.Close() + waitStatus <- err + }() + + baseName := path.Base(filename) + zipCopy, err := writer.Create(path.Join(prefix, baseName)) + if err != nil { + return err + } + + bufReader := bufio.NewReader(pipeReader) + + err = bb.copyAndRedactFile(zipCopy, bufReader) + if err != nil { + return err + } + + return <-waitStatus } func (bb *bundleBuilder) getPlatformData() (*PlatformData, error) { @@ -280,7 +342,11 @@ func bundleFileName() string { } func fileShouldBeExcluded(filename string, exclude []string) bool { - fileAbs, err := filepath.Abs(filename) + realFilename := filename + if isFileFromVM(filename) { + _, realFilename, _ = strings.Cut(filename, ":") + } + fileAbs, err := filepath.Abs(realFilename) if err != nil { return true } @@ -292,9 +358,13 @@ func fileShouldBeExcluded(filename string, exclude []string) bool { if fileAbs == excludeAbs { return true } - if path.Base(filename) == excludeFile { + if path.Base(realFilename) == excludeFile { return true } } return false } + +func isFileFromVM(filename string) bool { + return strings.HasPrefix(filename, "vm:") +} diff --git a/pkg/support/support_test.go b/pkg/support/support_test.go index 33c16c30e..cfca25974 100644 --- a/pkg/support/support_test.go +++ b/pkg/support/support_test.go @@ -5,6 +5,7 @@ package support import ( "archive/zip" + "io" "os/user" "testing" "time" @@ -23,13 +24,14 @@ func TestSupport_NewBundleBuilder(t *testing.T) { ctrl := gomock.NewController(t) ecc := mocks.NewCommandCreator(ctrl) + lcc := mocks.NewLimaCmdCreator(ctrl) logger := mocks.NewLogger(ctrl) fs := afero.NewMemMapFs() finch := fpath.Finch("mockfinch") lima := mocks.NewMockLimaWrapper(ctrl) config := NewBundleConfig(finch, "mockhome") - NewBundleBuilder(logger, fs, config, finch, ecc, lima) + NewBundleBuilder(logger, fs, config, finch, ecc, lcc, lima) } func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { @@ -41,7 +43,15 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { testCases := []struct { name string - mockSvc func(*mocks.Logger, *mocks.BundleConfig, *mocks.CommandCreator, *mocks.Command, *mocks.MockLimaWrapper) + mockSvc func( + *mocks.Logger, + *mocks.BundleConfig, + *mocks.CommandCreator, + *mocks.LimaCmdCreator, + *mocks.Command, + *mocks.MockLimaWrapper, + afero.Fs, + ) include []string exclude []string }{ @@ -51,8 +61,10 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { logger *mocks.Logger, config *mocks.BundleConfig, ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, cmd *mocks.Command, lima *mocks.MockLimaWrapper, + fs afero.Fs, ) { logger.EXPECT().Debugf("Creating %s...", gomock.Any()) logger.EXPECT().Debugln("Gathering platform data...") @@ -91,8 +103,10 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { logger *mocks.Logger, config *mocks.BundleConfig, ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, cmd *mocks.Command, lima *mocks.MockLimaWrapper, + fs afero.Fs, ) { logger.EXPECT().Debugf("Creating %s...", gomock.Any()) logger.EXPECT().Debugln("Gathering platform data...") @@ -128,8 +142,10 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { logger *mocks.Logger, config *mocks.BundleConfig, ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, cmd *mocks.Command, lima *mocks.MockLimaWrapper, + fs afero.Fs, ) { logger.EXPECT().Debugf("Creating %s...", gomock.Any()) logger.EXPECT().Debugln("Gathering platform data...") @@ -164,8 +180,10 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { logger *mocks.Logger, config *mocks.BundleConfig, ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, cmd *mocks.Command, lima *mocks.MockLimaWrapper, + fs afero.Fs, ) { logger.EXPECT().Debugf("Creating %s...", gomock.Any()) logger.EXPECT().Debugln("Gathering platform data...") @@ -200,8 +218,10 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { logger *mocks.Logger, config *mocks.BundleConfig, ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, cmd *mocks.Command, lima *mocks.MockLimaWrapper, + fs afero.Fs, ) { logger.EXPECT().Debugf("Creating %s...", gomock.Any()) logger.EXPECT().Debugln("Gathering platform data...") @@ -231,6 +251,64 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { include: []string{"extra1"}, exclude: []string{"extra1"}, }, + { + name: "Generate support bundle with a VM file included", + mockSvc: func( + logger *mocks.Logger, + config *mocks.BundleConfig, + ecc *mocks.CommandCreator, + lcc *mocks.LimaCmdCreator, + cmd *mocks.Command, + lima *mocks.MockLimaWrapper, + fs afero.Fs, + ) { + logger.EXPECT().Debugf("Creating %s...", gomock.Any()) + logger.EXPECT().Debugln("Gathering platform data...") + + ecc.EXPECT().Create("sw_vers", "-productVersion").Return(cmd) + cmd.EXPECT().Output().Return([]byte("1.2.3\n"), nil) + ecc.EXPECT().Create("uname", "-m").Return(cmd) + cmd.EXPECT().Output().Return([]byte("arch\n"), nil) + + config.EXPECT().LogFiles().Return([]string{ + "log1", + }) + + config.EXPECT().ConfigFiles().Return([]string{ + "config1", + }) + + logger.EXPECT().Debugln("Copying in log files...") + logger.EXPECT().Debugf("Copying %s...", "log1") + logger.EXPECT().Debugln("Copying in config files...") + logger.EXPECT().Debugf("Copying %s...", "config1") + logger.EXPECT().Debugln("Copying in additional files...") + logger.EXPECT().Debugf("Copying %s...", "vm:extra1") + + var catWriter io.Writer + waitChan := make(chan int) + lcc.EXPECT().CreateWithoutStdio("shell", "finch", "sudo", "cat", "extra1").Return(cmd) + cmd.EXPECT().SetStdout(gomock.Any()).Do(func(writer io.Writer) { + catWriter = writer + }) + cmd.EXPECT().SetStderr(gomock.Any()) + cmd.EXPECT().Start().DoAndReturn(func() error { + go func() { + _, err := catWriter.Write([]byte("file contents\n")) + require.NoError(t, err) + waitChan <- 1 + }() + return nil + }) + cmd.EXPECT().Wait().DoAndReturn(func() error { + <-waitChan + return nil + }) + + lima.EXPECT().LimaUser(false).Return(mockUser, nil).AnyTimes() + }, + include: []string{"vm:extra1"}, + }, } for _, tc := range testCases { @@ -244,6 +322,7 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { config := mocks.NewBundleConfig(ctrl) finch := fpath.Finch("mockfinch") ecc := mocks.NewCommandCreator(ctrl) + lcc := mocks.NewLimaCmdCreator(ctrl) lima := mocks.NewMockLimaWrapper(ctrl) cmd := mocks.NewCommand(ctrl) @@ -253,6 +332,7 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { config: config, finch: finch, ecc: ecc, + lcc: lcc, lima: lima, } @@ -273,7 +353,7 @@ func TestSupportBundleBuilder_GenerateSupportBundle(t *testing.T) { require.NoError(t, err) } - tc.mockSvc(logger, config, ecc, cmd, lima) + tc.mockSvc(logger, config, ecc, lcc, cmd, lima, fs) zipFile, err := builder.GenerateSupportBundle(tc.include, tc.exclude) assert.NoError(t, err) @@ -377,6 +457,18 @@ func TestSupport_fileShouldBeExcluded(t *testing.T) { exclude: []string{"other.file"}, result: false, }, + { + name: "vm file with its whole path excluded", + file: "vm:/path/to/file", + exclude: []string{"/path/to/file"}, + result: true, + }, + { + name: "vm file with its base path excluded", + file: "vm:/path/to/file", + exclude: []string{"file"}, + result: true, + }, } for _, tc := range testCases {