Golang 1.2:解压缩密码保护的zip文件?

查看最新版本(1.2)zip包 - 如何解压缩受密码保护的文件(使用7zip,AES-256编码)? 我没有看到在那里/如何添加这些信息。 一个简单的例子会很棒!


archive/zip包似乎只提供基本的zip功能。 我会使用7zip解压缩使用os/exec软件包的受密码保护的zip文件。

在线7-zip用户指南

了解7zip的最佳指南是7-zip.chm,它位于windows命令行的zip文件中。

下面的代码不是最优的,但它会告诉你如何完成工作。

使用7zip提取受密码保护的zip的代码

func extractZipWithPassword() {
    fmt.Printf("Unzipping `%s` to directory `%s`n", zip_path, extract_path)
    commandString := fmt.Sprintf(`7za e %s -o%s -p"%s" -aoa`, zip_path, extract_path, zip_password)
    commandSlice := strings.Fields(commandString)
    fmt.Println(commandString)
    c := exec.Command(commandSlice[0], commandSlice[1:]...)
    e := c.Run()
    checkError(e)
}

示例程序

// Shows how to extract an passsword encrypted zip file using 7zip.
// By Larry Battle <https://github.com/LarryBattle>
// Answer to http://stackoverflow.com/questions/20330210/golang-1-2-unzip-password-protected-zip-file
// 7-zip.chm - http://sevenzip.sourceforge.jp/chm/cmdline/switches/index.htm
// Effective Golang - http://golang.org/doc/effective_go.html
package main

import (
    "fmt"
    "os"
    "os/exec"
    "path/filepath"
    "strings"
)

var (
    txt_content     = "Sample file created."
    txt_filename    = "name.txt"
    zip_filename    = "sample.zip"
    zip_password    = "42"
    zip_encryptType = "AES256"
    base_path       = "./"

    test_path          = filepath.Join(base_path, "test")
    src_path           = filepath.Join(test_path, "src")
    extract_path       = filepath.Join(test_path, "extracted")
    extracted_txt_path = filepath.Join(extract_path, txt_filename)
    txt_path           = filepath.Join(src_path, txt_filename)
    zip_path           = filepath.Join(src_path, zip_filename)
)
var txt_fileSize int64

func checkError(e error) {
    if e != nil {
        panic(e)
    }
}
func setupTestDir() {
    fmt.Printf("Removing `%s`n", test_path)
    var e error
    os.Remove(test_path)
    fmt.Printf("Creating `%s`,`%s`n", extract_path, src_path)
    e = os.MkdirAll(src_path, os.ModeDir|os.ModePerm)
    checkError(e)
    e = os.MkdirAll(extract_path, os.ModeDir|os.ModePerm)
    checkError(e)
}
func createSampleFile() {
    fmt.Println("Creating", txt_path)
    file, e := os.Create(txt_path)
    checkError(e)
    defer file.Close()
    _, e = file.WriteString(txt_content)
    checkError(e)
    fi, e := file.Stat()
    txt_fileSize = fi.Size()
}
func createZipWithPassword() {
    fmt.Println("Creating", zip_path)
    commandString := fmt.Sprintf(`7za a %s %s -p"%s" -mem=%s`, zip_path, txt_path, zip_password, zip_encryptType)
    commandSlice := strings.Fields(commandString)
    fmt.Println(commandString)
    c := exec.Command(commandSlice[0], commandSlice[1:]...)
    e := c.Run()
    checkError(e)
}
func extractZipWithPassword() {
    fmt.Printf("Unzipping `%s` to directory `%s`n", zip_path, extract_path)
    commandString := fmt.Sprintf(`7za e %s -o%s -p"%s" -aoa`, zip_path, extract_path, zip_password)
    commandSlice := strings.Fields(commandString)
    fmt.Println(commandString)
    c := exec.Command(commandSlice[0], commandSlice[1:]...)
    e := c.Run()
    checkError(e)
}
func checkFor7Zip() {
    _, e := exec.LookPath("7za")
    if e != nil {
        fmt.Println("Make sure 7zip is install and include your path.")
    }
    checkError(e)
}
func checkExtractedFile() {
    fmt.Println("Reading", extracted_txt_path)
    file, e := os.Open(extracted_txt_path)
    checkError(e)
    defer file.Close()
    buf := make([]byte, txt_fileSize)
    n, e := file.Read(buf)
    checkError(e)
    if !strings.Contains(string(buf[:n]), strings.Fields(txt_content)[0]) {
        panic(fmt.Sprintf("File`%s` is corrupted.n", extracted_txt_path))
    }
}
func main() {
    fmt.Println("# Setup")
    checkFor7Zip()
    setupTestDir()
    createSampleFile()
    createZipWithPassword()
    fmt.Println("# Answer to question...")
    extractZipWithPassword()
    checkExtractedFile()
    fmt.Println("Done.")
}

产量

# Setup
Removing `test`
Creating `test/extracted`,`test/src`
Creating test/src/name.txt
Creating test/src/sample.zip
7za a test/src/sample.zip test/src/name.txt -p"42" -mem=AES256
# Answer to question...
Unzipping `test/src/sample.zip` to directory `test/extracted`
7za e test/src/sample.zip -otest/extracted -p"42" -aoa
Reading test/extracted/name.txt
Done.

如果其他人遇到这个提取失败并出现密码错误,请尝试删除引号。 在我的情况下,他们正在逃走,并导致提取失败。


https://github.com/yeka/zip提供了提取受密码保护的zip文件(AES&Zip标准加密aka ZipCrypto)的功能。

下面是一个如何使用它的例子:

package main

import (
    "os"
    "io"
    "github.com/yeka/zip"
)

func main() {
    file := "file.zip"
    password := "password"
    r, err := zip.OpenReader(file)
    if nil != err {
        panic(err)
    }
    defer r.Close()

    for _, f := range r.File {
        f.SetPassword(password)
        w, err := os.Create(f.Name)
        if nil != err {
            panic(err)
        }
        io.Copy(w, f)
        w.Close()
    }
}

这项工作是来自https://github.com/alexmullins/zip的一个分支,它只添加对AES的支持。

链接地址: http://www.djcxy.com/p/75931.html

上一篇: Golang 1.2: Unzip password protected zip file?

下一篇: native utility gone in cygwin 1.7.26 for 64bit windows?