Go за Прикладом: Директорії

У Go є кілька корисних функцій для роботи з директоріями файлової системи.

package main
import (
    "fmt"
    "io/fs"
    "os"
    "path/filepath"
)
func check(e error) {
    if e != nil {
        panic(e)
    }
}
func main() {

Створюємо директорію за назвою subdir. Другим параметром є налаштування параметрів доступу до директорії.

    err := os.Mkdir("subdir", 0o755)
    check(err)

При створенні тимчасової директорії, гарною ідеєю є викликати її видалення за допомогою відкладення. Функція os.RemoveAll видаляє директорію повністю (разом з вкладеними), за аналогією доrm -rf subdir в UNIX-подібних системах.

    defer os.RemoveAll("subdir")

Допоміжна функція що створює тимчасовий файл.

    createEmptyFile := func(name string) {
        d := []byte("")
        check(os.WriteFile(name, d, 0o644))
    }
    createEmptyFile("subdir/file1")

Також можливо створити повну ієрархію директорій за допомогою MkdirAll. Ця функція є аналогом до команди mkdir -p subdir/parent/child.

    err = os.MkdirAll("subdir/parent/child", 0o755)
    check(err)
    createEmptyFile("subdir/parent/file2")
    createEmptyFile("subdir/parent/file3")
    createEmptyFile("subdir/parent/child/file4")

ReadDir зчитає зміст директорії parent, і поверне зріз об’єктів os.FileInfo.

    c, err := os.ReadDir("subdir/parent")
    check(err)
    fmt.Println("Listing subdir/parent")
    for _, entry := range c {
        fmt.Println(" ", entry.Name(), entry.IsDir())
    }

Chdir змінює поточну робочу директорію (так само, як і команда cd).

    err = os.Chdir("subdir/parent/child")
    check(err)

Ось, ‘ioutil.ReadDir’ повертає зріз об’єктів os.FileInfo для поточної директорії (якою на даний момент є subdir/parent/child).

    c, err = os.ReadDir(".")
    check(err)
    fmt.Println("Listing subdir/parent/child")
    for _, entry := range c {
        fmt.Println(" ", entry.Name(), entry.IsDir())
    }

Повертаємось до початкової директорії.

    err = os.Chdir("../../..")
    check(err)

Ми можемо обійти директорію рекурсивно, відвідуючи всі вкладені директорії. Walk другим параметром приймає функцію зворотного виклику, яка викликається для кожного знайденого файлу та директорії.

    fmt.Println("Visiting subdir")
    err = filepath.WalkDir("subdir", visit)
}

Функція visit буде викликана для кожного файлу чи директорії знайденого в процесі обходу subdir.

func visit(path string, d fs.DirEntry, err error) error {
    if err != nil {
        return err
    }
    fmt.Println(" ", path, d.IsDir())
    return nil
}
$ go run directories.go
Listing subdir/parent
  child true
  file2 false
  file3 false
Listing subdir/parent/child
  file4 false
Visiting subdir
  subdir true
  subdir/file1 false
  subdir/parent true
  subdir/parent/child true
  subdir/parent/child/file4 false
  subdir/parent/file2 false
  subdir/parent/file3 false

Наступний приклад: Тимчасові Файли та Директорії.