]> Sergey Matveev's repositories - meta4ra.git/blob - cmd/meta4-create/main.go
SHAKE it
[meta4ra.git] / cmd / meta4-create / main.go
1 /*
2 meta4a -- Metalink 4.0 creator
3 Copyright (C) 2021-2023 Sergey Matveev <stargrave@stargrave.org>
4
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation, version 3 of the License.
8
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 GNU General Public License for more details.
13
14 You should have received a copy of the GNU General Public License
15 along with this program.  If not, see <http://www.gnu.org/licenses/>.
16 */
17
18 // Metalink 4.0 creator
19 package main
20
21 import (
22         "bufio"
23         "crypto/sha256"
24         "crypto/sha512"
25         "encoding/hex"
26         "encoding/xml"
27         "flag"
28         "hash"
29         "io"
30         "log"
31         "os"
32         "path"
33         "sync"
34         "time"
35
36         "go.cypherpunks.ru/gogost/v5/gost34112012256"
37         "go.cypherpunks.ru/gogost/v5/gost34112012512"
38         "go.stargrave.org/meta4ra"
39         "golang.org/x/crypto/sha3"
40 )
41
42 type MultiHasher struct {
43         sha256h      hash.Hash
44         sha512h      hash.Hash
45         streebog256h hash.Hash
46         streebog512h hash.Hash
47         shake128h    sha3.ShakeHash
48         shake256h    sha3.ShakeHash
49         g            sync.WaitGroup
50 }
51
52 func NewMultiHasher() *MultiHasher {
53         return &MultiHasher{
54                 sha256h:      sha256.New(),
55                 sha512h:      sha512.New(),
56                 streebog256h: gost34112012256.New(),
57                 streebog512h: gost34112012512.New(),
58                 shake128h:    sha3.NewShake128(),
59                 shake256h:    sha3.NewShake256(),
60         }
61 }
62
63 func (h *MultiHasher) Write(p []byte) (n int, err error) {
64         h.g.Add(6)
65         go func() {
66                 if _, err := h.sha256h.Write(p); err != nil {
67                         panic(err)
68                 }
69                 h.g.Done()
70         }()
71         go func() {
72                 if _, err := h.sha512h.Write(p); err != nil {
73                         panic(err)
74                 }
75                 h.g.Done()
76         }()
77         go func() {
78                 if _, err := h.streebog256h.Write(p); err != nil {
79                         panic(err)
80                 }
81                 h.g.Done()
82         }()
83         go func() {
84                 if _, err := h.streebog512h.Write(p); err != nil {
85                         panic(err)
86                 }
87                 h.g.Done()
88         }()
89         go func() {
90                 if _, err := h.shake128h.Write(p); err != nil {
91                         panic(err)
92                 }
93                 h.g.Done()
94         }()
95         go func() {
96                 if _, err := h.shake256h.Write(p); err != nil {
97                         panic(err)
98                 }
99                 h.g.Done()
100         }()
101         h.g.Wait()
102         return len(p), nil
103 }
104
105 func main() {
106         fn := flag.String("fn", "", "Filename")
107         mtime := flag.String("mtime", "", "Take that file's mtime as a Published date")
108         desc := flag.String("desc", "", "Description")
109         sig := flag.String("sig", "", "Path to signature file")
110         torrent := flag.String("torrent", "", "Torrent URL")
111         log.SetFlags(log.Lshortfile)
112         flag.Parse()
113         if *fn == "" {
114                 log.Fatalln("empty -fn")
115         }
116         urls := make([]meta4ra.URL, 0, len(flag.Args()))
117         for _, u := range flag.Args() {
118                 urls = append(urls, meta4ra.URL{URL: u})
119         }
120         br := bufio.NewReaderSize(os.Stdin, 1<<20)
121         buf := make([]byte, 1<<20)
122         h := NewMultiHasher()
123         size, err := io.CopyBuffer(h, br, buf)
124         if err != nil {
125                 log.Fatalln(err)
126         }
127         shake128 := make([]byte, 32)
128         if _, err = io.ReadFull(h.shake128h, shake128); err != nil {
129                 panic(err)
130         }
131         shake256 := make([]byte, 64)
132         if _, err = io.ReadFull(h.shake256h, shake256); err != nil {
133                 panic(err)
134         }
135         f := meta4ra.File{
136                 Name:        path.Base(*fn),
137                 Description: *desc,
138                 Size:        uint64(size),
139                 URLs:        urls,
140                 Hashes: []meta4ra.Hash{
141                         {Type: meta4ra.HashSHA256, Hash: hex.EncodeToString(h.sha256h.Sum(nil))},
142                         {Type: meta4ra.HashSHA512, Hash: hex.EncodeToString(h.sha512h.Sum(nil))},
143                         {Type: meta4ra.HashStreebog256, Hash: hex.EncodeToString(h.streebog256h.Sum(nil))},
144                         {Type: meta4ra.HashStreebog512, Hash: hex.EncodeToString(h.streebog512h.Sum(nil))},
145                         {Type: meta4ra.HashSHAKE128, Hash: hex.EncodeToString(shake128)},
146                         {Type: meta4ra.HashSHAKE256, Hash: hex.EncodeToString(shake256)},
147                 },
148         }
149         if *sig != "" {
150                 sigData, err := os.ReadFile(*sig)
151                 if err != nil {
152                         log.Fatalln(err)
153                 }
154                 f.Signature = &meta4ra.Signature{
155                         MediaType: meta4ra.GPGSigMediaType,
156                         Signature: "\n" + string(sigData),
157                 }
158         }
159         if *torrent != "" {
160                 f.MetaURLs = []meta4ra.MetaURL{{MediaType: "torrent", URL: *torrent}}
161         }
162         var published time.Time
163         if *mtime == "" {
164                 published = time.Now()
165         } else {
166                 fi, err := os.Stat(*mtime)
167                 if err != nil {
168                         log.Fatalln(err)
169                 }
170                 published = fi.ModTime()
171         }
172         published = published.UTC().Truncate(time.Second)
173         m := meta4ra.Metalink{
174                 Files:     []meta4ra.File{f},
175                 Generator: meta4ra.Generator,
176                 Published: published,
177         }
178         out, err := xml.MarshalIndent(&m, "", "  ")
179         if err != nil {
180                 log.Fatalln(err)
181         }
182         os.Stdout.Write([]byte(xml.Header))
183         os.Stdout.Write(out)
184         os.Stdout.Write([]byte("\n"))
185 }