为什么 rename() 在将文件成功移动到 NFS 挂载磁盘后仍返回 false


Why does rename() return false despite moving a file successully to an NFS mounted disk?

将文件从本地磁盘移动到 NFS 挂载磁盘时收到Invalid argument警告。尽管出现错误消息,文件仍已成功移动

Warning: rename(/tmp/image.jpg,/mnt/remote.server-disk1/image.jpg): Invalid argument

装载的磁盘:

$ df
remote.server:/disk1 917G  269M  871G   1% /mnt/remote.server-disk1

远程服务器上导出的磁盘:

$ cat /etc/exports
/disk1 remote.server(rw,sync,root_squash,secure,crossmnt,anonuid=504,anongid=504)

重命名() 之前的本地磁盘上的文件:

$ stat /tmp/image.jpg
File: `image.jpg'
Size: 2105          Blocks: 8          IO Block: 4096   regular file
Device: 803h/2051d  Inode: 33556339    Links: 1
Access: (0777/-rwxrwxrwx)  Uid: (  501/  apache)   Gid: (  501/  apache)
...

重命名() 后远程磁盘上的文件:

$ stat /disk1/image.jpg
File: `image.jpg'
Size: 2105          Blocks: 8          IO Block: 4096   regular file
Device: 821h/2081d  Inode: 34603214    Links: 1
Access: (0777/-rwxrwxrwx)  Uid: (  501/  apache)   Gid: (  501/  apache)
...

有什么想法吗?谢谢

Unix 您无法重命名或在文件系统之间移动,而是必须将文件从一个源位置复制到目标位置,然后删除源。 这将解释您收到的错误消息。 然而,似乎不清楚为什么它仍然这样做。 这可能与权限或 NFS 挂载的磁盘在本地缓存有关。

也许有点晚了,但我认为答案可能是因为链接到目标文件系统的权限。

我们

遇到了同样的问题,跟踪为我们提供了正确的诊断:

strace php -r "rename('SOURCE_FILE', 'DST_FILE');";
...
chown("SOURCE_FILE", 0, 0) = -1 EINVAL (Invalid argument)                                                                                                                                                       
write(2, "PHP Warning:  rename(SOURCE_FILE"..., 192PHP Warning:  rename(SOURCE_FILE): Invalid argument in Command line code on line 1
) = 192
write(1, "bool(false)'n", 12bool(false)

在我们的例子中,目标文件系统是版本为 4 的 NFS,并且启用了 idmap。

即使是简单的

chown $(whoami) $DST_FILE

不工作。

这是默认行为(至少在 debian 下)使用 nfs-common utils 启用 idmap。因此,即使您使用复制 + 取消链接(顺便说一下,这是最好的方法)修复它,它仍然可能隐藏底层文件系统中的问题。

(要禁用 idmap:https://forums.aws.amazon.com/thread.jspa?threadID=235501)

也许是因为你这样做没有引号

rename(/tmp/image.jpg,/mnt/remote.server-disk1/image.jpg);

尝试添加引号

rename('/tmp/image.jpg', '/mnt/remote.server-disk1/image.jpg');

我无法解决这个问题,但是 copy() 然后 unlink() 工作没有错误。