From 32d59cecbde2fec1e9d5e145987501e49ed1df2c Mon Sep 17 00:00:00 2001 From: Joachim Metz Date: Thu, 29 Jul 2021 12:24:07 +0200 Subject: [PATCH] Added os extended attribute support for Linux #504 (#595) --- .github/workflows/test_docker.yml | 4 +- .github/workflows/test_tox.yml | 2 +- config/appveyor/install.ps1 | 2 +- config/dpkg/control | 2 +- dependencies.ini | 8 +++ dfvfs/file_io/os_file_io.py | 2 +- dfvfs/vfs/os_attribute.py | 116 ++++++++++++++++++++++++++++++ dfvfs/vfs/os_file_entry.py | 30 +++++--- dfvfs/vfs/os_file_system.py | 4 +- requirements.txt | 1 + setup.cfg | 1 + tests/vfs/os_file_entry.py | 3 +- 12 files changed, 156 insertions(+), 19 deletions(-) create mode 100644 dfvfs/vfs/os_attribute.py diff --git a/.github/workflows/test_docker.yml b/.github/workflows/test_docker.yml index 56fcbaca..4a2c7c45 100644 --- a/.github/workflows/test_docker.yml +++ b/.github/workflows/test_docker.yml @@ -17,7 +17,7 @@ jobs: - name: Install dependencies run: | dnf copr -y enable @gift/dev - dnf install -y python3 libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi python3-cryptography python3-dfdatetime python3-dtfabric python3-idna python3-mock python3-pbr python3-pytsk3 python3-pyyaml python3-setuptools python3-six + dnf install -y python3 libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi python3-cryptography python3-dfdatetime python3-dtfabric python3-idna python3-mock python3-pbr python3-pytsk3 python3-pyxattr python3-pyyaml python3-setuptools python3-six - name: Run tests env: LANG: C.utf8 @@ -57,7 +57,7 @@ jobs: run: | add-apt-repository -y ppa:gift/dev apt-get update -q - apt-get install -y python3 libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi-backend python3-cryptography python3-dfdatetime python3-distutils python3-dtfabric python3-idna python3-mock python3-pbr python3-pytsk3 python3-setuptools python3-six python3-yaml + apt-get install -y python3 libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi-backend python3-cryptography python3-dfdatetime python3-distutils python3-dtfabric python3-idna python3-mock python3-pbr python3-pytsk3 python3-pyxattr python3-setuptools python3-six python3-yaml - name: Run tests env: LANG: en_US.UTF-8 diff --git a/.github/workflows/test_tox.yml b/.github/workflows/test_tox.yml index 1c3bc0cf..cf1fc5f1 100644 --- a/.github/workflows/test_tox.yml +++ b/.github/workflows/test_tox.yml @@ -47,7 +47,7 @@ jobs: add-apt-repository -y ppa:deadsnakes/ppa add-apt-repository -y ppa:gift/dev apt-get update -q - apt-get install -y build-essential git libffi-dev python${{ matrix.python-version }} python${{ matrix.python-version }}-dev python${{ matrix.python-version }}-venv libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi-backend python3-cryptography python3-dfdatetime python3-distutils python3-dtfabric python3-idna python3-mock python3-pbr python3-pip python3-pytsk3 python3-setuptools python3-six python3-yaml + apt-get install -y build-essential git libffi-dev python${{ matrix.python-version }} python${{ matrix.python-version }}-dev python${{ matrix.python-version }}-venv libbde-python3 libewf-python3 libfsapfs-python3 libfsext-python3 libfshfs-python3 libfsntfs-python3 libfsxfs-python3 libfvde-python3 libfwnt-python3 libluksde-python3 libmodi-python3 libqcow-python3 libsigscan-python3 libsmdev-python3 libsmraw-python3 libvhdi-python3 libvmdk-python3 libvsgpt-python3 libvshadow-python3 libvslvm-python3 python3-cffi-backend python3-cryptography python3-dfdatetime python3-distutils python3-dtfabric python3-idna python3-mock python3-pbr python3-pip python3-pytsk3 python3-pyxattr python3-setuptools python3-six python3-yaml - name: Install tox run: | python3 -m pip install tox diff --git a/config/appveyor/install.ps1 b/config/appveyor/install.ps1 index 90222687..1ba2de2e 100644 --- a/config/appveyor/install.ps1 +++ b/config/appveyor/install.ps1 @@ -1,6 +1,6 @@ # Script to set up tests on AppVeyor Windows. -$Dependencies = "PyYAML cffi cryptography dfdatetime dtfabric idna libbde libewf libfsapfs libfsext libfshfs libfsntfs libfsxfs libfvde libfwnt libluksde libmodi libqcow libsigscan libsmdev libsmraw libvhdi libvmdk libvsgpt libvshadow libvslvm mock pbr pytsk3 six" +$Dependencies = "PyYAML cffi cryptography dfdatetime dtfabric idna libbde libewf libfsapfs libfsext libfshfs libfsntfs libfsxfs libfvde libfwnt libluksde libmodi libqcow libsigscan libsmdev libsmraw libvhdi libvmdk libvsgpt libvshadow libvslvm mock pbr pytsk3 six xattr" $Dependencies = ${Dependencies} -split " " $Output = Invoke-Expression -Command "git clone https://github.com/log2timeline/l2tdevtools.git ..\l2tdevtools 2>&1" diff --git a/config/dpkg/control b/config/dpkg/control index 477dac01..09014bf9 100644 --- a/config/dpkg/control +++ b/config/dpkg/control @@ -9,7 +9,7 @@ Homepage: https://github.com/log2timeline/dfvfs Package: python3-dfvfs Architecture: all -Depends: libbde-python3 (>= 20140531), libewf-python3 (>= 20131210), libfsapfs-python3 (>= 20201107), libfsext-python3 (>= 20210721), libfshfs-python3 (>= 20210722), libfsntfs-python3 (>= 20200921), libfsxfs-python3 (>= 20210726), libfvde-python3 (>= 20160719), libfwnt-python3 (>= 20210717), libluksde-python3 (>= 20200101), libmodi-python3 (>= 20210405), libqcow-python3 (>= 20201213), libsigscan-python3 (>= 20191221), libsmdev-python3 (>= 20140529), libsmraw-python3 (>= 20140612), libvhdi-python3 (>= 20201014), libvmdk-python3 (>= 20140421), libvsgpt-python3 (>= 20210207), libvshadow-python3 (>= 20160109), libvslvm-python3 (>= 20160109), python3-cffi-backend (>= 1.9.1), python3-cryptography (>= 2.0.2), python3-dfdatetime (>= 20210509), python3-dtfabric (>= 20170524), python3-idna (>= 2.5), python3-pytsk3 (>= 20210419), python3-yaml (>= 3.10), ${python3:Depends}, ${misc:Depends} +Depends: libbde-python3 (>= 20140531), libewf-python3 (>= 20131210), libfsapfs-python3 (>= 20201107), libfsext-python3 (>= 20210721), libfshfs-python3 (>= 20210722), libfsntfs-python3 (>= 20200921), libfsxfs-python3 (>= 20210726), libfvde-python3 (>= 20160719), libfwnt-python3 (>= 20210717), libluksde-python3 (>= 20200101), libmodi-python3 (>= 20210405), libqcow-python3 (>= 20201213), libsigscan-python3 (>= 20191221), libsmdev-python3 (>= 20140529), libsmraw-python3 (>= 20140612), libvhdi-python3 (>= 20201014), libvmdk-python3 (>= 20140421), libvsgpt-python3 (>= 20210207), libvshadow-python3 (>= 20160109), libvslvm-python3 (>= 20160109), python3-cffi-backend (>= 1.9.1), python3-cryptography (>= 2.0.2), python3-dfdatetime (>= 20210509), python3-dtfabric (>= 20170524), python3-idna (>= 2.5), python3-pytsk3 (>= 20210419), python3-pyxattr (>= 0.7.2), python3-yaml (>= 3.10), ${python3:Depends}, ${misc:Depends} Description: Python 3 module of dfVFS dfVFS, or Digital Forensics Virtual File System, provides read-only access to file-system objects from various storage media types and file formats. The goal diff --git a/dependencies.ini b/dependencies.ini index 71da9f47..b28ec6bd 100644 --- a/dependencies.ini +++ b/dependencies.ini @@ -195,6 +195,14 @@ pypi_name: libvslvm-python rpm_name: libvslvm-python3 version_property: get_version() +[xattr] +dpkg_name: python3-pyxattr +is_optional: true +minimum_version: 0.7.2 +pypi_name: pyxattr +rpm_name: python3-pyxattr +version_property: __version__ + [yaml] dpkg_name: python3-yaml l2tbinaries_name: PyYAML diff --git a/dfvfs/file_io/os_file_io.py b/dfvfs/file_io/os_file_io.py index 5cda9171..f2197f40 100644 --- a/dfvfs/file_io/os_file_io.py +++ b/dfvfs/file_io/os_file_io.py @@ -11,7 +11,7 @@ class OSFile(file_io.FileIO): - """File input/output (IO) object using os.""" + """File input/output (IO) object that uses the operating system.""" def __init__(self, resolver_context, path_spec): """Initializes a file input/output (IO) object. diff --git a/dfvfs/vfs/os_attribute.py b/dfvfs/vfs/os_attribute.py new file mode 100644 index 00000000..b26e887c --- /dev/null +++ b/dfvfs/vfs/os_attribute.py @@ -0,0 +1,116 @@ +# -*- coding: utf-8 -*- +"""The operating system attribute implementation.""" + +import os + +try: + import xattr +except ImportError: + xattr = None + +from dfvfs.vfs import attribute + + +class OSExtendedAttribute(attribute.Attribute): + """Extended attribute that uses the operating system.""" + + def __init__(self, location, name): + """Initializes an attribute. + + Args: + location (str): path of the file. + name (str): name of the extended attribute. + """ + super(OSExtendedAttribute, self).__init__() + self._current_offset = 0 + self._data = xattr.getxattr(location, name) + self._location = location + self._name = name + self._size = len(self._data) + + @property + def name(self): + """str: name.""" + return self._name + + # Note: that the following functions do not follow the style guide + # because they are part of the file-like object interface. + # pylint: disable=invalid-name + + def read(self, size=None): + """Reads a byte string from the file input/output (IO) object. + + The function will read a byte string of the specified size or + all of the remaining data if no size was specified. + + Args: + size (Optional[int]): number of bytes to read, where None is all + remaining data. + + Returns: + bytes: data read. + + Raises: + IOError: if the read failed. + OSError: if the read failed. + """ + if size is None: + size = self._size + + data = self._data[self._current_offset:self._current_offset + size] + self._current_offset += len(data) + return data + + def seek(self, offset, whence=os.SEEK_SET): + """Seeks to an offset within the file input/output (IO) object. + + Args: + offset (int): offset to seek. + whence (Optional[int]): value that indicates whether offset is an + absolute or relative position within the file. + + Raises: + IOError: if the seek failed. + OSError: if the seek failed. + """ + if whence == os.SEEK_CUR: + offset += self._current_offset + elif whence == os.SEEK_END: + offset = self._size - offset + elif whence != os.SEEK_SET: + raise IOError('Invalid whence') + + if offset < 0: + raise IOError('Invalid offset') + + self._current_offset = offset + + # get_offset() is preferred above tell() by the libbfio layer used in libyal. + def get_offset(self): + """Retrieves the current offset into the file input/output (IO) object. + + Returns: + int: current offset into the file input/output (IO) object. + """ + return self._current_offset + + # Pythonesque alias for get_offset(). + def tell(self): + """Retrieves the current offset into the file input/output (IO) object.""" + return self.get_offset() + + def get_size(self): + """Retrieves the size of the file input/output (IO) object. + + Returns: + int: size of the file input/output (IO) object. + """ + return self._size + + def seekable(self): + """Determines if a file input/output (IO) object is seekable. + + Returns: + bool: True since a file IO object provides a seek method. + """ + return True diff --git a/dfvfs/vfs/os_file_entry.py b/dfvfs/vfs/os_file_entry.py index fb830b83..27c7bb1a 100644 --- a/dfvfs/vfs/os_file_entry.py +++ b/dfvfs/vfs/os_file_entry.py @@ -7,6 +7,11 @@ import pysmdev +try: + import xattr +except ImportError: + xattr = None + from dfdatetime import posix_time as dfdatetime_posix_time from dfvfs.lib import definitions @@ -14,10 +19,11 @@ from dfvfs.path import os_path_spec from dfvfs.vfs import attribute from dfvfs.vfs import file_entry +from dfvfs.vfs import os_attribute class OSDirectory(file_entry.Directory): - """File system directory that uses os.""" + """File system directory that uses the operating system.""" def _EntriesGenerator(self): """Retrieves directory entries. @@ -99,6 +105,7 @@ def __init__(self, resolver_context, file_system, path_spec, is_root=False): resolver_context, file_system, path_spec, is_root=is_root, is_virtual=False) self._is_windows_device = is_windows_device + self._location = location self._name = None self._stat_info = stat_info @@ -140,6 +147,12 @@ def _GetAttributes(self): stat_attribute = self._GetStatAttribute() self._attributes = [stat_attribute] + if xattr: + for name in xattr.listxattr(self._location): + extended_attribute = os_attribute.OSExtendedAttribute( + self._location, name) + self._attributes.append(extended_attribute) + return self._attributes def _GetDirectory(self): @@ -162,11 +175,10 @@ def _GetLink(self): if self._link is None: self._link = '' - location = getattr(self.path_spec, 'location', None) - if location is None: + if self._location is None: return self._link - self._link = os.readlink(location) + self._link = os.readlink(self._location) self._link = os.path.abspath(self._link) return self._link @@ -291,9 +303,8 @@ def modification_time(self): def name(self): """str: name of the file entry, without the full path.""" if self._name is None: - location = getattr(self.path_spec, 'location', None) - if location is not None: - self._name = self._file_system.BasenamePath(location) + if self._location is not None: + self._name = self._file_system.BasenamePath(self._location) return self._name @property @@ -323,11 +334,10 @@ def GetParentFileEntry(self): Returns: OSFileEntry: parent file entry or None if not available. """ - location = getattr(self.path_spec, 'location', None) - if location is None: + if self._location is None: return None - parent_location = self._file_system.DirnamePath(location) + parent_location = self._file_system.DirnamePath(self._location) if parent_location is None: return None diff --git a/dfvfs/vfs/os_file_system.py b/dfvfs/vfs/os_file_system.py index 69580efc..7dbce19e 100644 --- a/dfvfs/vfs/os_file_system.py +++ b/dfvfs/vfs/os_file_system.py @@ -1,5 +1,5 @@ # -*- coding: utf-8 -*- -"""The file system implementation that is provided by the operating system.""" +"""The operating system file system implementation.""" import os import platform @@ -14,7 +14,7 @@ class OSFileSystem(file_system.FileSystem): - """File system provided by the operating system.""" + """File system that uses the operating system.""" if platform.system() == 'Windows': PATH_SEPARATOR = '\\' diff --git a/requirements.txt b/requirements.txt index 93de9dd4..3331971f 100644 --- a/requirements.txt +++ b/requirements.txt @@ -26,3 +26,4 @@ libvsgpt-python >= 20210207 libvshadow-python >= 20160109 libvslvm-python >= 20160109 pytsk3 >= 20210419 +pyxattr >= 0.7.2 diff --git a/setup.cfg b/setup.cfg index 81047f47..033d88b8 100644 --- a/setup.cfg +++ b/setup.cfg @@ -43,6 +43,7 @@ requires = libbde-python3 >= 20140531 python3-dtfabric >= 20170524 python3-idna >= 2.5 python3-pytsk3 >= 20210419 + python3-pyxattr >= 0.7.2 python3-pyyaml >= 3.10 [bdist_wheel] diff --git a/tests/vfs/os_file_entry.py b/tests/vfs/os_file_entry.py index 427dc89d..47f4daa4 100644 --- a/tests/vfs/os_file_entry.py +++ b/tests/vfs/os_file_entry.py @@ -95,7 +95,8 @@ def testGetAttributes(self): file_entry._GetAttributes() self.assertIsNotNone(file_entry._attributes) - self.assertEqual(len(file_entry._attributes), 1) + # Note that on some platforms this file can have extended attributes. + self.assertGreaterEqual(len(file_entry._attributes), 1) test_attribute = file_entry._attributes[0] self.assertIsInstance(test_attribute, attribute.StatAttribute)