如何让SCons不包含zip文件中的基础目录?

SCons提供了一个Zip构建器,用于从文件组生成zip文件.

例如,假设我们有一个文件夹foo,如下所示:

foo/
foo/blah.txt

我们从文件夹foo创建zip文件foo.zip:

env.Zip('foo.zip', 'foo/')

这会产生一个zip文件:

$unzip -l foo.zip
Archive:  foo.zip
  foo/
  foo/foo.txt

但是,假设我们使用的是VariantDir,它包含foo:

bar/
bar/foo/
bar/foo/foo.txt

因为我们在VariantDir中,所以我们仍然使用相同的命令来创建zip文件,即使它具有稍微不同的效果:

env.Zip('foo.zip', 'foo/')

这会生成zip文件:

$unzip -l bar/foo.zip
Archive:  bar/foo.zip
  bar/foo/
  bar/foo/foo.txt

问题是zip中每个文件的额外bar /前缀.如果这不是SCons,那么简单的解决方案就是cd进入bar并用cd bar之类的东西从那里调用zip; zip -r foo.zip foo /.然而,这对SCons来说很奇怪/困难,而且无论如何看起来都非常像SCons一样.有更好的解决方案吗?

最佳答案 您可以创建一个SCons Builder来完成此任务.我们可以使用标准的Python zipfile来制作zip文件.我们利用zipfile.write,它允许我们指定要添加的文件,以及它应该在zip中调用的内容:

zf.write('foo/bar', 'bar') # save foo/bar as bar

为了获得正确的路径,我们使用os.path.relpath和基本文件的路径来查找整个文件的路径.

最后,我们使用os.walk来遍历我们想要添加的目录的内容,并调用前两个函数将它们正确地添加到最终的zip中.

import os.path
import zipfile

def zipbetter(target, source, env):
    # Open the zip file with appending, so multiple calls will add more files
    zf = zipfile.ZipFile(str(target[0]), 'a', zipfile.ZIP_DEFLATED)
    for s in source:
        # Find the path of the base file
        basedir = os.path.dirname(str(s))
        if s.isdir():
            # If the source is a directory, walk through its files
            for dirpath, dirnames, filenames in os.walk(str(s)):
                for fname in filenames:
                    path = os.path.join(dirpath, fname)
                    if os.path.isfile(path):
                        # If this is a file, write it with its relative path
                        zf.write(path, os.path.relpath(path, basedir))
        else:
            # Otherwise, just write it to the file
            flatname = os.path.basename(str(s))
            zf.write(str(s), flatname)
    zf.close()

# Make a builder using the zipbetter function, that takes SCons files
zipbetter_bld = Builder(action = zipbetter,
                        target_factory = SCons.Node.FS.default_fs.Entry,
                        source_factory = SCons.Node.FS.default_fs.Entry)

# Add the builder to the environment
env.Append(BUILDERS = {'ZipBetter' : zipbetter_bld})

像普通的SCons Zip一样调用它:

env.ZipBetter('foo.zip', 'foo/')
点赞