Skip to main content

Python wrapper for using Win32's IFileOperation for manipulating the filesystem.

Project description

tests License Code style: black Imports: isort

IFileOperation- a simple wrapper to use Windows shell file operations.

This is a very small wrapper around IFileOperation to expose methods to perform file opereration ( move, copy, rename, new, delete) on files using Windows shell operations. This means these operations can be undone, files can be recycled, and you can allow users to handle name conflicts: just like when you do any of these through Windows Explorer.

At the moment this is a very young and so only exposes the file operations:

  • Move
  • Copy
  • Rename
  • New
  • Delete

More features may be added in the future, feel free to open a feature request for something specific.

Installation:

pip install ifileoperation

At the moment the library is tested to run on Python 3.11, but other version may work if pywin32 and comtypes are available.

Usage

All of the functionality is accessed via the FileOperator class. This is a scheduling context manager. If you're familiar with IFileOperation, you'll know that you queue a bunch of operations you wish to be performed, then order them to be executed all at once:

from Pathlib import Path
from ifileoperation import FileOperator, FileOperationFlags


if __name__ == '__main__':
    # Configure for recycling, showing a prompt if the file(s) can't be recyled
    # and must be deleted instead.
    recycle_flags = (
      # ALLOW_UNDO: for versions < Win8, ADDUNDORECORD is preferred for Win8+
      FileOperationFlags.ALLOW_UNDO | FileOperationFlags.ADDUNDORECORD |
      # RECYCLEONDELETE: Perform deletes as recycles
      FileOperationFlags.RECYCLEONDELETE |
      # WANTNUKEWARNING: If a file is too big to be recyled, show a warning that
      #  it will be deleted instead
      FileOperationFlags.WANTNUKEWARNING |
    )
    # Don't show any confirmation dialogs (except for WANTNUKEWARNING), or a UAC
    # prompt if elevation is required (default behavior)
    ui_flags = (
      # NO_CONFIMATION: Don't show confirmation dialogs (ie, treat all dialogs
      # as if 'Yes to All' was selected)
      FileOperationFlags.NO_CONFIRMATION |
    )
    config = recycle_flags | ui_flags
    with FileOperator(flags=config, commit_on_exit=True) as op:
        op.delete_file('Q:/foo.txt')
        op.delete_file(Path.cwd().join('bar.txt'))

FileOperator

A thin wrapper around IFileOperation, exposing the move, copy, rename, and delete operations. All paths can be specified as anything implementing the os.PathLike[str] interface: strings, Path objects, or your own custom PathLike class. The FileOperator instance is a context manager: you queue operations in the with block, and then commit them or have them committed automatically.

NOTE: FileOperator is not reentrant. Once you've entered the with block and exited, you should create a new instance to perform more operations.

Configuring:

Configuration is done via the constructor:

__init__(self, parent=None, flags=None, commit_on_exit=False):
  • parent: A optional HANDLE to the parent that should own any dialog boxes shown. You may also pass a wx.Window object, and FileOperator will extract the handle for you.
  • flags: A FileOperationFlags instance indicating the options you want when performing the operations. See the Microsoft documentation on these flags for more details on what each means. These are also in the docstrings for FileOperationFlags, so you can read there them as well.
  • commit_on_exit: If set to True, all the queued operations will be automatically commited as long as no exceptions occurred withing the with block before exiting. Exceptions might still occur during the committing itself though!

Operations supported:

The basic file operations can be scheduled. In these method signatures, StrPath is defined as str | os.PathLike[str].

  • move_file(source: StrPath, destination: StrPath, new_name: str | None = None): Moves the file at source to the destination directory destination. If new_name is supplied, the file will also be renamed to the new name.
  • move_files(sources: Iterable[StrPath], destination: StrPath, new_name: str | None = None): Moves the specified files sources to the destination directory destination. If new_name is supplied, the files will all be renamed to the new name. In this case, you probably also want to supply FileOperationFlags.RENAMEONCOLLISION.
  • copy_file(source: StrPath, destination: StrPath, new_name: str | None = None): Like move, but performs a copy instead.
  • copy_files(sources: Iterable[StrPath], destination: StrPath): Like move (without the rename), but performs a copy instead.
  • rename_file(source: StrPath, new_name: str, allow_move: bool = True): Renames a file to a new name. Normally, new_name should just be the name of the new file, without its directory. However, if you're used to other file APIs that let you "rename" a file into a different directory, rename_file supports this as well. When allow_move is True, if it is detected that new_name is actually a path resolving to a different directory, this will automatically be transformed into a move_file operation instead.
  • rename_files(sources: Iterable[StrPath], new_name: str): Renames the given files to have the new name. Unlike rename_file, you cannot rename files into files in different directories.
  • delete_file(source: StrPath): Deletes the target file.
  • delete_files(sources: Iterable[StrPath]): Deletes the target files.
  • commit(): Cause all the queue operations to be performed. After a commit, the .aborted attribute will be set to True or False depending on if any file operations were Canceled/Aborted.

Exceptions

All exceptions raised by this library are based on IFileOperationError, except for any exceptions that are translated into standard library exceptions (at the moment, none of them are). These exceptions are just a simple translation of pythoncom.com_error, so you don't have to have any knowledge of the pywin32 library.

I have plans to convert some of the exceptions into standard OSErrors, for example FileNotFoundError, PermimssionError, etc.

Contributing

Pull requests are welcome! We use branch protection with some (minimal) tests at the moment: just formatting really. To install the dependencies needed for development, use: pip install -r requirements-dev.txt

And before committing make sure to check against the tests and formatters:

  • pytest: No tests at the moment
  • black ifileoperation: Code formatting
  • isort ifileoperation: Code formatting
  • flake8 ifileoperation: Linting

IFileOperation uses semantic versioning, but if you're unsure if your changes need a version bump feel free to note that in your pull request and you'll get feedback!

If you're not into coding, you can open a Bug Report or Feature Request and I'll look into it.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

ifileoperation-1.0.0.tar.gz (16.5 kB view hashes)

Uploaded Source

Built Distribution

ifileoperation-1.0.0-py3-none-any.whl (14.6 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page