Fill test disk partitions with random noise to expose more bugs.
[dcpomatic.git] / test / hints_test.cc
index aabcaaeace0e8f8376c29495e054f0ea90db2356..ff989647359bfe8e6aaa9357e0b6757f42d90f31 100644 (file)
@@ -1,5 +1,5 @@
 /*
-    Copyright (C) 2020 Carl Hetherington <cth@carlh.net>
+    Copyright (C) 2020-2022 Carl Hetherington <cth@carlh.net>
 
     This file is part of DCP-o-matic.
 
@@ -19,6 +19,8 @@
 */
 
 
+#include "lib/audio_content.h"
+#include "lib/config.h"
 #include "lib/content.h"
 #include "lib/content_factory.h"
 #include "lib/cross.h"
 #include "lib/text_content.h"
 #include "lib/util.h"
 #include "test.h"
-#include <boost/shared_ptr.hpp>
 #include <boost/test/unit_test.hpp>
 
 
+using std::make_shared;
+using std::shared_ptr;
 using std::string;
 using std::vector;
 using boost::optional;
-using boost::shared_ptr;
 
 
 vector<string> current_hints;
@@ -55,10 +57,13 @@ get_hints (shared_ptr<Film> film)
 {
        current_hints.clear ();
        Hints hints (film);
+       /* None of our tests need the audio analysis, and it is quite time-consuming */
+       hints.disable_audio_analysis ();
        hints.Hint.connect (collect_hint);
        hints.start ();
        hints.join ();
        while (signal_manager->ui_idle()) {}
+       hints.rethrow();
        return current_hints;
 }
 
@@ -67,15 +72,16 @@ static
 void
 check (TextType type, string name, optional<string> expected_hint = optional<string>())
 {
-       shared_ptr<Film> film = new_test_film2 (name);
-       shared_ptr<Content> content = content_factory("test/data/" + name + ".srt").front();
+       auto film = new_test_film2 (name);
+       auto content = content_factory("test/data/" + name + ".srt").front();
        content->text.front()->set_type (type);
+       content->text.front()->set_language (dcp::LanguageTag("en-US"));
        film->examine_and_add_content (content);
        BOOST_REQUIRE (!wait_for_jobs());
-       vector<string> hints = get_hints (film);
+       auto hints = get_hints (film);
 
        if (expected_hint) {
-               BOOST_REQUIRE_EQUAL (hints.size(), 1);
+               BOOST_REQUIRE_EQUAL (hints.size(), 1U);
                BOOST_CHECK_EQUAL (hints[0], *expected_hint);
        } else {
                BOOST_CHECK (hints.empty());
@@ -86,7 +92,7 @@ check (TextType type, string name, optional<string> expected_hint = optional<str
 BOOST_AUTO_TEST_CASE (hint_closed_caption_too_long)
 {
        check (
-               TEXT_CLOSED_CAPTION,
+               TextType::CLOSED_CAPTION,
                "hint_closed_caption_too_long",
                String::compose("At least one of your closed caption lines has more than %1 characters.  It is advisable to make each line %1 characters at most in length.", MAX_CLOSED_CAPTION_LENGTH, MAX_CLOSED_CAPTION_LENGTH)
              );
@@ -96,7 +102,7 @@ BOOST_AUTO_TEST_CASE (hint_closed_caption_too_long)
 BOOST_AUTO_TEST_CASE (hint_many_closed_caption_lines)
 {
        check (
-               TEXT_CLOSED_CAPTION,
+               TextType::CLOSED_CAPTION,
                "hint_many_closed_caption_lines",
                String::compose("Some of your closed captions span more than %1 lines, so they will be truncated.", MAX_CLOSED_CAPTION_LINES)
              );
@@ -106,7 +112,7 @@ BOOST_AUTO_TEST_CASE (hint_many_closed_caption_lines)
 BOOST_AUTO_TEST_CASE (hint_subtitle_too_early)
 {
        check (
-               TEXT_OPEN_SUBTITLE,
+               TextType::OPEN_SUBTITLE,
                "hint_subtitle_too_early",
                string("It is advisable to put your first subtitle at least 4 seconds after the start of the DCP to make sure it is seen.")
                );
@@ -116,7 +122,7 @@ BOOST_AUTO_TEST_CASE (hint_subtitle_too_early)
 BOOST_AUTO_TEST_CASE (hint_short_subtitles)
 {
        check (
-               TEXT_OPEN_SUBTITLE,
+               TextType::OPEN_SUBTITLE,
                "hint_short_subtitles",
                string("At least one of your subtitles lasts less than 15 frames.  It is advisable to make each subtitle at least 15 frames long.")
                );
@@ -126,7 +132,7 @@ BOOST_AUTO_TEST_CASE (hint_short_subtitles)
 BOOST_AUTO_TEST_CASE (hint_subtitles_too_close)
 {
        check (
-               TEXT_OPEN_SUBTITLE,
+               TextType::OPEN_SUBTITLE,
                "hint_subtitles_too_close",
                string("At least one of your subtitles starts less than 2 frames after the previous one.  It is advisable to make the gap between subtitles at least 2 frames.")
              );
@@ -136,7 +142,7 @@ BOOST_AUTO_TEST_CASE (hint_subtitles_too_close)
 BOOST_AUTO_TEST_CASE (hint_many_subtitle_lines)
 {
        check (
-               TEXT_OPEN_SUBTITLE,
+               TextType::OPEN_SUBTITLE,
                "hint_many_subtitle_lines",
                string("At least one of your subtitles has more than 3 lines.  It is advisable to use no more than 3 lines.")
              );
@@ -146,9 +152,19 @@ BOOST_AUTO_TEST_CASE (hint_many_subtitle_lines)
 BOOST_AUTO_TEST_CASE (hint_subtitle_too_long)
 {
        check (
-               TEXT_OPEN_SUBTITLE,
+               TextType::OPEN_SUBTITLE,
                "hint_subtitle_too_long",
-               string("At least one of your subtitle lines has more than 52 characters.  It is advisable to make each line 52 characters at most in length.")
+               string("At least one of your subtitle lines has more than 52 characters.  It is recommended to make each line 52 characters at most in length.")
+             );
+}
+
+
+BOOST_AUTO_TEST_CASE (hint_subtitle_much_too_long)
+{
+       check (
+               TextType::OPEN_SUBTITLE,
+               "hint_subtitle_much_too_long",
+               string("At least one of your subtitle lines has more than 79 characters.  You should make each line 79 characters at most in length.")
              );
 }
 
@@ -157,19 +173,23 @@ BOOST_AUTO_TEST_CASE (hint_subtitle_mxf_too_big)
 {
        string const name = "hint_subtitle_mxf_too_big";
 
-       shared_ptr<Film> film = new_test_film2 (name);
-       shared_ptr<Content> content = content_factory("test/data/" + name + ".srt").front();
-       content->text.front()->set_type (TEXT_OPEN_SUBTITLE);
-       for (int i = 1; i < 512; ++i) {
-               shared_ptr<dcpomatic::Font> font(new dcpomatic::Font(String::compose("font_%1", i)));
-               font->set_file ("test/data/LiberationSans-Regular.ttf");
-               content->text.front()->add_font(font);
+       dcp::File fake_font("build/test/hint_subtitle_mxf_too_big.ttf", "w");
+       for (int i = 0; i < 4096; ++i) {
+               std::vector<uint8_t> rubbish(65536);
+               fake_font.write(rubbish.data(), 1, rubbish.size());
        }
+       fake_font.close();
+
+       auto film = new_test_film2 (name);
+       auto content = content_factory("test/data/" + name + ".srt").front();
+       content->text.front()->set_type (TextType::OPEN_SUBTITLE);
+       content->text.front()->set_language (dcp::LanguageTag("en-US"));
        film->examine_and_add_content (content);
        BOOST_REQUIRE (!wait_for_jobs());
-       vector<string> hints = get_hints (film);
+       content->text.front()->get_font("")->set_file("build/test/hint_subtitle_mxf_too_big.ttf");
+       auto hints = get_hints (film);
 
-       BOOST_REQUIRE_EQUAL (hints.size(), 1);
+       BOOST_REQUIRE_EQUAL (hints.size(), 1U);
        BOOST_CHECK_EQUAL (
                hints[0],
                "At least one of your subtitle files is larger than " MAX_TEXT_MXF_SIZE_TEXT " in total.  "
@@ -182,26 +202,27 @@ BOOST_AUTO_TEST_CASE (hint_closed_caption_xml_too_big)
 {
        string const name = "hint_closed_caption_xml_too_big";
 
-       shared_ptr<Film> film = new_test_film2 (name);
+       auto film = new_test_film2 (name);
 
-       FILE* ccap = fopen_boost (String::compose("build/test/%1.srt", name), "w");
+       dcp::File ccap(String::compose("build/test/%1.srt", name), "w");
        BOOST_REQUIRE (ccap);
        for (int i = 0; i < 2048; ++i) {
-               fprintf(ccap, "%d\n", i + 1);
+               fprintf(ccap.get(), "%d\n", i + 1);
                int second = i * 2;
                int minute = second % 60;
-               fprintf(ccap, "00:%02d:%02d,000 --> 00:%02d:%02d,000\n", minute, second, minute, second + 1);
-               fprintf(ccap, "Here are some closed captions.\n\n");
+               fprintf(ccap.get(), "00:%02d:%02d,000 --> 00:%02d:%02d,000\n", minute, second, minute, second + 1);
+               fprintf(ccap.get(), "Here are some closed captions.\n\n");
        }
-       fclose (ccap);
+       ccap.close();
 
-       shared_ptr<Content> content = content_factory("build/test/" + name + ".srt").front();
-       content->text.front()->set_type (TEXT_CLOSED_CAPTION);
+       auto content = content_factory("build/test/" + name + ".srt").front();
+       content->text.front()->set_type (TextType::CLOSED_CAPTION);
+       content->text.front()->set_language (dcp::LanguageTag("en-US"));
        film->examine_and_add_content (content);
        BOOST_REQUIRE (!wait_for_jobs());
-       vector<string> hints = get_hints (film);
+       auto hints = get_hints (film);
 
-       BOOST_REQUIRE_EQUAL (hints.size(), 1);
+       BOOST_REQUIRE_EQUAL (hints.size(), 1U);
        BOOST_CHECK_EQUAL (
                hints[0],
                "At least one of your closed caption files' XML part is larger than " MAX_CLOSED_CAPTION_XML_SIZE_TEXT ".  "
@@ -209,3 +230,53 @@ BOOST_AUTO_TEST_CASE (hint_closed_caption_xml_too_big)
                );
 }
 
+
+BOOST_AUTO_TEST_CASE (hints_destroyed_while_running)
+{
+       auto film = new_test_film2 ("hints_destroyed_while_running");
+       auto content = content_factory(TestPaths::private_data() / "boon_telly.mkv").front();
+       film->examine_and_add_content (content);
+       BOOST_REQUIRE (!wait_for_jobs());
+
+       auto hints = make_shared<Hints>(film);
+       hints->start ();
+       dcpomatic_sleep_seconds (1);
+       hints.reset ();
+       dcpomatic_sleep_seconds (1);
+}
+
+
+BOOST_AUTO_TEST_CASE (hints_audio_with_no_language)
+{
+       auto content = content_factory("test/data/sine_440.wav").front();
+       auto film = new_test_film2 ("hints_audio_with_no_language", { content });
+       content->audio->set_gain (-6);
+
+       auto hints = get_hints (film);
+       BOOST_REQUIRE_EQUAL (hints.size(), 1U);
+       BOOST_CHECK_EQUAL (
+               hints[0],
+               "Some of your content has audio but you have not set the audio language.  It is advisable to set the audio language "
+               "in the \"DCP\" tab unless your audio has no spoken parts."
+               );
+}
+
+
+BOOST_AUTO_TEST_CASE (hints_certificate_validity)
+{
+       ConfigRestorer cr;
+
+       Config::instance()->set_signer_chain(make_shared<dcp::CertificateChain>(openssl_path(), 40 * 365));
+
+       auto film = new_test_film2 ("hints_certificate_validity");
+       auto hints = get_hints (film);
+       BOOST_REQUIRE_EQUAL (hints.size(), 1U);
+       BOOST_CHECK_EQUAL (
+               hints[0],
+               "The certificate chain that DCP-o-matic uses for signing DCPs and KDMs has a validity period "
+               "that is too long.  This will cause problems playing back DCPs on some systems. "
+               "You are advised to re-create the signing certificate chain by clicking the "
+               "\"Re-make certificates and key...\" button in the Keys page of Preferences."
+               );
+}
+