diff options
| author | Joker | 2024-06-15 18:04:03 +0800 | 
|---|---|---|
| committer | rfjakob | 2024-07-27 21:05:00 +0200 | 
| commit | 533c9eb7cedeaaa23d0901d703e1c416b0fd0151 (patch) | |
| tree | dd02308063b67f9d4255353cfbd1e0a84b05d8ff | |
| parent | f06f27e7bc098e334024c365004f9303e79997d9 (diff) | |
cli: deduplicate kernel options
Merge stock kernel options with user-provided ones before passing to go-fuse.
Before: `-ko volname=custom` would result in `-o volname=mountpoint,volname=custom` to macFUSE.
After: `-ko volname=custom` would produce `-o volname=custom` with no duplicates.
Fixes #854 and #557
| -rw-r--r-- | mount.go | 41 | 
1 files changed, 28 insertions, 13 deletions
| @@ -402,12 +402,13 @@ func initGoFuse(rootNode fs.InodeEmbedder, args *argContainer) *fuse.Server {  	}  	mOpts := &fuseOpts.MountOptions +	opts := make(map[string]string)  	if args.allow_other {  		tlog.Info.Printf(tlog.ColorYellow + "The option \"-allow_other\" is set. Make sure the file " +  			"permissions protect your data from unwanted access." + tlog.ColorReset)  		mOpts.AllowOther = true  		// Make the kernel check the file permissions for us -		mOpts.Options = append(mOpts.Options, "default_permissions") +		opts["default_permissions"] = ""  	}  	if args.acl {  		mOpts.EnableAcl = true @@ -415,7 +416,7 @@ func initGoFuse(rootNode fs.InodeEmbedder, args *argContainer) *fuse.Server {  	// fusermount from libfuse 3.x removed the "nonempty" option and exits  	// with an error if it sees it. Only add it to the options on libfuse 2.x.  	if args.nonempty && haveFusermount2() { -		mOpts.Options = append(mOpts.Options, "nonempty") +		opts["nonempty"] = ""  	}  	// Set values shown in "df -T" and friends  	// First column, "Filesystem" @@ -437,40 +438,54 @@ func initGoFuse(rootNode fs.InodeEmbedder, args *argContainer) *fuse.Server {  	// Add a volume name if running osxfuse. Otherwise the Finder will show it as  	// something like "osxfuse Volume 0 (gocryptfs)".  	if runtime.GOOS == "darwin" { -		volname := strings.Replace(path.Base(args.mountpoint), ",", "_", -1) -		mOpts.Options = append(mOpts.Options, "volname="+volname) +		opts["volname"] = strings.Replace(path.Base(args.mountpoint), ",", "_", -1)  	}  	// The kernel enforces read-only operation, we just have to pass "ro".  	// Reverse mounts are always read-only.  	if args.ro || args.reverse { -		mOpts.Options = append(mOpts.Options, "ro") +		opts["ro"] = ""  	} else if args.rw { -		mOpts.Options = append(mOpts.Options, "rw") +		opts["rw"] = ""  	}  	// If both "nosuid" & "suid", "nodev" & "dev", etc were passed, the safer  	// option wins.  	if args.nosuid { -		mOpts.Options = append(mOpts.Options, "nosuid") +		opts["nosuid"] = ""  	} else if args.suid { -		mOpts.Options = append(mOpts.Options, "suid") +		opts["suid"] = ""  	}  	if args.nodev { -		mOpts.Options = append(mOpts.Options, "nodev") +		opts["nodev"] = ""  	} else if args.dev { -		mOpts.Options = append(mOpts.Options, "dev") +		opts["dev"] = ""  	}  	if args.noexec { -		mOpts.Options = append(mOpts.Options, "noexec") +		opts["noexec"] = ""  	} else if args.exec { -		mOpts.Options = append(mOpts.Options, "exec") +		opts["exec"] = ""  	}  	// Add additional mount options (if any) after the stock ones, so the user has  	// a chance to override them.  	if args.ko != "" {  		parts := strings.Split(args.ko, ",")  		tlog.Debug.Printf("Adding -ko mount options: %v", parts) -		mOpts.Options = append(mOpts.Options, parts...) +		for _, part := range parts { +			kv := strings.SplitN(part, "=", 2) +			if len(kv) == 2 { +				opts[kv[0]] = kv[1] +			} else { +				opts[kv[0]] = "" +			} +		} +	} +	for k, v := range opts { +		if v == "" { +			mOpts.Options = append(mOpts.Options, k) +		} else { +			mOpts.Options = append(mOpts.Options, k+"="+v) +		}  	} +  	srv, err := fs.Mount(args.mountpoint, rootNode, fuseOpts)  	if err != nil {  		tlog.Fatal.Printf("fs.Mount failed: %s", strings.TrimSpace(err.Error())) | 
