Golang правильно создает экземпляр ОС.Содержит filemode

Я видел бесчисленные примеры и учебники, которые показывают, как создать файл, и все они "обманывают", просто установив биты разрешений файла. Я хотел бы знать / узнать, как правильно создать ОС.FileMode для предоставления записи во время создания / обновления файла.

грубый пример ниже:

func FileWrite(path string, r io.Reader, uid, gid int, perms string) (int64, error){
    w, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0664)
    if err != nil {
        if path == "" {
            w = os.Stdout
        } else {
            return 0, err
        }
    }
    defer w.Close()

    size, err := io.Copy(w, r)

    if err != nil {
        return 0, err
    }
    return size, err
}

в основной функции выше бит разрешения 0664 установлен, и хотя это может иметь смысл, иногда я предпочитаю иметь правильный способ правильно установить filemode. Как видно выше, общим примером может быть то, что UID / GID известен и уже предоставлен как значения int, а perms-восьмеричные цифры, которые ранее были собраны и вставлены в БД в виде строки.

2 ответов


FileMode - это просто uint32. http://golang.org/pkg/os/#FileMode

настройка через константы не "обманывает", вы используете его как другие числовые значения. Если константа не используется, можно использовать преобразование для допустимых числовых значений:

mode := int(0777)
os.FileMode(mode)

мое исправление заключалось в определении моих собственных констант, поскольку я не мог найти их в os или syscall:

    const (
        OS_READ = 04
        OS_WRITE = 02
        OS_EX = 01
        OS_USER_SHIFT = 6
        OS_GROUP_SHIFT = 3
        OS_OTH_SHIFT = 0

        OS_USER_R = OS_READ<<OS_USER_SHIFT
        OS_USER_W = OS_WRITE<<OS_USER_SHIFT
        OS_USER_X = OS_EX<<OS_USER_SHIFT
        OS_USER_RW = OS_USER_R | OS_USER_W
        OS_USER_RWX = OS_USER_RW | OS_USER_X

        OS_GROUP_R = OS_READ<<OS_GROUP_SHIFT
        OS_GROUP_W = OS_WRITE<<OS_GROUP_SHIFT
        OS_GROUP_X = OS_EX<<OS_GROUP_SHIFT
        OS_GROUP_RW = OS_GROUP_R | OS_GROUP_W
        OS_GROUP_RWX = OS_GROUP_RW | OS_GROUP_X

        OS_OTH_R = OS_READ<<OS_OTH_SHIFT
        OS_OTH_W = OS_WRITE<<OS_OTH_SHIFT
        OS_OTH_X = OS_EX<<OS_OTH_SHIFT
        OS_OTH_RW = OS_OTH_R | OS_OTH_W
        OS_OTH_RWX = OS_OTH_RW | OS_OTH_X

        OS_ALL_R = OS_USER_R | OS_GROUP_R | OS_OTH_R
        OS_ALL_W = OS_USER_W | OS_GROUP_W | OS_OTH_W
        OS_ALL_X = OS_USER_X | OS_GROUP_X | OS_OTH_X
        OS_ALL_RW = OS_ALL_R | OS_ALL_W
        OS_ALL_RWX = OS_ALL_RW | OS_GROUP_X
)

это позволяет мне указать мое намерение напрямую:

        // Create any directories needed to put this file in them
        var dir_file_mode os.FileMode
        dir_file_mode = os.ModeDir | (OS_USER_RWX | OS_ALL_R)
        os.MkdirAll(dir_str, dir_file_mode)

Я уверен, что это может быть улучшено с помощью iota и некоторых других комбинаций разрешений, но пока это работает для меня.