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