Verify a whole bunch of DCPs made by unit tests.
[dcpomatic.git] / test / import_dcp_test.cc
1 /*
2     Copyright (C) 2014-2021 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
22 /** @file  test/import_dcp_test.cc
23  *  @brief Test import of encrypted DCPs.
24  *  @ingroup feature
25  */
26
27
28 #include "test.h"
29 #include "lib/film.h"
30 #include "lib/screen.h"
31 #include "lib/dcp_subtitle_content.h"
32 #include "lib/ratio.h"
33 #include "lib/dcp_content_type.h"
34 #include "lib/dcp_content.h"
35 #include "lib/ffmpeg_content.h"
36 #include "lib/examine_content_job.h"
37 #include "lib/job_manager.h"
38 #include "lib/config.h"
39 #include "lib/cross.h"
40 #include "lib/video_content.h"
41 #include "lib/content_factory.h"
42 #include <dcp/cpl.h>
43 #include <boost/test/unit_test.hpp>
44
45
46 using std::vector;
47 using std::string;
48 using std::map;
49 using std::shared_ptr;
50 using std::dynamic_pointer_cast;
51 using std::make_shared;
52
53
54 /** Make an encrypted DCP, import it and make a new unencrypted DCP */
55 BOOST_AUTO_TEST_CASE (import_dcp_test)
56 {
57         auto A = new_test_film ("import_dcp_test");
58         A->set_container (Ratio::from_id ("185"));
59         A->set_dcp_content_type (DCPContentType::from_isdcf_name ("TLR"));
60         A->set_name ("frobozz");
61         A->set_interop (false);
62
63         auto c = make_shared<FFmpegContent>("test/data/test.mp4");
64         A->examine_and_add_content (c);
65         A->set_encrypted (true);
66         BOOST_CHECK (!wait_for_jobs ());
67
68         A->make_dcp ();
69         BOOST_CHECK (!wait_for_jobs ());
70
71         dcp::DCP A_dcp ("build/test/import_dcp_test/" + A->dcp_name());
72         A_dcp.read ();
73
74         Config::instance()->set_decryption_chain (make_shared<dcp::CertificateChain>(openssl_path()));
75
76         /* Dear future-carl: I suck!  I thought you wouldn't still be running these tests in 2030!  Sorry! */
77         auto kdm = A->make_kdm (
78                 Config::instance()->decryption_chain()->leaf (),
79                 vector<string>(),
80                 A_dcp.cpls().front()->file().get(),
81                 dcp::LocalTime ("2030-07-21T00:00:00+00:00"),
82                 dcp::LocalTime ("2031-07-21T00:00:00+00:00"),
83                 dcp::Formulation::MODIFIED_TRANSITIONAL_1,
84                 true, 0
85                 );
86
87         auto B = new_test_film ("import_dcp_test2");
88         B->set_container (Ratio::from_id ("185"));
89         B->set_dcp_content_type (DCPContentType::from_isdcf_name ("TLR"));
90         B->set_name ("frobozz");
91         B->set_interop (false);
92
93         auto d = make_shared<DCPContent>("build/test/import_dcp_test/" + A->dcp_name());
94         B->examine_and_add_content (d);
95         BOOST_CHECK (!wait_for_jobs ());
96         d->add_kdm (kdm);
97         JobManager::instance()->add (make_shared<ExamineContentJob>(B, d));
98         BOOST_CHECK (!wait_for_jobs ());
99
100         B->make_dcp ();
101         BOOST_CHECK (!wait_for_jobs ());
102
103         /* Should be 1s red, 1s green, 1s blue */
104         check_dcp ("test/data/import_dcp_test2", "build/test/import_dcp_test2/" + B->dcp_name());
105 }
106
107
108 /** Check that DCP markers are imported correctly */
109 BOOST_AUTO_TEST_CASE (import_dcp_markers_test)
110 {
111         Cleanup cl;
112
113         /* Make a DCP with some markers */
114         auto content = content_factory("test/data/flat_red.png").front();
115         auto film = new_test_film2 ("import_dcp_markers_test", {content}, &cl);
116
117         content->video->set_length (24 * 60 * 10);
118
119         film->set_marker(dcp::Marker::FFOC, dcpomatic::DCPTime::from_seconds(1.91));
120         film->set_marker(dcp::Marker::FFMC, dcpomatic::DCPTime::from_seconds(9.4));
121         film->set_marker(dcp::Marker::LFMC, dcpomatic::DCPTime::from_seconds(9.99));
122
123         film->make_dcp ();
124         BOOST_REQUIRE (!wait_for_jobs());
125
126         /* Import the DCP to a new film and check the markers */
127         auto imported = make_shared<DCPContent>(film->dir(film->dcp_name()));
128         auto film2 = new_test_film2 ("import_dcp_markers_test2", {imported}, &cl);
129         film2->write_metadata ();
130
131         /* When import_dcp_markers_test was made a LFOC marker will automatically
132          * have been added.
133          */
134         BOOST_CHECK_EQUAL (imported->markers().size(), 4U);
135
136         auto markers = imported->markers();
137         BOOST_REQUIRE(markers.find(dcp::Marker::FFOC) != markers.end());
138         BOOST_CHECK(markers[dcp::Marker::FFOC] == dcpomatic::ContentTime(184000));
139         BOOST_REQUIRE(markers.find(dcp::Marker::FFMC) != markers.end());
140         BOOST_CHECK(markers[dcp::Marker::FFMC] == dcpomatic::ContentTime(904000));
141         BOOST_REQUIRE(markers.find(dcp::Marker::LFMC) != markers.end());
142         BOOST_CHECK(markers[dcp::Marker::LFMC] == dcpomatic::ContentTime(960000));
143
144         /* Load that film and check that the markers have been loaded */
145         auto film3 = make_shared<Film>(boost::filesystem::path("build/test/import_dcp_markers_test2"));
146         film3->read_metadata ();
147         BOOST_REQUIRE_EQUAL (film3->content().size(), 1U);
148         auto reloaded = dynamic_pointer_cast<DCPContent>(film3->content().front());
149         BOOST_REQUIRE (reloaded);
150
151         BOOST_CHECK_EQUAL (reloaded->markers().size(), 4U);
152
153         markers = reloaded->markers();
154         BOOST_REQUIRE(markers.find(dcp::Marker::FFOC) != markers.end());
155         BOOST_CHECK(markers[dcp::Marker::FFOC] == dcpomatic::ContentTime(184000));
156         BOOST_REQUIRE(markers.find(dcp::Marker::FFMC) != markers.end());
157         BOOST_CHECK(markers[dcp::Marker::FFMC] == dcpomatic::ContentTime(904000));
158         BOOST_REQUIRE(markers.find(dcp::Marker::LFMC) != markers.end());
159         BOOST_CHECK(markers[dcp::Marker::LFMC] == dcpomatic::ContentTime(960000));
160
161         cl.run ();
162 }
163
164
165 /** Check that DCP metadata (ratings and content version) are imported correctly */
166 BOOST_AUTO_TEST_CASE (import_dcp_metadata_test)
167 {
168         /* Make a DCP with some ratings and a content version */
169         auto film = new_test_film2 ("import_dcp_metadata_test");
170         auto content = content_factory("test/data/flat_red.png").front();
171         film->examine_and_add_content (content);
172         BOOST_REQUIRE (!wait_for_jobs());
173
174         content->video->set_length (10);
175
176         vector<dcp::Rating> ratings = { {"BBFC", "15"}, {"MPAA", "NC-17"} };
177         film->set_ratings (ratings);
178
179         vector<string> cv = { "Fred "};
180         film->set_content_versions (cv);
181
182         film->make_dcp ();
183         BOOST_REQUIRE (!wait_for_jobs());
184
185         /* Import the DCP to a new film and check the metadata */
186         auto film2 = new_test_film2 ("import_dcp_metadata_test2");
187         auto imported = make_shared<DCPContent>(film->dir(film->dcp_name()));
188         film2->examine_and_add_content (imported);
189         BOOST_REQUIRE (!wait_for_jobs());
190         film2->write_metadata ();
191
192         BOOST_CHECK (imported->ratings() == ratings);
193         BOOST_CHECK (imported->content_versions() == cv);
194
195         /* Load that film and check that the metadata has been loaded */
196         auto film3 = make_shared<Film>(boost::filesystem::path("build/test/import_dcp_metadata_test2"));
197         film3->read_metadata ();
198         BOOST_REQUIRE_EQUAL (film3->content().size(), 1U);
199         auto reloaded = dynamic_pointer_cast<DCPContent>(film3->content().front());
200         BOOST_REQUIRE (reloaded);
201
202         BOOST_CHECK (reloaded->ratings() == ratings);
203         BOOST_CHECK (reloaded->content_versions() == cv);
204 }
205