40

Sometimes people delete files they shouldn't, a long-running process still has the file open, and recovering the data by catting /proc/<pid>/fd/N just isn't awesome enough. Awesome enough would be if you could "undo" the delete by running some magic option to ln that would let you re-link to the inode number (recovered through lsof).

I can't find any Linux tools to do this, least with cursory Googling.

What do you got, serverfault?

EDIT1: The reason catting the file from /proc/<pid>/fd/N isn't awesome enough is because the process which still has the file open is still writing to it. A delete removes the reference to the inode from the filesystem namespace. What I want is a way of re-creating the reference.

EDIT2: 'debugfs ln' works but the risk is too high since it frobs raw filesystem data. The recovered file is also crazy inconsistent. The link count is zero and I can't add links to it. I'm worse off this way since I can just use /proc/<pid>/fd/N to access the data without corrupting my fs.

mbac32768
  • 848
  • 1
  • 7
  • 13

8 Answers8

19

Awesome enough would be if you could "undo" the delete by running some magic option to ln that would let you re-link to the inode number (recovered through lsof).

This awesomeness was introduced to ln in v8.0 (GNU/coreutils) with the -L|--logical option which causes ln to dereference a /proc/<pid>/fd/<handle> first. So a simple

ln -L /proc/<pid>/fd/<handle> /path/to/deleted/file

is enough to relink a deleted file.

tnimeu
  • 191
  • 1
  • 2
  • 12
    This doesn't work; if the file is deleted it will fail. – Random832 Jun 20 '11 at 22:47
  • 2
    No, it won't. I regularly use this to restore deleted but still opened files. But you need to make sure, that the new `/path/to/deleted/file` is on the **same filesystem** as the file was just before it was deleted, else this will - indeed - fail. (You can get the old path with `ls -l /proc//fd/`) – tnimeu Jul 28 '11 at 20:39
  • 4
    This sort of functionality (See [this question](http://stackoverflow.com/questions/4171713/relinking-an-anonymous-unlinked-but-open-file) and answer) was specifically _rejected_ as a security risk [to a hypothetical security scheme revolving around a privileged process giving your process a read-only file handle to a file you own but do not otherwise have access to]; I tried it (though with a small C program to directly use the relevant system call) and it did not work. – Random832 Aug 24 '11 at 19:28
  • 10
    I, of course, did test this before posting my solution and at that time it actually worked for me. What I wasn't aware of is that it only worked on `tmpfs` filesystems but not on e.g. `ext3`. Furthermore this _feature_ got completely disabled in 2.6.39, see [the commit](http://git.kernel.org/?p=linux/kernel/git/torvalds/linux.git;a=commitdiff;h=aae8a97d3ec30788790d1720b71d76fd8eb44b73). So therefore this solution won't work with kernel 2.6.39 or newer anymore and in earlier versions it depends on the filesystem. – tnimeu Jan 26 '12 at 14:04
  • 9
    @tnimeu `ln -L` does not work for me. I have a deleted file and I tried to relink it to the original path. `ln` gives me a `ln: failed to create hard link /my/path/file.pdf => /proc/19674/fd/16: No such file or directory`. But I can e.g. successfully `cat /proc/19674/fd/16` – Evgeniy Berezovsky Aug 21 '13 at 06:46
  • 3
    Don't bother with `ln -L`, it's been **[disabled](https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=aae8a97d3ec30788790d1720b71d76fd8eb44b73)** in the Linux kernel since 2011 due to security concerns. – rustyx Apr 29 '20 at 21:36
14

It sounds like you already understand a lot, so I won't go into excess detail. There's several methods to find the inode and you can usually cat and redirect STDOUT. You can use debugfs. Run this command within:

ln <$INODE> FILENAME

Make sure you have backups of the filesystem. You'll probably need to run a fsck afterwards. I tested this successfully with an inode still being written to and it does work to create a new hard link to a dereferenced inode.

If the file is unlinked with an unopen file in ext3, the data is lost. I'm not sure how consistently true this is but most of my data recovery experience is with ext2. From the ext3 FAQ:

Q: How can I recover (undelete) deleted files from my ext3 partition? Actually, you can't! This is what one of the developers, Andreas Dilger, said about it:

In order to ensure that ext3 can safely resume an unlink after a crash, it actually zeros out the block pointers in the inode, whereas ext2 just marks these blocks as unused in the block bitmaps and marks the inode as "deleted" and leaves the block pointers alone.

Your only hope is to "grep" for parts of your files that have been deleted and hope for the best.

There's also relevant information in this question:

I overwrote a large file with a blank one on a linux server. Can I recover the existing file?

Warner
  • 23,440
  • 2
  • 57
  • 69
  • Comment hopefully deleted as it was not revelent. – mdpc Aug 10 '10 at 16:39
  • 1
    In the case of a deleted but still open file I don't think it would zero out the pointers in the inode. Also, instead of using "ln" in debugfs I'd use "undel" so that the inode reference counts will correctly get updated. – Mark Wagner Aug 10 '10 at 19:13
  • I didn't mean to allude as such, embobo. It doesn't, I tested the performance. I've clarified my language. – Warner Aug 10 '10 at 19:21
  • Clever, but corrupts my filesystem. :) – mbac32768 Aug 11 '10 at 17:32
  • It's the only solution for the scenario as you describe it. Manipulating a filesystem on a low level that is mounted rw and actively being written to is likely to cause corruption in almost all scenarios. – Warner Aug 11 '10 at 17:44
  • **It's dangerous**. The new created inode file can't be link by ln. so kernel / fs still think inode is deleted, ( then something maybe went wrong step by step, I guess, I'm not sure, I didn't read the code ) – yurenchen Mar 18 '20 at 14:36
9

the debugfs way as you saw doesn't really work and at best your file will be deleted automatically (due to the journal) after reboot and at worst you can trash your filesystem resulting to "reboot cycle of death". The Right Solution (TM) is to perform the undelete in the VFS level (which also has the added benefit of working with practically all current Linux filesystems). The system call way (flink) has been shot down every time it appeared in LKML so the best way is through a module + ioctl.

A project that implements this approach and has reasonably small and clean code is fdlink (https://github.com/pkt/fdlink.git for a version tested with ubuntu maverick's kernel). With it, after you insert the module (sudo insmod flink_dev.ko) you can just do "./flinkapp /proc//fd/X /my/link/path" and it will do exactly what you want.

You can also use a forward-ported version of vfs-undelete.sourceforge.net that also works (and can also automatically relink to the original name), but fdlink's code is simpler and it works just as well, so it is my preference.

pktoss
  • 91
  • 1
  • 1
6

Ran into the same problem today. The best I could come up with is to run

% tail -n +0 -f /proc/<pid>/fd/<fd> /path/to/deleted_file

in a tmux/screen session until the process ends.

nickray
  • 61
  • 1
  • 1
4

I don't know how to do exactly what you want, but what I would do is:

  • Open the file RO from another process
  • Wait for the original process to exit
  • Copy the data from your open FD to a file

Not ideal, obviously, but possible. The other option is to play around with debugfs (using the link command), but that's kind of scary on a production machine!

Bill Weiss
  • 10,782
  • 3
  • 37
  • 65
2

Interesting question. An interviewer asked the same question to me in a job interview. What I told him was that there was not a easy way to do this and in general was not worth the time and effort involved. I did ask him what he thought the solution to this issue was ....

  1. Use lsof to find the inode number on the disk for the process as it will still appear even if the file has been deleted...the key is that it is still open.
  2. Extract the information from the filesystem based on this via a filesystem debugger.
mdpc
  • 11,698
  • 28
  • 51
  • 65
2

Use Sleuthkit icat.

sudo icat /dev/rdisk1s2 5484287 > accidentally_deleted_open_file
Isaac
  • 29
  • 1
  • 2
    This works by bypassing the operating system's filesystem functionality and parsing the disk bytes directly. – Flimm Jul 03 '14 at 12:24
2

The quick solution that worked for me, without intimidating tools:

1) find the process+fd by looking directly in /proc:

ls -al /proc/*/fd/* 2>/dev/null | grep {filename}

2) Then a similar technique to @nickray's, with pv thrown in:

tail -c +0 -f /proc/{procnum}/fd/{fdnum} | pv -s {expectedsize} > {recovery_filename}

You may need to Ctrl-C when done (ls /proc/{procnum}/fd/{fdnum} will tell you that the file no longer exists)), but if you know the exact size in bytes, you can use pv -S to make it exit when the count is reached.

xenoid
  • 233
  • 1
  • 2
  • 10