10

Go で tar_gz ツールを作成したいと考えています。入力はLinuxコマンドと同じです:

$tar czvf targetFileName inputDirectoryPath

以下のような構造の inputDirectory があるとします。

test [dir]
-- 0.txt
    -- 1 [sub dir]
         -- 1.txt

例: 次のコマンドを使用します。

$tar czvf test.tar.gz test/

テスト ディレクトリ全体を tar および gzip できます。


私の問題は、tar と gz ルートを記述して、テスト ディレクトリ内のすべてのファイルを再帰的に反復し、そのファイルを test.tar.gz ファイルに書き込むことができることです。しかし、test.tar.gz にディレクトリを書き込む方法がわかりません。私のプログラムを実行した後、test.tar.gz ファイルの構造は次のようになります。

0.txt
1.txt

ディレクトリを再帰的に出力 tar.gz ファイルに書き込む方法を教えてください。どうもありがとう。

    package main

    import (
      "fmt"
      "os"
      "io"
      "log"
      "strings"
      "archive/tar"
      "compress/gzip"
    )

    func handleError( _e error ) {
      if _e != nil {
        log.Fatal( _e )
      }
    }

    func TarGzWrite( _path string, tw *tar.Writer, fi os.FileInfo ) {
      fr, err := os.Open( _path )
      handleError( err )
      defer fr.Close()

      h := new( tar.Header )
      h.Name = fi.Name()
      h.Size = fi.Size()
      h.Mode = int64( fi.Mode() )
      h.ModTime = fi.ModTime()

      err = tw.WriteHeader( h )
      handleError( err )

      _, err = io.Copy( tw, fr )
      handleError( err )
    }

    func IterDirectory( dirPath string, tw *tar.Writer ) {
      dir, err := os.Open( dirPath )
      handleError( err )
      defer dir.Close()
      fis, err := dir.Readdir( 0 )
      handleError( err )
      for _, fi := range fis {
        curPath := dirPath + "/" + fi.Name()
        if fi.IsDir() {
          //TarGzWrite( curPath, tw, fi )
          IterDirectory( curPath, tw )
        } else {
          fmt.Printf( "adding... %s\n", curPath )
          TarGzWrite( curPath, tw, fi )
        }
      }
    }

    func TarGz( outFilePath string, inPath string ) {
      // file write
      fw, err := os.Create( outFilePath )
      handleError( err )
      defer fw.Close()

      // gzip write
      gw := gzip.NewWriter( fw )
      defer gw.Close()

      // tar write
      tw := tar.NewWriter( gw )
      defer tw.Close()

      IterDirectory( inPath, tw )

      fmt.Println( "tar.gz ok" )
    }

    func main() {
      targetFilePath := "test.tar.gz"
      inputDirPath := "test/"
      TarGz( targetFilePath, strings.TrimRight( inputDirPath, "/" ) )
      fmt.Println( "Hello, World" )
    }
4

2 に答える 2

13

パス全体ではなく、ファイル名のみを tar に追加しています。Tar がディレクトリを理解できるようにするには、パス全体を保持する必要があります。次の 1 行を変更するだけです。

h.Name = fi.Name()

次のようにする必要があります。

h.Name = _path

Linux では、次の出力tar -tvf test.tar.gz:

-rw-rw-r-- 0/0               0 2012-11-28 11:17 test/0.txt
-rw-rw-r-- 0/0               0 2012-11-28 11:17 test/sub/1.txt
于 2012-11-28T18:22:37.817 に答える
6

別の方法は、組み込みの filepath.Walk 関数を使用することです。

// root_directory has been set further up

walkFn := func(path string, info os.FileInfo, err error) error {
    if info.Mode().IsDir() {
        return nil
    }
    // Because of scoping we can reference the external root_directory variable
    new_path := path[len(root_directory):]
    if len(new_path) == 0 {
        return nil
    }
    fr, err := os.Open(path)
    if err != nil {
        return err
    }
    defer fr.Close()

    if h, err := tar.FileInfoHeader(info, new_path); err != nil {
        log.Fatalln(err)
    } else {
        h.Name = new_path
        if err = tw.WriteHeader(h); err != nil {
            log.Fatalln(err)
        }
    }
    if length, err := io.Copy( tw, fr ); err != nil {
        log.Fatalln(err)
    } else {
        fmt.Println(length)
    }
    return nil
}

if err = filepath.Walk(root_directory, walkFn); err != nil {
    return err
}
于 2013-12-09T11:33:28.337 に答える