You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							262 lines
						
					
					
						
							6.6 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							262 lines
						
					
					
						
							6.6 KiB
						
					
					
				| package mount | |
| 
 | |
| import ( | |
| 	"github.com/hanwen/go-fuse/v2/fuse" | |
| 	"github.com/seaweedfs/seaweedfs/weed/filer" | |
| 	"github.com/seaweedfs/seaweedfs/weed/glog" | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb" | |
| 	"os" | |
| 	"syscall" | |
| 	"time" | |
| ) | |
| 
 | |
| func (wfs *WFS) GetAttr(cancel <-chan struct{}, input *fuse.GetAttrIn, out *fuse.AttrOut) (code fuse.Status) { | |
| 	if input.NodeId == 1 { | |
| 		wfs.setRootAttr(out) | |
| 		return fuse.OK | |
| 	} | |
| 
 | |
| 	inode := input.NodeId | |
| 	_, _, entry, status := wfs.maybeReadEntry(inode) | |
| 	if status == fuse.OK { | |
| 		out.AttrValid = 1 | |
| 		wfs.setAttrByPbEntry(&out.Attr, inode, entry, true) | |
| 		return status | |
| 	} else { | |
| 		if fh, found := wfs.fhmap.FindFileHandle(inode); found { | |
| 			out.AttrValid = 1 | |
| 			wfs.setAttrByPbEntry(&out.Attr, inode, fh.entry.GetEntry(), true) | |
| 			out.Nlink = 0 | |
| 			return fuse.OK | |
| 		} | |
| 	} | |
| 
 | |
| 	return status | |
| } | |
| 
 | |
| func (wfs *WFS) SetAttr(cancel <-chan struct{}, input *fuse.SetAttrIn, out *fuse.AttrOut) (code fuse.Status) { | |
| 
 | |
| 	if wfs.IsOverQuota { | |
| 		return fuse.Status(syscall.ENOSPC) | |
| 	} | |
| 
 | |
| 	path, fh, entry, status := wfs.maybeReadEntry(input.NodeId) | |
| 	if status != fuse.OK { | |
| 		return status | |
| 	} | |
| 	if fh != nil { | |
| 		fh.entryLock.Lock() | |
| 		defer fh.entryLock.Unlock() | |
| 	} | |
| 
 | |
| 	if size, ok := input.GetSize(); ok && entry != nil { | |
| 		glog.V(4).Infof("%v setattr set size=%v chunks=%d", path, size, len(entry.GetChunks())) | |
| 		if size < filer.FileSize(entry) { | |
| 			// fmt.Printf("truncate %v \n", fullPath) | |
| 			var chunks []*filer_pb.FileChunk | |
| 			var truncatedChunks []*filer_pb.FileChunk | |
| 			for _, chunk := range entry.GetChunks() { | |
| 				int64Size := int64(chunk.Size) | |
| 				if chunk.Offset+int64Size > int64(size) { | |
| 					// this chunk is truncated | |
| 					int64Size = int64(size) - chunk.Offset | |
| 					if int64Size > 0 { | |
| 						chunks = append(chunks, chunk) | |
| 						glog.V(4).Infof("truncated chunk %+v from %d to %d\n", chunk.GetFileIdString(), chunk.Size, int64Size) | |
| 						chunk.Size = uint64(int64Size) | |
| 					} else { | |
| 						glog.V(4).Infof("truncated whole chunk %+v\n", chunk.GetFileIdString()) | |
| 						truncatedChunks = append(truncatedChunks, chunk) | |
| 					} | |
| 				} else { | |
| 					chunks = append(chunks, chunk) | |
| 				} | |
| 			} | |
| 			// set the new chunks and reset entry cache | |
| 			entry.Chunks = chunks | |
| 			if fh != nil { | |
| 				fh.entryChunkGroup.SetChunks(chunks) | |
| 			} | |
| 		} | |
| 		entry.Attributes.Mtime = time.Now().Unix() | |
| 		entry.Attributes.FileSize = size | |
| 
 | |
| 	} | |
| 
 | |
| 	if mode, ok := input.GetMode(); ok { | |
| 		// glog.V(4).Infof("setAttr mode %o", mode) | |
| 		entry.Attributes.FileMode = chmod(entry.Attributes.FileMode, mode) | |
| 		if input.NodeId == 1 { | |
| 			wfs.option.MountMode = os.FileMode(chmod(uint32(wfs.option.MountMode), mode)) | |
| 		} | |
| 	} | |
| 
 | |
| 	if uid, ok := input.GetUID(); ok { | |
| 		entry.Attributes.Uid = uid | |
| 		if input.NodeId == 1 { | |
| 			wfs.option.MountUid = uid | |
| 		} | |
| 	} | |
| 
 | |
| 	if gid, ok := input.GetGID(); ok { | |
| 		entry.Attributes.Gid = gid | |
| 		if input.NodeId == 1 { | |
| 			wfs.option.MountGid = gid | |
| 		} | |
| 	} | |
| 
 | |
| 	if atime, ok := input.GetATime(); ok { | |
| 		entry.Attributes.Mtime = atime.Unix() | |
| 	} | |
| 
 | |
| 	if mtime, ok := input.GetMTime(); ok { | |
| 		entry.Attributes.Mtime = mtime.Unix() | |
| 	} | |
| 
 | |
| 	out.AttrValid = 1 | |
| 	size, includeSize := input.GetSize() | |
| 	if includeSize { | |
| 		out.Attr.Size = size | |
| 	} | |
| 	wfs.setAttrByPbEntry(&out.Attr, input.NodeId, entry, !includeSize) | |
| 
 | |
| 	if fh != nil { | |
| 		fh.dirtyMetadata = true | |
| 		return fuse.OK | |
| 	} | |
| 
 | |
| 	return wfs.saveEntry(path, entry) | |
| 
 | |
| } | |
| 
 | |
| func (wfs *WFS) setRootAttr(out *fuse.AttrOut) { | |
| 	now := uint64(time.Now().Unix()) | |
| 	out.AttrValid = 119 | |
| 	out.Ino = 1 | |
| 	setBlksize(&out.Attr, blockSize) | |
| 	out.Uid = wfs.option.MountUid | |
| 	out.Gid = wfs.option.MountGid | |
| 	out.Mtime = now | |
| 	out.Ctime = now | |
| 	out.Atime = now | |
| 	out.Mode = toSyscallType(os.ModeDir) | uint32(wfs.option.MountMode) | |
| 	out.Nlink = 1 | |
| } | |
| 
 | |
| func (wfs *WFS) setAttrByPbEntry(out *fuse.Attr, inode uint64, entry *filer_pb.Entry, calculateSize bool) { | |
| 	out.Ino = inode | |
| 	out.Blocks = (out.Size + blockSize - 1) / blockSize | |
| 	setBlksize(out, blockSize) | |
| 	if entry == nil { | |
| 		return | |
| 	} | |
| 	if entry.Attributes != nil && entry.Attributes.Inode != 0 { | |
| 		out.Ino = entry.Attributes.Inode | |
| 	} | |
| 	if calculateSize { | |
| 		out.Size = filer.FileSize(entry) | |
| 	} | |
| 	if entry.FileMode()&os.ModeSymlink != 0 { | |
| 		out.Size = uint64(len(entry.Attributes.SymlinkTarget)) | |
| 	} | |
| 	out.Mtime = uint64(entry.Attributes.Mtime) | |
| 	out.Ctime = uint64(entry.Attributes.Mtime) | |
| 	out.Atime = uint64(entry.Attributes.Mtime) | |
| 	out.Mode = toSyscallMode(os.FileMode(entry.Attributes.FileMode)) | |
| 	if entry.HardLinkCounter > 0 { | |
| 		out.Nlink = uint32(entry.HardLinkCounter) | |
| 	} else { | |
| 		out.Nlink = 1 | |
| 	} | |
| 	out.Uid = entry.Attributes.Uid | |
| 	out.Gid = entry.Attributes.Gid | |
| 	out.Rdev = entry.Attributes.Rdev | |
| } | |
| 
 | |
| func (wfs *WFS) setAttrByFilerEntry(out *fuse.Attr, inode uint64, entry *filer.Entry) { | |
| 	out.Ino = inode | |
| 	out.Size = entry.FileSize | |
| 	if entry.Mode&os.ModeSymlink != 0 { | |
| 		out.Size = uint64(len(entry.SymlinkTarget)) | |
| 	} | |
| 	out.Blocks = (out.Size + blockSize - 1) / blockSize | |
| 	setBlksize(out, blockSize) | |
| 	out.Atime = uint64(entry.Attr.Mtime.Unix()) | |
| 	out.Mtime = uint64(entry.Attr.Mtime.Unix()) | |
| 	out.Ctime = uint64(entry.Attr.Mtime.Unix()) | |
| 	out.Mode = toSyscallMode(entry.Attr.Mode) | |
| 	if entry.HardLinkCounter > 0 { | |
| 		out.Nlink = uint32(entry.HardLinkCounter) | |
| 	} else { | |
| 		out.Nlink = 1 | |
| 	} | |
| 	out.Uid = entry.Attr.Uid | |
| 	out.Gid = entry.Attr.Gid | |
| 	out.Rdev = entry.Attr.Rdev | |
| } | |
| 
 | |
| func (wfs *WFS) outputPbEntry(out *fuse.EntryOut, inode uint64, entry *filer_pb.Entry) { | |
| 	out.NodeId = inode | |
| 	out.Generation = 1 | |
| 	out.EntryValid = 1 | |
| 	out.AttrValid = 1 | |
| 	wfs.setAttrByPbEntry(&out.Attr, inode, entry, true) | |
| } | |
| 
 | |
| func (wfs *WFS) outputFilerEntry(out *fuse.EntryOut, inode uint64, entry *filer.Entry) { | |
| 	out.NodeId = inode | |
| 	out.Generation = 1 | |
| 	out.EntryValid = 1 | |
| 	out.AttrValid = 1 | |
| 	wfs.setAttrByFilerEntry(&out.Attr, inode, entry) | |
| } | |
| 
 | |
| func chmod(existing uint32, mode uint32) uint32 { | |
| 	return existing&^07777 | mode&07777 | |
| } | |
| 
 | |
| func toSyscallMode(mode os.FileMode) uint32 { | |
| 	return toSyscallType(mode) | uint32(mode) | |
| } | |
| 
 | |
| func toSyscallType(mode os.FileMode) uint32 { | |
| 	switch mode & os.ModeType { | |
| 	case os.ModeDir: | |
| 		return syscall.S_IFDIR | |
| 	case os.ModeSymlink: | |
| 		return syscall.S_IFLNK | |
| 	case os.ModeNamedPipe: | |
| 		return syscall.S_IFIFO | |
| 	case os.ModeSocket: | |
| 		return syscall.S_IFSOCK | |
| 	case os.ModeDevice: | |
| 		return syscall.S_IFBLK | |
| 	case os.ModeCharDevice: | |
| 		return syscall.S_IFCHR | |
| 	default: | |
| 		return syscall.S_IFREG | |
| 	} | |
| } | |
| 
 | |
| func toOsFileType(mode uint32) os.FileMode { | |
| 	switch mode & (syscall.S_IFMT & 0xffff) { | |
| 	case syscall.S_IFDIR: | |
| 		return os.ModeDir | |
| 	case syscall.S_IFLNK: | |
| 		return os.ModeSymlink | |
| 	case syscall.S_IFIFO: | |
| 		return os.ModeNamedPipe | |
| 	case syscall.S_IFSOCK: | |
| 		return os.ModeSocket | |
| 	case syscall.S_IFBLK: | |
| 		return os.ModeDevice | |
| 	case syscall.S_IFCHR: | |
| 		return os.ModeCharDevice | |
| 	default: | |
| 		return 0 | |
| 	} | |
| } | |
| 
 | |
| func toOsFileMode(mode uint32) os.FileMode { | |
| 	return toOsFileType(mode) | os.FileMode(mode&07777) | |
| }
 |