Allow specification of map CPLs by ID (#2702).
authorCarl Hetherington <cth@carlh.net>
Thu, 4 Jan 2024 00:21:10 +0000 (01:21 +0100)
committerCarl Hetherington <cth@carlh.net>
Thu, 4 Jan 2024 00:21:10 +0000 (01:21 +0100)
src/lib/map_cli.cc
test/map_cli_test.cc

index e38af69cf08e9773dba3d7b00b18683c06ad434b..b158499a8d204628a87ddeb275d102d5f9556242 100644 (file)
@@ -56,7 +56,7 @@ using boost::optional;
 static void
 help(std::function<void (string)> out)
 {
-       out(String::compose("Syntax: %1 [OPTION} <cpl-file> [<cpl-file> ... ]", program_name));
+       out(String::compose("Syntax: %1 [OPTION} <cpl-file|ID> [<cpl-file|ID> ... ]", program_name));
        out("  -V, --version    show libdcp version");
        out("  -h, --help       show this help");
        out("  -o, --output     output directory");
@@ -138,12 +138,12 @@ map_cli(int argc, char* argv[], std::function<void (string)> out)
                State::override_path = *config_dir;
        }
 
-       vector<boost::filesystem::path> cpl_filenames;
+       vector<string> cpl_filenames_or_ids;
        for (int i = optind; i < argc; ++i) {
-               cpl_filenames.push_back(argv[i]);
+               cpl_filenames_or_ids.push_back(argv[i]);
        }
 
-       if (cpl_filenames.empty()) {
+       if (cpl_filenames_or_ids.empty()) {
                return string{"No CPL specified."};
        }
 
@@ -181,13 +181,28 @@ map_cli(int argc, char* argv[], std::function<void (string)> out)
 
        /* Find all the CPLs */
        vector<shared_ptr<dcp::CPL>> cpls;
-       for (auto filename: cpl_filenames) {
-               try {
-                       auto cpl = make_shared<dcp::CPL>(filename);
-                       cpl->resolve_refs(assets);
-                       cpls.push_back(cpl);
-               } catch (std::exception& e) {
-                       return String::compose("Could not read CPL %1: %2", filename, e.what());
+       for (auto filename_or_id: cpl_filenames_or_ids) {
+               if (boost::filesystem::exists(filename_or_id)) {
+                       try {
+                               auto cpl = make_shared<dcp::CPL>(filename_or_id);
+                               cpl->resolve_refs(assets);
+                               cpls.push_back(cpl);
+                       } catch (std::exception& e) {
+                               return String::compose("Could not read CPL %1: %2", filename_or_id, e.what());
+                       }
+               } else {
+                       auto cpl_iter = std::find_if(assets.begin(), assets.end(), [filename_or_id](shared_ptr<dcp::Asset> asset) {
+                               return asset->id() == filename_or_id;
+                       });
+                       if (cpl_iter == assets.end()) {
+                               return String::compose("Could not find CPL with ID %1", filename_or_id);
+                       }
+                       if (auto cpl = dynamic_pointer_cast<dcp::CPL>(*cpl_iter)) {
+                               cpl->resolve_refs(assets);
+                               cpls.push_back(cpl);
+                       } else {
+                               return String::compose("Could not find CPL with ID %1", filename_or_id);
+                       }
                }
        }
 
index 5505a197bbd1e50aec5c312ca3493f8ef587caba..5a774b6a44ccb26f046473e4953237b05df31584 100644 (file)
@@ -116,6 +116,38 @@ BOOST_AUTO_TEST_CASE(map_simple_dcp_copy)
 }
 
 
+/** Map a single DCP into a new DCP, referring to the CPL by ID */
+BOOST_AUTO_TEST_CASE(map_simple_dcp_copy_by_id)
+{
+       string const name = "map_simple_dcp_copy_by_id";
+       string const out = String::compose("build/test/%1_out", name);
+
+       auto content = content_factory("test/data/flat_red.png");
+       auto film = new_test_film2(name + "_in", content);
+       make_and_verify_dcp(film);
+
+       dcp::CPL cpl(find_cpl(film->dir(film->dcp_name())));
+
+       vector<string> const args = {
+               "map_cli",
+               "-o", out,
+               "-d", film->dir(film->dcp_name()).string(),
+               cpl.id()
+       };
+
+       boost::filesystem::remove_all(out);
+
+       vector<string> output_messages;
+       auto error = run(args, output_messages);
+       BOOST_CHECK(!error);
+
+       verify_dcp(out, {});
+
+       BOOST_CHECK(boost::filesystem::is_regular_file(find_prefix(out, "j2c_")));
+       BOOST_CHECK(boost::filesystem::is_regular_file(find_prefix(out, "pcm_")));
+}
+
+
 /** Map a single DCP into a new DCP using the symlink option */
 BOOST_AUTO_TEST_CASE(map_simple_dcp_copy_with_symlinks)
 {