]> Cypherpunks.ru repositories - nncp.git/blob - src/cypherpunks.ru/nncp/cmd/nncp-file/main.go
43259146071cd558e3e4e99df367e65ec751354a
[nncp.git] / src / cypherpunks.ru / nncp / cmd / nncp-file / main.go
1 /*
2 NNCP -- Node to Node copy, utilities for store-and-forward data exchange
3 Copyright (C) 2016-2018 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, either version 3 of the License, or
8 (at your option) any later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program.  If not, see <http://www.gnu.org/licenses/>.
17 */
18
19 // Send file via NNCP
20 package main
21
22 import (
23         "flag"
24         "fmt"
25         "log"
26         "os"
27         "strings"
28
29         "cypherpunks.ru/nncp"
30 )
31
32 func usage() {
33         fmt.Fprintf(os.Stderr, nncp.UsageHeader())
34         fmt.Fprintf(os.Stderr, "nncp-file -- send file\n\n")
35         fmt.Fprintf(os.Stderr, "Usage: %s [options] SRC NODE:[DST]\nOptions:\n", os.Args[0])
36         flag.PrintDefaults()
37         fmt.Fprint(os.Stderr, `
38 If SRC equals to -, then read data from stdin to temporary file.
39
40 -minsize/-chunked take NODE's FreqMinSize/FreqChunked configuration
41 options by default. You can forcefully turn them off by specifying 0 value.
42 `)
43 }
44
45 func main() {
46         var (
47                 cfgPath      = flag.String("cfg", nncp.DefaultCfgPath, "Path to configuration file")
48                 niceRaw      = flag.Int("nice", nncp.DefaultNiceFile, "Outbound packet niceness")
49                 argMinSize   = flag.Int64("minsize", -1, "Minimal required resulting packet size, in KiB")
50                 argChunkSize = flag.Int64("chunked", -1, "Split file on specified size chunks, in KiB")
51                 viaOverride  = flag.String("via", "", "Override Via path to destination node")
52                 spoolPath    = flag.String("spool", "", "Override path to spool")
53                 logPath      = flag.String("log", "", "Override path to logfile")
54                 quiet        = flag.Bool("quiet", false, "Print only errors")
55                 debug        = flag.Bool("debug", false, "Print debug messages")
56                 version      = flag.Bool("version", false, "Print version information")
57                 warranty     = flag.Bool("warranty", false, "Print warranty information")
58         )
59         flag.Usage = usage
60         flag.Parse()
61         if *warranty {
62                 fmt.Println(nncp.Warranty)
63                 return
64         }
65         if *version {
66                 fmt.Println(nncp.VersionGet())
67                 return
68         }
69         if flag.NArg() != 2 {
70                 usage()
71                 os.Exit(1)
72         }
73         if *niceRaw < 1 || *niceRaw > 255 {
74                 log.Fatalln("-nice must be between 1 and 255")
75         }
76         nice := uint8(*niceRaw)
77
78         ctx, err := nncp.CtxFromCmdline(*cfgPath, *spoolPath, *logPath, *quiet, *debug)
79         if err != nil {
80                 log.Fatalln("Error during initialization:", err)
81         }
82         if ctx.Self == nil {
83                 log.Fatalln("Config lacks private keys")
84         }
85
86         splitted := strings.SplitN(flag.Arg(1), ":", 2)
87         if len(splitted) != 2 {
88                 usage()
89                 os.Exit(1)
90         }
91         node, err := ctx.FindNode(splitted[0])
92         if err != nil {
93                 log.Fatalln("Invalid NODE specified:", err)
94         }
95
96         if *viaOverride != "" {
97                 vias := make([]*nncp.NodeId, 0, strings.Count(*viaOverride, ",")+1)
98                 for _, via := range strings.Split(*viaOverride, ",") {
99                         foundNodeId, err := ctx.FindNode(via)
100                         if err != nil {
101                                 log.Fatalln("Invalid Via node specified:", err)
102                         }
103                         vias = append(vias, foundNodeId.Id)
104                 }
105                 node.Via = vias
106         }
107
108         var minSize int64
109         if *argMinSize < 0 {
110                 minSize = node.FreqMinSize
111         } else if *argMinSize > 0 {
112                 minSize = *argMinSize * 1024
113         }
114
115         var chunkSize int64
116         if *argChunkSize < 0 {
117                 chunkSize = node.FreqChunked
118         } else if *argChunkSize > 0 {
119                 chunkSize = *argChunkSize * 1024
120         }
121
122         if chunkSize == 0 {
123                 err = ctx.TxFile(
124                         node,
125                         nice,
126                         flag.Arg(0),
127                         splitted[1],
128                         minSize,
129                 )
130         } else {
131                 err = ctx.TxFileChunked(
132                         node,
133                         nice,
134                         flag.Arg(0),
135                         splitted[1],
136                         minSize,
137                         chunkSize,
138                 )
139         }
140         if err != nil {
141                 log.Fatalln(err)
142         }
143 }