]> Sergey Matveev's repositories - meta4ra.git/blob - cmd/meta4-create/main.go
ea0a776086014ce50817dc0431742e4e1b8ee942
[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         "sync"
33         "time"
34
35         "go.cypherpunks.ru/gogost/v5/gost34112012256"
36         "go.cypherpunks.ru/gogost/v5/gost34112012512"
37         "go.stargrave.org/meta4ra"
38 )
39
40 type MultiHasher struct {
41         sha256h      hash.Hash
42         sha512h      hash.Hash
43         streebog256h hash.Hash
44         streebog512h hash.Hash
45         g            sync.WaitGroup
46 }
47
48 func NewMultiHasher() *MultiHasher {
49         return &MultiHasher{
50                 sha256h:      sha256.New(),
51                 sha512h:      sha512.New(),
52                 streebog256h: gost34112012256.New(),
53                 streebog512h: gost34112012512.New(),
54         }
55 }
56
57 func (h *MultiHasher) Write(p []byte) (n int, err error) {
58         h.g.Add(4)
59         go func() {
60                 if _, err := h.sha256h.Write(p); err != nil {
61                         panic(err)
62                 }
63                 h.g.Done()
64         }()
65         go func() {
66                 if _, err := h.sha512h.Write(p); err != nil {
67                         panic(err)
68                 }
69                 h.g.Done()
70         }()
71         go func() {
72                 if _, err := h.streebog256h.Write(p); err != nil {
73                         panic(err)
74                 }
75                 h.g.Done()
76         }()
77         go func() {
78                 if _, err := h.streebog512h.Write(p); err != nil {
79                         panic(err)
80                 }
81                 h.g.Done()
82         }()
83         h.g.Wait()
84         return len(p), nil
85 }
86
87 func main() {
88         fn := flag.String("fn", "", "Filename")
89         mtime := flag.String("mtime", "", "Take that file's mtime as a Published date")
90         desc := flag.String("desc", "", "Description")
91         sig := flag.String("sig", "", "Path to signature file")
92         torrent := flag.String("torrent", "", "Torrent URL")
93         log.SetFlags(log.Lshortfile)
94         flag.Parse()
95         if *fn == "" {
96                 log.Fatalln("empty -fn")
97         }
98         urls := make([]meta4ra.URL, 0, len(flag.Args()))
99         for _, u := range flag.Args() {
100                 urls = append(urls, meta4ra.URL{URL: u})
101         }
102         br := bufio.NewReaderSize(os.Stdin, 1<<20)
103         buf := make([]byte, 1<<20)
104         h := NewMultiHasher()
105         size, err := io.CopyBuffer(h, br, buf)
106         if err != nil {
107                 log.Fatalln(err)
108         }
109         f := meta4ra.File{
110                 Name:        *fn,
111                 Description: *desc,
112                 Size:        uint64(size),
113                 URLs:        urls,
114                 Hashes: []meta4ra.Hash{
115                         {Type: meta4ra.HashSHA256, Hash: hex.EncodeToString(h.sha256h.Sum(nil))},
116                         {Type: meta4ra.HashSHA512, Hash: hex.EncodeToString(h.sha512h.Sum(nil))},
117                         {Type: meta4ra.HashStreebog256, Hash: hex.EncodeToString(h.streebog256h.Sum(nil))},
118                         {Type: meta4ra.HashStreebog512, Hash: hex.EncodeToString(h.streebog512h.Sum(nil))},
119                 },
120         }
121         if *sig != "" {
122                 sigData, err := os.ReadFile(*sig)
123                 if err != nil {
124                         log.Fatalln(err)
125                 }
126                 f.Signature = &meta4ra.Signature{
127                         MediaType: meta4ra.GPGSigMediaType,
128                         Signature: "\n" + string(sigData),
129                 }
130         }
131         if *torrent != "" {
132                 f.MetaURLs = []meta4ra.MetaURL{{MediaType: "torrent", URL: *torrent}}
133         }
134         var published time.Time
135         if *mtime == "" {
136                 published = time.Now()
137         } else {
138                 fi, err := os.Stat(*mtime)
139                 if err != nil {
140                         log.Fatalln(err)
141                 }
142                 published = fi.ModTime()
143         }
144         published = published.UTC().Truncate(time.Second)
145         m := meta4ra.Metalink{
146                 Files:     []meta4ra.File{f},
147                 Generator: meta4ra.Generator,
148                 Published: published,
149         }
150         out, err := xml.MarshalIndent(&m, "", "  ")
151         if err != nil {
152                 log.Fatalln(err)
153         }
154         os.Stdout.Write([]byte(xml.Header))
155         os.Stdout.Write(out)
156         os.Stdout.Write([]byte("\n"))
157 }