package sftp

// ssh_FXP_ATTRS support
// see https://filezilla-project.org/specs/draft-ietf-secsh-filexfer-02.txt#section-5

import (
	
	
)

const (
	sshFileXferAttrSize        = 0x00000001
	sshFileXferAttrUIDGID      = 0x00000002
	sshFileXferAttrPermissions = 0x00000004
	sshFileXferAttrACmodTime   = 0x00000008
	sshFileXferAttrExtended    = 0x80000000

	sshFileXferAttrAll = sshFileXferAttrSize | sshFileXferAttrUIDGID | sshFileXferAttrPermissions |
		sshFileXferAttrACmodTime | sshFileXferAttrExtended
)

// fileInfo is an artificial type designed to satisfy os.FileInfo.
type fileInfo struct {
	name string
	stat *FileStat
}

// Name returns the base name of the file.
func ( *fileInfo) () string { return .name }

// Size returns the length in bytes for regular files; system-dependent for others.
func ( *fileInfo) () int64 { return int64(.stat.Size) }

// Mode returns file mode bits.
func ( *fileInfo) () os.FileMode { return toFileMode(.stat.Mode) }

// ModTime returns the last modification time of the file.
func ( *fileInfo) () time.Time { return time.Unix(int64(.stat.Mtime), 0) }

// IsDir returns true if the file is a directory.
func ( *fileInfo) () bool { return .Mode().IsDir() }

func ( *fileInfo) () interface{} { return .stat }

// 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( *FileStat,  string) os.FileInfo {
	return &fileInfo{
		name: ,
		stat: ,
	}
}

// FileInfoUidGid extends os.FileInfo and adds callbacks for Uid and Gid retrieval,
// as an alternative to *syscall.Stat_t objects on unix systems.
type FileInfoUidGid interface {
	os.FileInfo
	Uid() uint32
	Gid() uint32
}

// FileInfoUidGid extends os.FileInfo and adds a callbacks for extended data retrieval.
type FileInfoExtendedData interface {
	os.FileInfo
	Extended() []StatExtended
}

func fileStatFromInfo( os.FileInfo) (uint32, *FileStat) {
	 := .ModTime().Unix()
	 := 
	var  uint32 = sshFileXferAttrSize |
		sshFileXferAttrPermissions |
		sshFileXferAttrACmodTime

	 := &FileStat{
		Size:  uint64(.Size()),
		Mode:  fromFileMode(.Mode()),
		Mtime: uint32(),
		Atime: uint32(),
	}

	// os specific file stat decoding
	fileStatFromInfoOs(, &, )

	// The call above will include the sshFileXferAttrUIDGID in case
	// the os.FileInfo can be casted to *syscall.Stat_t on unix.
	// If fi implements FileInfoUidGid, retrieve Uid, Gid from it instead.
	if ,  := .(FileInfoUidGid);  {
		 |= sshFileXferAttrUIDGID
		.UID = .Uid()
		.GID = .Gid()
	}

	// if fi implements FileInfoExtendedData, retrieve extended data from it
	if ,  := .(FileInfoExtendedData);  {
		.Extended = .Extended()
		if len(.Extended) > 0 {
			 |= sshFileXferAttrExtended
		}
	}

	return , 
}