6

作为脚本的主要目的完成后的“清理”,调用一个函数以递归方式查看每个文件夹并删除所有以预定扩展名结尾的文件。

在测试过程中,我发现要删除的文件列表中的一些文件扩展名实际上会引发错误:[Errno 1] Operation not permitted: '/location/of/locked/file.png. 查看文件本身,它似乎已锁定(在 Mac 上)。

  1. 我将如何使用 Python 从每个文件/文件夹中删除锁定的属性(如果它存在),然后如果文件以扩展名结尾,则删除该文件?
    最好这可以在下面的同一个函数中完成,因为遍历输入目录需要很长时间 - 只处理一次是要走的路。
  2. 这如何影响脚本在 Windows 上的完整性?
    我已经以一种使其在操作系统之间兼容的方式对其进行了编程,但是(据我所知)locked属性在 Windows 上并不像在 mac 上那样存在,并且可能会导致未知的副作用。

REMOVE_FILETYPES = ('.png', '.jpg', '.jpeg', '.pdf')

def cleaner(currentPath):
  if not os.path.isdir(currentPath):
    if currentPath.endswith(REMOVE_FILETYPES) or os.path.basename(currentPath).startswith('.'):
      try:
        os.remove(currentPath)
        print('REMOVED: \"{removed}\"'.format(removed = currentPath))
      except BaseException as e:
        print('ERROR: Could not remove: \"{failed}\"'.format(failed = str(e)))
      finally:
        return True
    return False
    
  if all([cleaner(os.path.join(currentPath, file)) for file in os.listdir(currentPath)]):
    try:
      os.rmdir(currentPath)
      print('REMOVED: \"{removed}\"'.format(removed = currentPath))
    except:
      print('ERROR: Could not remove: \"{failed}\"'.format(failed = currentPath))
    finally:
      return True
  return False

cleaner(r'/path/to/parent/dir')

如果有人能告诉我如何将这些功能集成到子程序中,我将不胜感激。干杯。


编辑:根据请求删除错误处理

def cleaner(currentPath):
    if sys.platform == 'darwin':
        os.system('chflags nouchg {}'.format(currentPath))
    if not os.path.isdir(currentPath):
        if currentPath.endswith(REMOVE_FILETYPES) or os.path.basename(currentPath).startswith('.'):
            try:
                os.remove(currentPath)
                print('REMOVED: \"{removed}\"'.format(removed=currentPath))
            except PermissionError:
                if sys.platform == 'darwin':
                    os.system('chflags nouchg {}'.format(currentPath))
                    os.remove(currentPath)
    if all([cleaner(os.path.join(currentPath, file)) for file in os.listdir(currentPath)]) and not currentPath == SOURCE_DIR:
        os.rmdir(currentPath)
        print('REMOVED: \"{removed}\"'.format(removed=currentPath))
4

1 回答 1

2

chflags您可以使用以下命令解锁文件:

os.system('chflags nouchg {}'.format(filename))

(有一个函数os.chflags,但与锁定状态相关联的标志不是常规标志,而是os模块文档所称的“用户定义”标志,如您所见os.stat(locked_filename).st_flags。)

为了解决您的问题,我将chflags上面的命令添加到特定except:于您尝试删除锁定文件的错误,以及平台检查:

try:
    os.remove(currentPath)
    print('REMOVED: \"{removed}\"'.format(removed = currentPath))
except PermissionError:
    if sys.platform == 'darwin':
        os.system('chflags nouchg {}'.format(currentPath))
        os.remove(currentPath)
    else:
        raise
except BaseException as e:
    ...
于 2018-02-08T04:52:20.470 回答