/my/dir
같은 파티션에 있는지 어떻게 확인할 수 /
있습니까?
이것은 스크립트 내 통합을위한 것입니다. 바인드 마운트는 올바르게 처리해야합니다. POSIX 호환 솔루션을 환영합니다.
/my/dir
같은 파티션에 있는지 어떻게 확인할 수 /
있습니까?
이것은 스크립트 내 통합을위한 것입니다. 바인드 마운트는 올바르게 처리해야합니다. POSIX 호환 솔루션을 환영합니다.
답변:
다음 명령은 파일을 포함하는 마운트 지점의 고유 한 이름을 제공합니다 $file
.
df -P -- "$file" | awk 'NR==2 {print $1}'
이것은 모든 POSIX 시스템 에서 작동 합니다. 이 -P
옵션은 예측 가능한 형식을 적용합니다. 두 번째 줄의 첫 번째 필드는“파일 시스템 이름”입니다. 따라서 두 파일이 동일한 마운트 지점에 있는지 확인하려면 다음을 수행하십시오.
if [ "$(df -P -- "$file1" | awk 'NR==2 {print $1}')" = \
"$(df -P -- "$file2" | awk 'NR==2 {print $1}')" ]; then
echo "$file1 and $file2 are on the same filesystem" ; fi
또는 몇 가지 프로세스 호출을 저장하려면 다음을 수행하십시오.
if df -P -- "$file1" "$file2" |
awk 'NR!=1 {dev[NR] = $1} END {exit(dev[2] != dev[3])}'; then
echo "$file1 and $file2 are on the same filesystem" ; fi
일부 운영 체제는 볼륨 이름에 공백이있을 수 있습니다. df
이 경우 출력 을 구문 분석하는 완전히 신뢰할 수있는 방법은 없습니다 .
후드에서에서 st_dev
반환 된 필드로 파일을 포함하는 파일 시스템을 식별 할 수 있습니다 stat
. 쉘 스크립트에서 이것을 수행하는 휴대용 방법은 없습니다. 일부 시스템에는 stat
유틸리티가 있지만 구문은 다양합니다.
stat
하면 st_dev
필드 가보고 됩니다 stat -c %D -- "$file"
.stat
GNU coreutils와 호환되는 파일이 포함되어 있습니다. 다른 사람들은 선택 stat
의 %c
여지 가 없습니다 . 사용할 수는 stat -t -- "$file" | awk '{print $8}'
있지만 파일 이름에 공백이 포함되어 있지 않거나 stat -t -- "$file" | awk 'END {print $(NF-8)}'
임의의 파일 이름을 처리하지만 나중에 stat
출력 에 필드를 추가 하지 않는 경우에만 작동합니다 .stat
유틸리티 가 필요합니다 stat -f %d -- "$file"
.stat
유틸리티 가 없습니다 .Perl을 사용할 수 있다면
perl -e 'print ((stat($ARGV[0]))[0])' -- "$file"
그리고 비교를하기 위해 :
perl -e 'exit((stat($ARGV[0]))[0] != (stat($ARGV[1]))[0])' -- "$file1" "$file2"
원하는 결과가 명확하지 않은 경우가 있습니다. 예를 들어, 리눅스의 바인드 마운트와, 후 mount --bind /foo /bar
, /foo
와 /bar
같은 파일 시스템으로 간주됩니다. 두 파일이 실제로 동일한 장치에있을 수는 있지만 항상 알 수는 없습니다. 예를 들어, 파일이 두 개의 다른 네트워크 마운트에있는 경우 클라이언트는 서버가 다른 파일 시스템을 내보내는 지 여부를 알 수 없습니다.
파일이 디렉토리이고 파일에 쓸 수있는 경우 다른 방법은 임시 파일을 작성하고 하드 링크를 작성하는 것입니다. 이것은 Linux 바인드 마운트에서 부정적인 결과를보고합니다.
tmp1=$(TMPDIR=$dir1 mktemp)
tmp2=$(TMPDIR=$dir2 mktemp)
if ln -f -- "$tmp1" "$tmp2"; then
echo "$dir1 and $dir2 are on the same filesystem, which supports hard links"
fi
rm -f "$tmp1" "$tmp2"
df
언제나처럼 언젠가 심볼릭 링크를 장치 이름을 제공하지 않지만 /dev/disk/by-uuid/ca09b761-ae1b-450f-8a46-583327b48fb4
제작 df
하지 신뢰성. 지금까지 신뢰할 수있는 유일한 옵션은 stat
기반 솔루션을 사용하는 것입니다.
df
이 장치에 대해보고 하든 그것은 두 호출간에 일관성이 있으므로 비교해도 좋습니다.
df
보고서 /dev/sda6
와는 /dev/disk/by-uuid/ca09b...
모두 동일한 장치를 참조하지만, 다른 지점을 탑재합니다. 각 마운트 지점에서 파일을 시도 할 때 문자열 비교 테스트는 분명히 실패합니다.
mount /dev/sda6 /mnt1
다음에 mount /dev/sda6 /mnt2
매력처럼 작동합니다. cat /proc/mounts
그것으로 괜찮습니다. 그러나 Wheezy 이후에만 루트 파일 시스템의 장치로 /dev/disk/by-uuid/ca09b...
표시됩니다 df
. 또한 시도는이 simlink를 사용하거나 탑재 UUID=ca09b...
구문보다 다른 것을 보여주는 끝나지 않는 마운트 /dev/sda6
에서을 df
(나는 그것이 부팅 과정에서 무슨 짓을 재현하는 방법을 몰라,하지만 여기에 문제가되지 않습니다).
test $(df -P $path1 $path2 | awk '{if (NR!=1) {print $6}}' | uniq | wc -l) -eq 1
여러 경로에서 작동합니다.
$6
장치 이름 ( $1
)이 아닌 마운트 포인트 ( )를 확인 하므로 문제가되지 않습니다.
POSIX에서 사용할 수있는 가장 확실한 솔루션 은 stat (2) 함수가 제공 하는 파일의 장치 ID를 비교하는 것입니다 .
펄은 Gilles가 지적한 것과 비슷한 통계 기능 을 가지고 있습니다 .
perl -e 'exit((stat($ARGV[0]))[0] != (stat($ARGV[1]))[0])' -- file1 file2
"POSIX 방식"은 다음과 같은 C 프로그램을 사용하는 것입니다.
./checksamedev file1 file2
소스 코드는 다음과 같습니다.
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
int main(int argc, char* argv[]) {
struct stat s1, s2;
if( argc==3 && lstat(argv[1], &s1)==0 && lstat(argv[2], &s2)==0 )
return !(s1.st_dev == s2.st_dev);
return 2;
}
두 파일의 장치 ID가 같으면 동일한 파일 시스템에서 호스트됩니다.이 경우 위의 명령은 0을 반환합니다 (그렇지 않으면 다른 값). 로 확인하십시오 echo $?
.
이것은 바인드 마운트에서는 잘 작동하지만 네트워크 마운트에서는 그렇지 않습니다.