如何检测已删除的文件?

时间:2018-11-07 06:29:51

标签: go

写入不存在的文件不会在Go中产生错误。

例如,下面是一个示例程序,它以循环方式写入文件:

package main

import (
    "log"
    "os"
    "time"
)

func main() {

    f, err := os.OpenFile("mytest.log", os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
    if err != nil {
        log.Fatal(err)
    }

    for {

        n, err := f.WriteString("blah\n")
        if err != nil {
            log.Fatal(err)
        }
        log.Printf("wrote %d bytes\n", n)
        time.Sleep(2 * time.Second)
    }
}

此程序正在运行时,我从命令行发出rm mytest.log,并观察到该程序在下次调用WriteString()时未产生错误。 (我在Linux上进行了测试,其他操作系统可能有所不同

是否有办法检测文件是否被删除(除了在每次写入之前对文件进行统计)?大概写入的字节只是被操作系统丢弃了?

1 个答案:

答案 0 :(得分:0)

  

此程序正在运行时,我从命令行发出rm mytest.log,观察到该程序在下次调用WriteString()时未产生错误

是的,这正是指定的行为。此外,该文件尚未删除。 rm唯一删除的是文件系统中的特定路径条目。单个文件可以具有多个路径,也称为 hardlinks

只有在通过文件系统条目(链接)或文件描述符(在程序中打开文件)最后一次引用实际文件时,该文件才会被删除。

通过在/dev/shm中创建和打开一个文件,然后删除文件系统条目,Unix文件模型的这种特殊行为被用于实现“未命名”共享内存的时间很长了-因为这种特殊的方式事情引入了竞争条件,对于安全敏感的应用程序,引入了新的系统调用,该系统调用允许创建匿名内存映射,最近,Linux甚至具有在文件系统中创建文件的功能,而无需创建路径条目(open O_TMPFILE标志)。

在较新的Linux版本上,您甚至可以为重新创建/使用linkat系统调用删除了最后一个条目的文件创建文件系统条目。

更新

问题是,如果最后一个文件系统条目消失了,您真的要出错吗?毕竟这不是一个坏情况,您可以安全地读写,没有问题,只是要知道,一旦关闭了文件的最后一个文件描述符,它将丢失。

完全有可能检测是否删除了最后一个文件系统条目,如果这样,则中止文件操作–但是请注意,这样的代码可能会带来很多问题,例如,如果程序希望创建一个新文件,使用linkat将所有内容正确写入文件后,进入文件系统条目。

无论如何,您可以做的是fstat-打开文件(在Go中为file.Stat),然后查看文件具有的硬链接数。如果该数字降至零,则所有文件系统条目均消失。实际上,在Go中获取该数字有些棘手,这里Counting hard links to a file in Go

进行了描述
package main

import (
    "fmt"
    "log"
    "os"
    "syscall"
    "time"
)

func main() {
    fmt.Println("Test Operation")
    f, err := os.OpenFile("test.txt", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0644)
    if err != nil {
        log.Fatal(err)
    }

    for {

        n, err := f.WriteString("blah\n")
        if err != nil {
            log.Fatal(err)
        }
        log.Printf("wrote %d bytes\n", n)
        time.Sleep(2 * time.Second)
        stat, err := f.Stat()
        if err != nil{
            log.Fatal(err)
        }
        if sys := stat.Sys(); sys != nil {
            if stat, ok := sys.(*syscall.Stat_t); ok {
                nlink := uint64(stat.Nlink)
                if 0 == nlink {
                    log.Printf("All filesystem entries to original file removed, exiting")
                    break
                }
            }
        }
    }
}