Merge pull request #9484 from Abdulkbk/refactor-makedir

lnutils: add CreateDir util function
This commit is contained in:
Oliver Gugger 2025-02-25 01:00:45 -06:00 committed by GitHub
commit 5d3680a6f6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 118 additions and 0 deletions

31
lnutils/fs.go Normal file
View file

@ -0,0 +1,31 @@
package lnutils
import (
"errors"
"fmt"
"os"
)
// CreateDir creates a directory if it doesn't exist and also handles
// symlink-related errors with user-friendly messages. It creates all necessary
// parent directories with the specified permissions.
func CreateDir(dir string, perm os.FileMode) error {
err := os.MkdirAll(dir, perm)
if err == nil {
return nil
}
// Show a nicer error message if it's because a symlink
// is linked to a directory that does not exist
// (probably because it's not mounted).
var pathErr *os.PathError
if errors.As(err, &pathErr) && os.IsExist(err) {
link, lerr := os.Readlink(pathErr.Path)
if lerr == nil {
return fmt.Errorf("is symlink %s -> %s "+
"mounted?", pathErr.Path, link)
}
}
return fmt.Errorf("failed to create directory '%s': %w", dir, err)
}

87
lnutils/fs_test.go Normal file
View file

@ -0,0 +1,87 @@
package lnutils
import (
"os"
"path/filepath"
"testing"
"github.com/stretchr/testify/require"
)
// TestCreateDir verifies the behavior of CreateDir function in various
// scenarios:
// - Creating a new directory when it doesn't exist
// - Handling an already existing directory
// - Dealing with symlinks pointing to non-existent directories
// - Handling invalid paths
// The test uses a temporary directory and runs multiple test cases to ensure
// proper directory creation, permission settings, and error handling.
func TestCreateDir(t *testing.T) {
t.Parallel()
tempDir := t.TempDir()
tests := []struct {
name string
setup func() string
wantError bool
}{
{
name: "create directory",
setup: func() string {
return filepath.Join(tempDir, "testdir")
},
wantError: false,
},
{
name: "existing directory",
setup: func() string {
dir := filepath.Join(tempDir, "testdir")
err := os.Mkdir(dir, 0700)
require.NoError(t, err)
return dir
},
wantError: false,
},
{
name: "symlink to non-existent directory",
setup: func() string {
dir := filepath.Join(tempDir, "testdir")
symlink := filepath.Join(tempDir, "symlink")
err := os.Symlink(dir, symlink)
require.NoError(t, err)
return symlink
},
wantError: true,
},
{
name: "invalid path",
setup: func() string {
return string([]byte{0})
},
wantError: true,
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
dir := tc.setup()
defer os.RemoveAll(dir)
err := CreateDir(dir, 0700)
if tc.wantError {
require.Error(t, err)
return
}
require.NoError(t, err)
info, err := os.Stat(dir)
require.NoError(t, err)
require.True(t, info.IsDir())
})
}
}