File: zip_extra.go

package info (click to toggle)
gitlab-ci-multi-runner 14.10.1-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 31,248 kB
  • sloc: sh: 1,694; makefile: 384; asm: 79; ruby: 68
file content (122 lines) | stat: -rw-r--r-- 2,311 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
package archives

import (
	"archive/zip"
	"bytes"
	"encoding/binary"
	"io"
	"os"
	"time"
)

const ZipUIDGidFieldType = 0x7875
const ZipTimestampFieldType = 0x5455

// ZipExtraField is taken from https://github.com/LuaDist/zip/blob/3.0/proginfo/extrafld.txt
type ZipExtraField struct {
	Type uint16
	Size uint16
}

type ZipUIDGidField struct {
	Version uint8
	UIDSize uint8
	UID     uint32
	GIDSize uint8
	Gid     uint32
}

type ZipTimestampField struct {
	Flags   uint8
	ModTime uint32
}

func createZipTimestampField(w io.Writer, fi os.FileInfo) (err error) {
	tsField := ZipTimestampField{
		1,
		uint32(fi.ModTime().Unix()),
	}
	tsFieldType := ZipExtraField{
		Type: ZipTimestampFieldType,
		Size: uint16(binary.Size(&tsField)),
	}
	err = binary.Write(w, binary.LittleEndian, &tsFieldType)
	if err == nil {
		err = binary.Write(w, binary.LittleEndian, &tsField)
	}
	return
}

func processZipTimestampField(data []byte, file *zip.FileHeader) error {
	if !file.Mode().IsDir() && !file.Mode().IsRegular() {
		return nil
	}

	var tsField ZipTimestampField
	err := binary.Read(bytes.NewReader(data), binary.LittleEndian, &tsField)
	if err != nil {
		return err
	}

	if (tsField.Flags & 1) == 1 {
		modTime := time.Unix(int64(tsField.ModTime), 0)
		acTime := time.Now()
		return os.Chtimes(file.Name, acTime, modTime)
	}

	return nil
}

func createZipExtra(fi os.FileInfo) []byte {
	var buffer bytes.Buffer
	err := createZipUIDGidField(&buffer, fi)
	if err == nil {
		err = createZipTimestampField(&buffer, fi)
	}
	if err == nil {
		return buffer.Bytes()
	}
	return nil
}

func readZipExtraField(r io.Reader) (field ZipExtraField, data []byte, err error) {
	err = binary.Read(r, binary.LittleEndian, &field)
	if err != nil {
		return
	}

	data = make([]byte, field.Size)
	_, err = r.Read(data)
	if err != nil {
		return
	}
	return
}

func processZipExtra(file *zip.FileHeader) error {
	if len(file.Extra) == 0 {
		return nil
	}

	r := bytes.NewReader(file.Extra)
	for {
		field, data, err := readZipExtraField(r)
		if err == io.EOF {
			break
		} else if err != nil {
			return err
		}

		switch field.Type {
		case ZipUIDGidFieldType:
			err = processZipUIDGidField(data, file)
		case ZipTimestampFieldType:
			err = processZipTimestampField(data, file)
		}
		if err != nil {
			return err
		}
	}

	return nil
}