web-dev-qa-db-ja.com

特定のファイルを含むファイルシステムのマウントポイントを取得する方法

特定のFILEを含むファイルシステムのマウントポイントをすばやく見つける方法を探しています。以下の私の解決策よりも簡単または直接的なものはありますか?

df -h FILE |tail -1 | awk -F% '{print $NF}' | tr -d ' '

同様の質問 " ディスクがマウントされている場所を確認するコマンドはありますか? "は、ディスクからの任意のファイルではなく、現在のディスクのデバイスノードを入力として使用します...

13
Stu

あなたは次のようなことができます

df -P FILE | awk 'NR==2{print $NF}'

あるいは

df -P FILE | awk 'END{print $NF}'

awkはデフォルトで空白で分割されるため、-Fを指定する必要はありません。また、trで空白をトリミングする必要もありません。最後に、対象の行番号(NR==2)を指定することで、tailを使用しないこともできます。

6
terdon

GNU/Linuxの場合、coreutils 8.6以降のGNU statがある場合、次のことができます。

stat -c %m -- "$file"

さもないと:

mount_point_of() {
  f=$(readlink -e -- "$1") &&
    until mountpoint -q -- "$f"; do
      f=${f%/*}; f=${f:-/}
    done &&
    printf '%s\n' "$f"
}

あなたのアプローチは有効ですが、マウントポイントにスペース、%、改行、またはその他の印刷できない文字が含まれていないことを前提としています。新しいバージョンのGNU df (8.21以上):

df --output=target FILE | tail -n +2
16

Linuxの場合、util-linuxから正確に作成されたことがわかります

findmnt -n -o TARGET --target /path/to/FILE

バインドマウントが複数ある場合、ランダムなマウントポイントの種類が返されることがあります。 dfを使用しても同じ問題があります。

8
rudimeier

statは「デバイス」フィールドを返すため、基になるstat()ライブラリー呼び出しを使用して、この情報をプログラムでPOSIX準拠の方法で取得する方法を知りたいと思っていました。

このCコードのスニペット:

_#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>

int main (int argc, const char *argv[]) {
    struct stat info;
    stat(argv[1], &info);
    printf("min: %d maj: %d\n",
        minor(info.st_dev),
        major(info.st_dev)
    );

    return 0;
}  
_

コマンドラインにリストされたファイルを含むデバイスのメジャーおよびマイナーデバイスIDを提供します(_argv[1]_)。残念ながら、major()minor()はPOSIXではありませんが、manページには、GNU/linux以外の「他の多くのシステムに存在する」と記載されています。

次に、たとえば、_/proc/diskstats_からデバイスのメジャー/マイナー番号とデバイスノード間の対応を取得し、それを_/proc/mounts_からマウントポイントにマッピングできます。 _/etc/mtab_。

したがって、これを行うコマンドラインユーティリティは非常に簡単です。

3
goldilocks

これをC++から実行する場合のC++コードは次のとおりです...

  #include <boost/filesystem.hpp>
  #include <sys/stat.h>

  /// returns true if the path is a mount point
  bool Stat::IsMount(const std::string& path)
  {

      if (path == "") return false;
      if (path == "/") return true;

      boost::filesystem::path path2(path);
      auto parent = path2.parent_path();

      struct stat sb_path;
      if (lstat(path.c_str(), &sb_path) == -1) return false; // path does not exist
      if (!S_ISDIR(sb_path.st_mode)) return false; // path is not a directory

      struct stat sb_parent;
      if (lstat(parent.string().c_str(), &sb_parent) == -1 ) return false; // parent does not exist

      if (sb_path.st_dev == sb_parent.st_dev) return false; // parent and child have same device id

      return true;

  }

  /// returns the path to the mount point that contains the path
  std::string Stat::MountPoint(const std::string& path0)
  {
      // first find the first "real" part of the path, because this file may not exist yet
      boost::filesystem::path path(path0);
      while(!boost::filesystem::exists(path) )
      {
          path = path.parent_path();
      }

      // then look for the mount point
      path = boost::filesystem::canonical(path);
      while(! IsMount(path.string()) )
      {
          path = path.parent_path();
      }

      return path.string();
  }

プログラムによる方法のその他のリンク

0
Mark Lakata