2

Python の問題 1578269を解決するために、シンボリック リンクのターゲットを堅牢な方法で解決することに取り組んできました。ここで推奨されている GetFinalPathNameByHandle を使用することから始めましたが、ターゲットが使用されている場合 (pagefile.sys など)、この手法は失敗することがわかりました。

そのため、CreateFile と DeviceIoControl を使用してこれを達成するための新しいルーチンを作成しました (これは Explorer が行うことのようです)。jaraco.windows.filesystemからの関連コードを以下に示します。

問題は、Windows でシンボリック リンクを確実に解決するためのより良い手法があるかということです。この実装の問題を特定できますか?

 def relpath(path, start=os.path.curdir):
  """
  Like os.path.relpath, but actually honors the start path
  if supplied. See http://bugs.python.org/issue7195
  """
  return os.path.normpath(os.path.join(start, path))

 def trace_symlink_target(link):
  """
  Given a file that is known to be a symlink, trace it to its ultimate
  target.

  Raises TargetNotPresent when the target cannot be determined.
  Raises ValueError when the specified link is not a symlink.
  """

  if not is_symlink(link):
   raise ValueError("link must point to a symlink on the system")
  while is_symlink(link):
   orig = os.path.dirname(link)
   link = _trace_symlink_immediate_target(link)
   link = relpath(link, orig)
  return link

 def _trace_symlink_immediate_target(link):
  handle = CreateFile(
   link,
   0,
   FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
   None,
   OPEN_EXISTING,
   FILE_FLAG_OPEN_REPARSE_POINT|FILE_FLAG_BACKUP_SEMANTICS,
   None,
   )

  res = DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, None, 10240)

  bytes = create_string_buffer(res)
  p_rdb = cast(bytes, POINTER(REPARSE_DATA_BUFFER))
  rdb = p_rdb.contents
  if not rdb.tag == IO_REPARSE_TAG_SYMLINK:
   raise RuntimeError("Expected IO_REPARSE_TAG_SYMLINK, but got %d" % rdb.tag)
  return rdb.get_print_name()
4

1 に答える 1

0

残念ながら、来週まで Vista でテストすることはできませんが、使用中のファイルであっても、GetFinalPathNameByHandle は機能するはずです。どのような問題に気付きましたか? 上記のコードでは、ファイル ハンドルを閉じるのを忘れています。

于 2009-10-24T10:46:31.133 に答える