/* Copyright 2025 Dnote Authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package utils import ( "io" "os" "path/filepath" "github.com/pkg/errors" ) // ReadFileAbs reads the content of the file with the given file path by resolving // it as an absolute path func ReadFileAbs(relpath string) []byte { fp, err := filepath.Abs(relpath) if err != nil { panic(err) } b, err := os.ReadFile(fp) if err != nil { panic(err) } return b } // FileExists checks if the file exists at the given path func FileExists(filepath string) (bool, error) { _, err := os.Stat(filepath) if err == nil { return true, nil } if os.IsNotExist(err) { return false, nil } return false, errors.Wrap(err, "getting file info") } // EnsureDir creates a directory if it doesn't exist. // Returns nil if the directory already exists or was successfully created. func EnsureDir(path string) error { ok, err := FileExists(path) if err != nil { return errors.Wrapf(err, "checking if dir exists at %s", path) } if ok { return nil } if err := os.MkdirAll(path, 0755); err != nil { return errors.Wrapf(err, "creating directory at %s", path) } return nil } // CopyDir copies a directory from src to dest, recursively copying nested // directories func CopyDir(src, dest string) error { srcPath := filepath.Clean(src) destPath := filepath.Clean(dest) fi, err := os.Stat(srcPath) if err != nil { return errors.Wrap(err, "getting the file info for the input") } if !fi.IsDir() { return errors.New("source is not a directory") } _, err = os.Stat(dest) if err != nil && !os.IsNotExist(err) { return errors.Wrap(err, "looking up the destination") } err = os.MkdirAll(dest, fi.Mode()) if err != nil { return errors.Wrap(err, "creating destination") } entries, err := os.ReadDir(src) if err != nil { return errors.Wrap(err, "reading the directory listing for the input") } for _, entry := range entries { srcEntryPath := filepath.Join(srcPath, entry.Name()) destEntryPath := filepath.Join(destPath, entry.Name()) if entry.IsDir() { if err = CopyDir(srcEntryPath, destEntryPath); err != nil { return errors.Wrapf(err, "copying %s", entry.Name()) } } else { if err = CopyFile(srcEntryPath, destEntryPath); err != nil { return errors.Wrapf(err, "copying %s", entry.Name()) } } } return nil } // CopyFile copies a file from the src to dest func CopyFile(src, dest string) error { in, err := os.Open(src) if err != nil { return errors.Wrap(err, "opening the input file") } defer in.Close() out, err := os.Create(dest) if err != nil { return errors.Wrap(err, "creating the output file") } if _, err = io.Copy(out, in); err != nil { return errors.Wrap(err, "copying the file content") } if err = out.Sync(); err != nil { return errors.Wrap(err, "flushing the output file to disk") } fi, err := os.Stat(src) if err != nil { return errors.Wrap(err, "getting the file info for the input file") } if err = os.Chmod(dest, fi.Mode()); err != nil { return errors.Wrap(err, "copying permission to the output file") } // Close the output file if err = out.Close(); err != nil { return errors.Wrap(err, "closing the output file") } return nil }