package compression

import (
	"archive/zip"
	"github.com/druidcaesa/gotool/openfile"
	"io"
	"log"
	"os"
	"strings"
)

type ZipUtils struct {
	f openfile.FileUtils
}

// Compress 压缩文件
//files 文件数组 可以是多目录文件
//dest 压缩文件存放地址
func (z ZipUtils) Compress(files []*os.File, dest string) (bool, error) {
	for _, file := range files {
		//防止用户打开os未关闭,这里调用方法前设置结束关闭
		defer file.Close()
	}
	//判断文件是否存在,存在的话删除
	if z.f.Exists(dest) {
		_, err := z.f.RemoveFile(dest)
		if err != nil {
			log.Fatal(err)
			return false, err
		}
	}
	d, _ := os.Create(dest)
	defer d.Close()
	w := zip.NewWriter(d)
	defer w.Close()
	for _, file := range files {
		err := compress(file, "", w)
		if err != nil {
			return false, err
		}
	}
	return true, nil
}

func compress(file *os.File, prefix string, zw *zip.Writer) error {
	info, err := file.Stat()
	if err != nil {
		return err
	}
	if info.IsDir() {
		prefix = prefix + "/" + info.Name()
		fileInfos, err := file.Readdir(-1)
		if err != nil {
			return err
		}
		for _, fi := range fileInfos {
			f, err := os.Open(file.Name() + "/" + fi.Name())
			if err != nil {
				return err
			}
			err = compress(f, prefix, zw)
			if err != nil {
				return err
			}
		}
	} else {
		header, err := zip.FileInfoHeader(info)
		header.Name = prefix + "/" + header.Name
		if err != nil {
			return err
		}
		writer, err := zw.CreateHeader(header)
		if err != nil {
			return err
		}
		_, err = io.Copy(writer, file)
		file.Close()
		if err != nil {
			return err
		}
	}
	return nil
}

// DeCompress 解压
func (ZipUtils) DeCompress(zipFile, dest string) (bool, error) {
	reader, err := zip.OpenReader(zipFile)
	if err != nil {
		return false, err
	}
	defer reader.Close()
	for _, file := range reader.File {
		rc, err := file.Open()
		if err != nil {
			return false, err
		}
		defer rc.Close()
		filename := dest + file.Name
		err = os.MkdirAll(getDir(filename), 0755)
		if err != nil {
			return false, err
		}
		w, err := os.Create(filename)
		if err != nil {
			return false, err
		}
		defer w.Close()
		_, err = io.Copy(w, rc)
		if err != nil {
			return false, err
		}
		w.Close()
		rc.Close()
	}
	return true, nil
}

func getDir(path string) string {
	return subString(path, 0, strings.LastIndex(path, "/"))
}

func subString(str string, start, end int) string {
	rs := []rune(str)
	length := len(rs)
	if start < 0 || start > length {
		panic("start is wrong")
	}
	if end < start || end > length {
		panic("end is wrong")
	}
	return string(rs[start:end])
}