Rework command-line parser for dcpomatic_create to allow options
[dcpomatic.git] / src / lib / create_cli.cc
1 /*
2     Copyright (C) 2019 Carl Hetherington <cth@carlh.net>
3
4     This file is part of DCP-o-matic.
5
6     DCP-o-matic is free software; you can redistribute it and/or modify
7     it under the terms of the GNU General Public License as published by
8     the Free Software Foundation; either version 2 of the License, or
9     (at your option) any later version.
10
11     DCP-o-matic is distributed in the hope that it will be useful,
12     but WITHOUT ANY WARRANTY; without even the implied warranty of
13     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14     GNU General Public License for more details.
15
16     You should have received a copy of the GNU General Public License
17     along with DCP-o-matic.  If not, see <http://www.gnu.org/licenses/>.
18
19 */
20
21 #include "create_cli.h"
22 #include "dcp_content_type.h"
23 #include "ratio.h"
24 #include "compose.hpp"
25 #include <dcp/raw_convert.h>
26 #include <string>
27
28 #include <iostream>
29
30 using std::string;
31 using std::cout;
32 using boost::optional;
33
34 string CreateCLI::_help =
35         "\nSyntax: %1 [OPTION] <CONTENT> [OPTION] [<CONTENT> ...]\n"
36         "  -v, --version                 show DCP-o-matic version\n"
37         "  -h, --help                    show this help\n"
38         "  -n, --name <name>             film name\n"
39         "  -t, --template <name>         template name\n"
40         "  -e, --encrypt                 make an encrypted DCP\n"
41         "  -c, --dcp-content-type <type> FTR, SHR, TLR, TST, XSN, RTG, TSR, POL, PSA or ADV\n"
42         "  -f, --dcp-frame-rate <rate>   set DCP video frame rate (otherwise guessed from content)\n"
43         "      --container-ratio <ratio> 119, 133, 137, 138, 166, 178, 185 or 239\n"
44         "      --content-ratio <ratio>   119, 133, 137, 138, 166, 178, 185 or 239\n"
45         "  -s, --still-length <n>        number of seconds that still content should last\n"
46         "      --standard <standard>     SMPTE or interop (default SMPTE)\n"
47         "      --no-use-isdcf-name       do not use an ISDCF name; use the specified name unmodified\n"
48         "      --no-sign                 do not sign the DCP\n"
49         "      --config <dir>            directory containing config.xml and cinemas.xml\n"
50         "  -o, --output <dir>            output directory\n"
51         "      --threed                  make a 3D DCP\n"
52         "      --left-eye                next piece of content is for the left eye\n"
53         "      --right-eye               next piece of content is for the right eye\n";
54
55 template <class T>
56 void
57 argument_option (int& n, int argc, char* argv[], string short_name, string long_name, bool* claimed, optional<string>* error, T* out)
58 {
59         string const a = argv[n];
60         if (a != short_name && a != long_name) {
61                 return;
62         }
63
64         if ((n + 1) >= argc) {
65                 **error = String::compose("%1: option %2 requires an argument", argv[0], long_name);
66                 return;
67         }
68
69         *out = dcp::raw_convert<T>(string(argv[++n]));
70         *claimed = true;
71 }
72
73 CreateCLI::CreateCLI (int argc, char* argv[])
74         : version (false)
75         , encrypt (false)
76         , threed (false)
77         , dcp_content_type (0)
78         , dcp_frame_rate (24)
79         , container_ratio (0)
80         , content_ratio (0)
81         , still_length (10)
82         , standard (dcp::SMPTE)
83         , no_use_isdcf_name (false)
84         , no_sign (false)
85 {
86         string dcp_content_type_string = "TST";
87         string content_ratio_string;
88         string container_ratio_string;
89         string standard_string = "SMPTE";
90         int dcp_frame_rate_int = 0;
91         string template_name_string;
92         string config_dir_string;
93         string output_dir_string;
94         VideoFrameType next_frame_type = VIDEO_FRAME_TYPE_2D;
95
96         int i = 1;
97         while (i < argc) {
98                 string const a = argv[i];
99                 bool claimed = false;
100
101                 if (a == "-v" || a == "--version") {
102                         version = true;
103                         return;
104                 } else if (a == "-h" || a == "--help") {
105                         error = "Create a film directory (ready for making a DCP) or metadata file from some content files.\n"
106                                 "A film directory will be created if -o or --output is specified, otherwise a metadata file\n"
107                                 "will be written to stdout.\n" + String::compose(_help, argv[0]);
108                         return;
109                 }
110
111                 if (a == "-e" || a == "--encrypt") {
112                         encrypt = claimed = true;
113                 } else if (a == "--no-use-isdcf-name") {
114                         no_use_isdcf_name = claimed = true;
115                 } else if (a == "--no-sign") {
116                         no_sign = claimed = true;
117                 } else if (a == "--threed") {
118                         threed = claimed = true;
119                 } else if (a == "--left-eye") {
120                         next_frame_type = VIDEO_FRAME_TYPE_3D_LEFT;
121                         claimed = true;
122                 } else if (a == "--right-eye") {
123                         next_frame_type = VIDEO_FRAME_TYPE_3D_RIGHT;
124                         claimed = true;
125                 }
126
127                 argument_option(i, argc, argv, "-n", "--name",             &claimed, &error, &name);
128                 argument_option(i, argc, argv, "-t", "--template",         &claimed, &error, &template_name_string);
129                 argument_option(i, argc, argv, "-c", "--dcp-content-type", &claimed, &error, &dcp_content_type_string);
130                 argument_option(i, argc, argv, "-f", "--dcp-frame-rate",   &claimed, &error, &dcp_frame_rate_int);
131                 argument_option(i, argc, argv, "",   "--container-ratio",  &claimed, &error, &container_ratio_string);
132                 argument_option(i, argc, argv, "",   "--content-ratio",    &claimed, &error, &content_ratio_string);
133                 argument_option(i, argc, argv, "-s", "--still-length",     &claimed, &error, &still_length);
134                 argument_option(i, argc, argv, "",   "--standard",         &claimed, &error, &standard_string);
135                 argument_option(i, argc, argv, "",   "--config",           &claimed, &error, &config_dir_string);
136                 argument_option(i, argc, argv, "-o", "--output",           &claimed, &error, &output_dir_string);
137
138                 if (!claimed) {
139                         if (a.length() > 2 && a.substr(0, 2) == "--") {
140                                 error = String::compose("%1: unrecognised option '%2'", argv[0], a) + String::compose(_help, argv[0]);
141                                 return;
142                         } else {
143                                 Content c;
144                                 c.path = a;
145                                 c.frame_type = next_frame_type;
146                                 content.push_back (c);
147                                 next_frame_type = VIDEO_FRAME_TYPE_2D;
148                         }
149                 }
150
151                 ++i;
152         }
153
154         if (!config_dir_string.empty()) {
155                 config_dir = config_dir_string;
156         }
157
158         if (!output_dir_string.empty()) {
159                 output_dir = output_dir_string;
160         }
161
162         if (!template_name_string.empty()) {
163                 template_name = template_name_string;
164         }
165
166         if (dcp_frame_rate_int) {
167                 dcp_frame_rate = dcp_frame_rate_int;
168         }
169
170         dcp_content_type = DCPContentType::from_isdcf_name(dcp_content_type_string);
171         if (!dcp_content_type) {
172                 error = String::compose("%1: unrecognised DCP content type '%2'", argv[0], dcp_content_type_string);
173                 return;
174         }
175
176         if (content_ratio_string.empty()) {
177                 error = String::compose("%1: missing required option --content-ratio", argv[0]);
178                 return;
179         }
180
181         content_ratio = Ratio::from_id (content_ratio_string);
182         if (!content_ratio) {
183                 error = String::compose("%1: unrecognised content ratio %2", content_ratio_string);
184                 return;
185         }
186
187         if (container_ratio_string.empty()) {
188                 container_ratio_string = content_ratio_string;
189         }
190
191         container_ratio = Ratio::from_id (container_ratio_string);
192         if (!container_ratio) {
193                 error = String::compose("%1: unrecognised container ratio %2", argv[0], container_ratio_string);
194                 return;
195         }
196
197         if (standard_string != "SMPTE" && standard_string != "interop") {
198                 error = String::compose("%1: standard must be SMPTE or interop", argv[0]);
199                 return;
200         }
201
202         if (content.empty()) {
203                 error = String::compose("%1: no content specified", argv[0]);
204                 return;
205         }
206
207         if (name.empty()) {
208                 name = content[0].path.leaf().string();
209         }
210 }