Rename read_interop_subtitle_test -> interop_subtitle_test.
[libdcp.git] / test / test.cc
1 /*
2     Copyright (C) 2012-2015 Carl Hetherington <cth@carlh.net>
3
4     This file is part of libdcp.
5
6     libdcp 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     libdcp 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 libdcp.  If not, see <http://www.gnu.org/licenses/>.
18 */
19
20 #define BOOST_TEST_DYN_LINK
21 #define BOOST_TEST_MODULE libdcp_test
22 #include "util.h"
23 #include "test.h"
24 #include <libxml++/libxml++.h>
25 #include <boost/test/unit_test.hpp>
26 #include <cstdio>
27 #include <iostream>
28
29 using std::string;
30 using std::min;
31 using std::list;
32
33 boost::filesystem::path private_test;
34
35 struct TestConfig
36 {
37         TestConfig()
38         {
39                 dcp::init ();
40                 if (boost::unit_test::framework::master_test_suite().argc >= 2) {
41                         private_test = boost::unit_test::framework::master_test_suite().argv[1];
42                 }
43         }
44 };
45
46 void
47 check_xml (xmlpp::Element* ref, xmlpp::Element* test, list<string> ignore)
48 {
49         BOOST_CHECK_EQUAL (ref->get_name (), test->get_name ());
50         BOOST_CHECK_EQUAL (ref->get_namespace_prefix (), test->get_namespace_prefix ());
51
52         if (find (ignore.begin(), ignore.end(), ref->get_name()) != ignore.end ()) {
53                 return;
54         }
55
56         xmlpp::Element::NodeList ref_children = ref->get_children ();
57         xmlpp::Element::NodeList test_children = test->get_children ();
58         BOOST_REQUIRE_MESSAGE (
59                 ref_children.size () == test_children.size (),
60                 "child counts of " << ref->get_name() << " differ; ref has " << ref_children.size() << ", test has " << test_children.size()
61                 );
62
63         xmlpp::Element::NodeList::iterator k = ref_children.begin ();
64         xmlpp::Element::NodeList::iterator l = test_children.begin ();
65         while (k != ref_children.end ()) {
66
67                 /* XXX: should be doing xmlpp::EntityReference, xmlpp::XIncludeEnd, xmlpp::XIncludeStart */
68
69                 xmlpp::Element* ref_el = dynamic_cast<xmlpp::Element*> (*k);
70                 xmlpp::Element* test_el = dynamic_cast<xmlpp::Element*> (*l);
71                 BOOST_CHECK ((ref_el && test_el) || (!ref_el && !test_el));
72                 if (ref_el && test_el) {
73                         check_xml (ref_el, test_el, ignore);
74                 }
75
76                 xmlpp::ContentNode* ref_cn = dynamic_cast<xmlpp::ContentNode*> (*k);
77                 xmlpp::ContentNode* test_cn = dynamic_cast<xmlpp::ContentNode*> (*l);
78                 BOOST_CHECK ((ref_cn && test_cn) || (!ref_cn && !test_cn));
79                 if (ref_cn && test_cn) {
80                         BOOST_CHECK_EQUAL (ref_cn->get_content(), test_cn->get_content ());
81                 }
82
83                 ++k;
84                 ++l;
85         }
86
87         xmlpp::Element::AttributeList ref_attributes = ref->get_attributes ();
88         xmlpp::Element::AttributeList test_attributes = test->get_attributes ();
89         BOOST_CHECK_EQUAL (ref_attributes.size(), test_attributes.size ());
90
91         xmlpp::Element::AttributeList::const_iterator m = ref_attributes.begin();
92         xmlpp::Element::AttributeList::const_iterator n = test_attributes.begin();
93         while (m != ref_attributes.end ()) {
94                 BOOST_CHECK_EQUAL ((*m)->get_name(), (*n)->get_name());
95                 BOOST_CHECK_EQUAL ((*m)->get_value(), (*n)->get_value());
96
97                 ++m;
98                 ++n;
99         }
100 }
101
102 void
103 check_xml (string ref, string test, list<string> ignore)
104 {
105         xmlpp::DomParser* ref_parser = new xmlpp::DomParser ();
106         ref_parser->parse_memory (ref);
107         xmlpp::Element* ref_root = ref_parser->get_document()->get_root_node ();
108         xmlpp::DomParser* test_parser = new xmlpp::DomParser ();
109         test_parser->parse_memory (test);
110         xmlpp::Element* test_root = test_parser->get_document()->get_root_node ();
111
112         check_xml (ref_root, test_root, ignore);
113 }
114
115 void
116 check_file (boost::filesystem::path ref, boost::filesystem::path check)
117 {
118         uintmax_t N = boost::filesystem::file_size (ref);
119         BOOST_CHECK_EQUAL (N, boost::filesystem::file_size (check));
120         FILE* ref_file = dcp::fopen_boost (ref, "rb");
121         BOOST_CHECK (ref_file);
122         FILE* check_file = dcp::fopen_boost (check, "rb");
123         BOOST_CHECK (check_file);
124
125         int const buffer_size = 65536;
126         uint8_t* ref_buffer = new uint8_t[buffer_size];
127         uint8_t* check_buffer = new uint8_t[buffer_size];
128
129         string error;
130         error = "File " + check.string() + " differs from reference " + ref.string();
131
132         while (N) {
133                 uintmax_t this_time = min (uintmax_t (buffer_size), N);
134                 size_t r = fread (ref_buffer, 1, this_time, ref_file);
135                 BOOST_CHECK_EQUAL (r, this_time);
136                 r = fread (check_buffer, 1, this_time, check_file);
137                 BOOST_CHECK_EQUAL (r, this_time);
138
139                 BOOST_CHECK_MESSAGE (memcmp (ref_buffer, check_buffer, this_time) == 0, error);
140                 if (memcmp (ref_buffer, check_buffer, this_time)) {
141                         break;
142                 }
143
144                 N -= this_time;
145         }
146
147         delete[] ref_buffer;
148         delete[] check_buffer;
149
150         fclose (ref_file);
151         fclose (check_file);
152 }
153
154 BOOST_GLOBAL_FIXTURE (TestConfig);