24 python windows file-attributes
上次我问了一个类似的问题,但这是关于svn相关的版本信息.现在我想知道如何查询Windows"文件版本"属性,例如.一个DLL.我也关注wmi和win32file模块,但没有成功.
小智 32
这是一个将所有文件属性作为字典读取的函数:
import win32api
#==============================================================================
def getFileProperties(fname):
#==============================================================================
"""
Read all properties of the given file return them as a dictionary.
"""
propNames = ('Comments', 'InternalName', 'ProductName',
'CompanyName', 'LegalCopyright', 'ProductVersion',
'FileDescription', 'LegalTrademarks', 'PrivateBuild',
'FileVersion', 'OriginalFilename', 'SpecialBuild')
props = {'FixedFileInfo': None, 'StringFileInfo': None, 'FileVersion': None}
try:
# backslash as parm returns dictionary of numeric info corresponding to VS_FIXEDFILEINFO struc
fixedInfo = win32api.GetFileVersionInfo(fname, '\\')
props['FixedFileInfo'] = fixedInfo
props['FileVersion'] = "%d.%d.%d.%d" % (fixedInfo['FileVersionMS'] / 65536,
fixedInfo['FileVersionMS'] % 65536, fixedInfo['FileVersionLS'] / 65536,
fixedInfo['FileVersionLS'] % 65536)
# \VarFileInfo\Translation returns list of available (language, codepage)
# pairs that can be used to retreive string info. We are using only the first pair.
lang, codepage = win32api.GetFileVersionInfo(fname, '\\VarFileInfo\\Translation')[0]
# any other must be of the form \StringfileInfo\%04X%04X\parm_name, middle
# two are language/codepage pair returned from above
strInfo = {}
for propName in propNames:
strInfoPath = u'\\StringFileInfo\\%04X%04X\\%s' % (lang, codepage, propName)
## print str_info
strInfo[propName] = win32api.GetFileVersionInfo(fname, strInfoPath)
props['StringFileInfo'] = strInfo
except:
pass
return props
Run Code Online (Sandbox Code Playgroud)
小智 20
最好添加一个try/except,以防文件没有版本号属性.
filever.py
from win32api import GetFileVersionInfo, LOWORD, HIWORD
def get_version_number (filename):
try:
info = GetFileVersionInfo (filename, "\\")
ms = info['FileVersionMS']
ls = info['FileVersionLS']
return HIWORD (ms), LOWORD (ms), HIWORD (ls), LOWORD (ls)
except:
return 0,0,0,0
if __name__ == '__main__':
import os
filename = os.environ["COMSPEC"]
print ".".join ([str (i) for i in get_version_number (filename)])
Run Code Online (Sandbox Code Playgroud)
yourscript.py:
import os,filever
myPath="C:\\path\\to\\check"
for root, dirs, files in os.walk(myPath):
for file in files:
file = file.lower() # Convert .EXE to .exe so next line works
if (file.count('.exe') or file.count('.dll')): # Check only exe or dll files
fullPathToFile=os.path.join(root,file)
major,minor,subminor,revision=filever.get_version_number(fullPathToFile)
print "Filename: %s \t Version: %s.%s.%s.%s" % (file,major,minor,subminor,revision)
Run Code Online (Sandbox Code Playgroud)
干杯!
the*_*oid 13
您可以使用http://sourceforge.net/projects/pywin32/上的pyWin32模块:
from win32com.client import Dispatch
ver_parser = Dispatch('Scripting.FileSystemObject')
info = ver_parser.GetFileVersion(path)
if info == 'No Version Information Available':
info = None
Run Code Online (Sandbox Code Playgroud)
der*_*cki 11
这是一个版本,也适用于非Windows环境,使用pefile-module:
import pefile
def LOWORD(dword):
return dword & 0x0000ffff
def HIWORD(dword):
return dword >> 16
def get_product_version(path):
pe = pefile.PE(path)
#print PE.dump_info()
ms = pe.VS_FIXEDFILEINFO.ProductVersionMS
ls = pe.VS_FIXEDFILEINFO.ProductVersionLS
return (HIWORD (ms), LOWORD (ms), HIWORD (ls), LOWORD (ls))
if __name__ == "__main__":
import sys
try:
print "%d.%d.%d.%d" % get_product_version(sys.argv[1])
except:
print "Version info not available. Maybe the file is not a Windows executable"
Run Code Online (Sandbox Code Playgroud)
我发现唯一的跨平台答案(使用pefile)无法找到我正在寻找的版本字符串,这是 Windows 上 .DLL 文件的“查看属性”的“详细信息”选项卡中显示的版本字符串。然而,该代码基于dump_info()中的函数pefile,能够找到这些属性,其中包括原始文件名、版权、公司名称以及文件和产品版本。
encoding如果它们不是使用 UTF-8 编码的,您可能需要替换为不同的编码。
import pefile
PATH_TO_FILE = 'C:\...'
pe = pefile.PE(PATH_TO_FILE)
if hasattr(pe, 'VS_VERSIONINFO'):
for idx in range(len(pe.VS_VERSIONINFO)):
if hasattr(pe, 'FileInfo') and len(pe.FileInfo) > idx:
for entry in pe.FileInfo[idx]:
if hasattr(entry, 'StringTable'):
for st_entry in entry.StringTable:
for str_entry in sorted(list(st_entry.entries.items())):
print('{0}: {1}'.format(
str_entry[0].decode('utf-8', 'backslashreplace'),
str_entry[1].decode('utf-8', 'backslashreplace')))
Run Code Online (Sandbox Code Playgroud)
这是马什答案的工作版本,仅使用标准库。像这样使用它get_version_string(file, "FileVersion")。有关有效版本字符串,请参阅VerQueryValueW() 。
from ctypes import *
# returns the requested version information from the given file
#
# `what` is one of the predefined version information strings, such as
# "FileVersion" or "CompanyName"
#
# `language` should be an 8-character string combining both the language and
# codepage (such as "040904b0"); if None, the first language in the translation
# table is used instead
#
def get_version_string(filename, what, language=None):
# VerQueryValue() returns an array of that for VarFileInfo\Translation
#
class LANGANDCODEPAGE(Structure):
_fields_ = [
("wLanguage", c_uint16),
("wCodePage", c_uint16)]
wstr_file = wstring_at(filename)
# getting the size in bytes of the file version info buffer
size = windll.version.GetFileVersionInfoSizeW(wstr_file, None)
if size == 0:
raise WinError()
buffer = create_string_buffer(size)
# getting the file version info data
if windll.version.GetFileVersionInfoW(wstr_file, None, size, buffer) == 0:
raise WinError()
# VerQueryValue() wants a pointer to a void* and DWORD; used both for
# getting the default language (if necessary) and getting the actual data
# below
value = c_void_p(0)
value_size = c_uint(0)
if language is None:
# file version information can contain much more than the version
# number (copyright, application name, etc.) and these are all
# translatable
#
# the following arbitrarily gets the first language and codepage from
# the list
ret = windll.version.VerQueryValueW(
buffer, wstring_at(r"\VarFileInfo\Translation"),
byref(value), byref(value_size))
if ret == 0:
raise WinError()
# value points to a byte inside buffer, value_size is the size in bytes
# of that particular section
# casting the void* to a LANGANDCODEPAGE*
lcp = cast(value, POINTER(LANGANDCODEPAGE))
# formatting language and codepage to something like "040904b0"
language = "{0:04x}{1:04x}".format(
lcp.contents.wLanguage, lcp.contents.wCodePage)
# getting the actual data
res = windll.version.VerQueryValueW(
buffer, wstring_at("\\StringFileInfo\\" + language + "\\" + what),
byref(value), byref(value_size))
if res == 0:
raise WinError()
# value points to a string of value_size characters, minus one for the
# terminating null
return wstring_at(value.value, value_size.value - 1)
Run Code Online (Sandbox Code Playgroud)
小智 4
我在“timgolden”网站找到了这个解决方案。工作正常。
from win32api import GetFileVersionInfo, LOWORD, HIWORD
def get_version_number (filename):
info = GetFileVersionInfo (filename, "\\")
ms = info['FileVersionMS']
ls = info['FileVersionLS']
return HIWORD (ms), LOWORD (ms), HIWORD (ls), LOWORD (ls)
if __name__ == '__main__':
import os
filename = os.environ["COMSPEC"]
print ".".join ([str (i) for i in get_version_number (filename)])
Run Code Online (Sandbox Code Playgroud)