Last active
October 29, 2024 18:08
-
-
Save vjt/5183305 to your computer and use it in GitHub Desktop.
Copy data from a Time Machine volume mounted on a Linux box.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/bin/bash | |
# | |
# Copy data from a Time Machine volume mounted on a Linux box. | |
# | |
# Usage: copy-from-time-machine.sh <source> <target> | |
# | |
# source: the source directory inside a time machine backup | |
# target: the target directory in which to copy the reconstructed | |
# directory trees. Created if it does not exists. | |
# | |
# Details: | |
# | |
# Time machine implements directory hard links by creating an | |
# empty file in place of the directory and storing in its | |
# "number of hard links" metadata attribute a pointer to a | |
# real directory in "/.HFS Private directory data^M" named | |
# "dir_$number". | |
# | |
# This script reconstructs a plain directory tree from this | |
# really ugly apple hack. Tested on a 650GB backup from OSX | |
# 10.6 mounted on a Linux 3.2.0-38 Ubuntu box. YMMV. | |
# | |
# MIT License. | |
# | |
# - [email protected] | |
# | |
self="$0" | |
source="$1" | |
target="$2" | |
hfsd="$3" | |
set -e | |
if [ -z "$source" -o -z "$target" ]; then | |
echo "Usage: $self <source> <target>" | |
exit -1 | |
fi | |
if [ ! -d "$target" ]; then | |
mkdir -p "$target" | |
fi | |
if [ -z "$hfsd" ]; then | |
# Look for HFS Private directory data | |
sysname="$(echo -ne '.HFS+ Private Directory Data\r')" | |
hfsd=$source | |
while [ "$hfsd" != "/" -a ! -d "$hfsd/$sysname" ]; do | |
hfsd=`dirname "$hfsd"`; | |
done | |
if [ "$hfsd" = '/' ]; then | |
echo "HFS Private Directory Data not found in $source, is it an HFS filesystem?" | |
exit -2 | |
else | |
echo "HFS Private Directory Data found in '$hfsd'" | |
hfsd="$hfsd/$sysname" | |
fi | |
fi | |
find "$source" -mindepth 1 -maxdepth 1 -and -not -name . -and -not -name .. | while read entry; do | |
dest="$target/`basename "$entry"`" | |
read hlnum type <<<$(stat -c '%h %F' "$entry") | |
case $type in | |
'regular file'|'symbolic link') | |
cp -van "$entry" "$dest" | |
;; | |
'directory') | |
# Recurse | |
$self "$entry" "$dest" "$hfsd" | |
;; | |
'regular empty file') | |
if [ -d "$hfsd/dir_$hlnum" ]; then | |
# Recurse | |
$self "$hfsd/dir_$hlnum" "$dest" "$hfsd" | |
else | |
echo "Skipping empty file $entry" | |
fi | |
;; | |
esac | |
done |
damn, this script keeps on giving! Life-saving 10 years after its release — thank you @vjt !
damn, this script keeps on giving! Life-saving 10 years after its release — thank you @vjt !
glad it’s useful! enjoy
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
@stanfrbd haha I am super glad to hear that! that’s what OSS is for :-) and luckily these internals rarely change over time :D cheers!