2

在我努力解决Python 问题 1578269的过程中,我一直在努力尝试以一种稳健的方式解决符号链接的目标。我首先按照stackoverflowMicrosoft的建议使用 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 回答