| package sftp |
| |
| // ssh_FXP_ATTRS support |
| // see http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-5 |
| |
| import ( |
| "os" |
| "time" |
| ) |
| |
| const ( |
| sshFileXferAttrSize = 0x00000001 |
| sshFileXferAttrUIDGID = 0x00000002 |
| sshFileXferAttrPermissions = 0x00000004 |
| sshFileXferAttrACmodTime = 0x00000008 |
| sshFileXferAttrExtented = 0x80000000 |
| |
| sshFileXferAttrAll = sshFileXferAttrSize | sshFileXferAttrUIDGID | sshFileXferAttrPermissions | |
| sshFileXferAttrACmodTime | sshFileXferAttrExtented |
| ) |
| |
| // fileInfo is an artificial type designed to satisfy os.FileInfo. |
| type fileInfo struct { |
| name string |
| size int64 |
| mode os.FileMode |
| mtime time.Time |
| sys interface{} |
| } |
| |
| // Name returns the base name of the file. |
| func (fi *fileInfo) Name() string { return fi.name } |
| |
| // Size returns the length in bytes for regular files; system-dependent for others. |
| func (fi *fileInfo) Size() int64 { return fi.size } |
| |
| // Mode returns file mode bits. |
| func (fi *fileInfo) Mode() os.FileMode { return fi.mode } |
| |
| // ModTime returns the last modification time of the file. |
| func (fi *fileInfo) ModTime() time.Time { return fi.mtime } |
| |
| // IsDir returns true if the file is a directory. |
| func (fi *fileInfo) IsDir() bool { return fi.Mode().IsDir() } |
| |
| func (fi *fileInfo) Sys() interface{} { return fi.sys } |
| |
| // FileStat holds the original unmarshalled values from a call to READDIR or |
| // *STAT. It is exported for the purposes of accessing the raw values via |
| // os.FileInfo.Sys(). It is also used server side to store the unmarshalled |
| // values for SetStat. |
| type FileStat struct { |
| Size uint64 |
| Mode uint32 |
| Mtime uint32 |
| Atime uint32 |
| UID uint32 |
| GID uint32 |
| Extended []StatExtended |
| } |
| |
| // StatExtended contains additional, extended information for a FileStat. |
| type StatExtended struct { |
| ExtType string |
| ExtData string |
| } |
| |
| func fileInfoFromStat(st *FileStat, name string) os.FileInfo { |
| fs := &fileInfo{ |
| name: name, |
| size: int64(st.Size), |
| mode: toFileMode(st.Mode), |
| mtime: time.Unix(int64(st.Mtime), 0), |
| sys: st, |
| } |
| return fs |
| } |
| |
| func fileStatFromInfo(fi os.FileInfo) (uint32, FileStat) { |
| mtime := fi.ModTime().Unix() |
| atime := mtime |
| var flags uint32 = sshFileXferAttrSize | |
| sshFileXferAttrPermissions | |
| sshFileXferAttrACmodTime |
| |
| fileStat := FileStat{ |
| Size: uint64(fi.Size()), |
| Mode: fromFileMode(fi.Mode()), |
| Mtime: uint32(mtime), |
| Atime: uint32(atime), |
| } |
| |
| // os specific file stat decoding |
| fileStatFromInfoOs(fi, &flags, &fileStat) |
| |
| return flags, fileStat |
| } |
| |
| func unmarshalAttrs(b []byte) (*FileStat, []byte) { |
| flags, b := unmarshalUint32(b) |
| return getFileStat(flags, b) |
| } |
| |
| func getFileStat(flags uint32, b []byte) (*FileStat, []byte) { |
| var fs FileStat |
| if flags&sshFileXferAttrSize == sshFileXferAttrSize { |
| fs.Size, b, _ = unmarshalUint64Safe(b) |
| } |
| if flags&sshFileXferAttrUIDGID == sshFileXferAttrUIDGID { |
| fs.UID, b, _ = unmarshalUint32Safe(b) |
| } |
| if flags&sshFileXferAttrUIDGID == sshFileXferAttrUIDGID { |
| fs.GID, b, _ = unmarshalUint32Safe(b) |
| } |
| if flags&sshFileXferAttrPermissions == sshFileXferAttrPermissions { |
| fs.Mode, b, _ = unmarshalUint32Safe(b) |
| } |
| if flags&sshFileXferAttrACmodTime == sshFileXferAttrACmodTime { |
| fs.Atime, b, _ = unmarshalUint32Safe(b) |
| fs.Mtime, b, _ = unmarshalUint32Safe(b) |
| } |
| if flags&sshFileXferAttrExtented == sshFileXferAttrExtented { |
| var count uint32 |
| count, b, _ = unmarshalUint32Safe(b) |
| ext := make([]StatExtended, count) |
| for i := uint32(0); i < count; i++ { |
| var typ string |
| var data string |
| typ, b, _ = unmarshalStringSafe(b) |
| data, b, _ = unmarshalStringSafe(b) |
| ext[i] = StatExtended{typ, data} |
| } |
| fs.Extended = ext |
| } |
| return &fs, b |
| } |
| |
| func marshalFileInfo(b []byte, fi os.FileInfo) []byte { |
| // attributes variable struct, and also variable per protocol version |
| // spec version 3 attributes: |
| // uint32 flags |
| // uint64 size present only if flag SSH_FILEXFER_ATTR_SIZE |
| // uint32 uid present only if flag SSH_FILEXFER_ATTR_UIDGID |
| // uint32 gid present only if flag SSH_FILEXFER_ATTR_UIDGID |
| // uint32 permissions present only if flag SSH_FILEXFER_ATTR_PERMISSIONS |
| // uint32 atime present only if flag SSH_FILEXFER_ACMODTIME |
| // uint32 mtime present only if flag SSH_FILEXFER_ACMODTIME |
| // uint32 extended_count present only if flag SSH_FILEXFER_ATTR_EXTENDED |
| // string extended_type |
| // string extended_data |
| // ... more extended data (extended_type - extended_data pairs), |
| // so that number of pairs equals extended_count |
| |
| flags, fileStat := fileStatFromInfo(fi) |
| |
| b = marshalUint32(b, flags) |
| if flags&sshFileXferAttrSize != 0 { |
| b = marshalUint64(b, fileStat.Size) |
| } |
| if flags&sshFileXferAttrUIDGID != 0 { |
| b = marshalUint32(b, fileStat.UID) |
| b = marshalUint32(b, fileStat.GID) |
| } |
| if flags&sshFileXferAttrPermissions != 0 { |
| b = marshalUint32(b, fileStat.Mode) |
| } |
| if flags&sshFileXferAttrACmodTime != 0 { |
| b = marshalUint32(b, fileStat.Atime) |
| b = marshalUint32(b, fileStat.Mtime) |
| } |
| |
| return b |
| } |