2014-06-04 18:42:17 -04:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <sstream>
|
2018-01-26 14:32:00 -05:00
|
|
|
#include <unordered_set>
|
2015-10-29 23:16:31 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
#include <xlnt/packaging/manifest.hpp>
|
|
|
|
#include <xlnt/workbook/workbook.hpp>
|
2017-04-20 14:03:03 -04:00
|
|
|
#include <detail/external/include_libstudxml.hpp>
|
|
|
|
#include <detail/serialization/vector_streambuf.hpp>
|
|
|
|
#include <detail/serialization/zstream.hpp>
|
2014-06-04 18:42:17 -04:00
|
|
|
|
2016-07-20 19:04:44 -04:00
|
|
|
class xml_helper
|
2014-06-04 18:42:17 -04:00
|
|
|
{
|
|
|
|
public:
|
2016-12-10 00:18:50 +00:00
|
|
|
static bool compare_files(const std::string &left,
|
2018-07-09 15:24:03 +12:00
|
|
|
const std::string &right, const std::string &content_type)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2016-12-26 09:38:26 -05:00
|
|
|
// content types are stored in unordered maps, too complicated to compare
|
2018-01-26 14:32:00 -05:00
|
|
|
if (content_type == "[Content_Types].xml")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-17 23:11:06 -06:00
|
|
|
// calcChain is optional
|
2018-01-26 14:32:00 -05:00
|
|
|
if (content_type == "application/vnd.openxmlformats-officedocument.spreadsheetml.calcChain+xml")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// compared already
|
|
|
|
if (content_type == "application/vnd.openxmlformats-package.relationships+xml")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2016-12-26 09:38:26 -05:00
|
|
|
|
2016-12-10 00:18:50 +00:00
|
|
|
auto is_xml = (content_type.substr(0, 12) == "application/"
|
2018-07-09 15:24:03 +12:00
|
|
|
&& content_type.substr(content_type.size() - 4) == "+xml")
|
2016-12-10 00:18:50 +00:00
|
|
|
|| content_type == "application/xml"
|
|
|
|
|| content_type == "[Content_Types].xml"
|
|
|
|
|| content_type == "application/vnd.openxmlformats-officedocument.vmlDrawing";
|
2017-01-04 19:02:31 -05:00
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
if (is_xml)
|
|
|
|
{
|
|
|
|
return compare_xml_exact(left, right);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto is_thumbnail = content_type == "image/jpeg";
|
|
|
|
|
|
|
|
if (is_thumbnail)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return left == right;
|
2016-12-10 00:18:50 +00:00
|
|
|
}
|
2016-09-05 19:17:36 -07:00
|
|
|
|
2017-01-04 19:02:31 -05:00
|
|
|
static bool compare_xml_exact(const std::string &left,
|
|
|
|
const std::string &right, bool suppress_debug_info = false)
|
2016-12-10 00:18:50 +00:00
|
|
|
{
|
|
|
|
xml::parser left_parser(left.data(), left.size(), "left");
|
|
|
|
xml::parser right_parser(right.data(), right.size(), "right");
|
|
|
|
|
|
|
|
bool difference = false;
|
|
|
|
auto right_iter = right_parser.begin();
|
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
auto is_whitespace = [](const std::string &v) {
|
2016-12-26 09:38:26 -05:00
|
|
|
return v.find_first_not_of("\n\r\t ") == std::string::npos;
|
2016-12-10 00:18:50 +00:00
|
|
|
};
|
2016-09-05 19:17:36 -07:00
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Iterate through each node in the left document
|
2016-12-10 00:18:50 +00:00
|
|
|
for (auto left_event : left_parser)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// Ignore entirely whitespace text
|
2016-12-10 00:18:50 +00:00
|
|
|
if (left_event == xml::parser::event_type::characters
|
|
|
|
&& is_whitespace(left_parser.value())) continue;
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the end of the right document is reached
|
2016-12-10 00:18:50 +00:00
|
|
|
if (right_iter == right_parser.end())
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2016-12-10 00:18:50 +00:00
|
|
|
difference = true;
|
|
|
|
break;
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
|
|
|
|
auto right_event = *right_iter;
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Iterate through right document until the first non-whitespace node is reached
|
2016-12-10 00:18:50 +00:00
|
|
|
while (right_iter != right_parser.end()
|
|
|
|
&& right_event == xml::parser::event_type::characters
|
|
|
|
&& is_whitespace(right_parser.value()))
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2016-12-10 00:18:50 +00:00
|
|
|
++right_iter;
|
|
|
|
right_event = *right_iter;
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the left node type differs from the right node type
|
2016-12-10 00:18:50 +00:00
|
|
|
if (left_event != right_event)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2016-12-10 00:18:50 +00:00
|
|
|
difference = true;
|
|
|
|
break;
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
|
|
|
|
if (left_event == xml::parser::event_type::start_element)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// Store a map of all attributes from left and right elements in locals
|
2016-12-10 00:18:50 +00:00
|
|
|
auto left_attr_map = left_parser.attribute_map();
|
|
|
|
auto right_attr_map = right_parser.attribute_map();
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Iterate through all attributes in the left element
|
2016-12-10 00:18:50 +00:00
|
|
|
for (auto attr : left_attr_map)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the rigght element doesn't have the attribute from the left element
|
2016-12-10 00:18:50 +00:00
|
|
|
if (right_attr_map.find(attr.first) == right_attr_map.end())
|
|
|
|
{
|
|
|
|
difference = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the value of the right attribute doesn't match the value of the left
|
2016-12-10 00:18:50 +00:00
|
|
|
if (attr.second.value != right_attr_map.at(attr.first).value)
|
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// Unless this exception holds
|
|
|
|
if (left_parser.qname() == xml::qname("urn:schemas-microsoft-com:vml", "shape")
|
|
|
|
&& attr.first == std::string("style"))
|
|
|
|
{
|
|
|
|
// for now this doesn't matter, so do nothing
|
|
|
|
// TODO: think of a better way to do this or prevent the difference in the first place
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
difference = true;
|
|
|
|
break;
|
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
}
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Iterate through all attributes in the right element
|
2016-12-10 00:18:50 +00:00
|
|
|
for (auto attr : right_attr_map)
|
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the left element doesn't have the attribute from the right element
|
2016-12-10 00:18:50 +00:00
|
|
|
if (left_attr_map.find(attr.first) == left_attr_map.end())
|
|
|
|
{
|
|
|
|
difference = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the value of the left attribute doesn't match the value of the right
|
2016-12-10 00:18:50 +00:00
|
|
|
if (attr.second.value != left_attr_map.at(attr.first).value)
|
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// Unless this exception holds
|
|
|
|
if (left_parser.qname() == xml::qname("urn:schemas-microsoft-com:vml", "shape")
|
|
|
|
&& attr.first == std::string("style"))
|
|
|
|
{
|
|
|
|
// for now this doesn't matter, so do nothing
|
|
|
|
// TODO: think of a better way to do this or prevent the difference in the first place
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
difference = true;
|
|
|
|
break;
|
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-05 19:17:36 -07:00
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// break out of outer for loop too if a difference was found in attribute for loops
|
2016-12-10 00:18:50 +00:00
|
|
|
if (difference)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2017-09-28 08:55:16 -04:00
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Finally, there's a difference if the names of the left and right elements don't match
|
2016-12-10 00:18:50 +00:00
|
|
|
if (left_parser.qname() != right_parser.qname())
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2016-12-10 00:18:50 +00:00
|
|
|
difference = true;
|
2016-09-05 19:17:36 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
else if (left_event == xml::parser::event_type::characters)
|
2016-09-05 19:17:36 -07:00
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// There's a difference if the left text doesn't match the right text
|
2016-12-10 00:18:50 +00:00
|
|
|
if (left_parser.value() != right_parser.value())
|
|
|
|
{
|
2017-10-26 12:54:54 -04:00
|
|
|
// Unless this exception holds
|
|
|
|
if (left_parser.qname() == xml::qname("urn:schemas-microsoft-com:office:excel", "Anchor"))
|
|
|
|
{
|
|
|
|
// for now this doesn't matter, so do nothing
|
|
|
|
// TODO: think of a better way to do this or prevent the difference in the first place
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
difference = true;
|
|
|
|
break;
|
|
|
|
}
|
2016-12-10 00:18:50 +00:00
|
|
|
}
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
|
|
|
|
2017-10-26 12:54:54 -04:00
|
|
|
// Move to the next node in the right document, left node is incremented by for loop
|
2016-12-10 00:18:50 +00:00
|
|
|
++right_iter;
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
2016-08-03 00:12:18 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
if (difference && !suppress_debug_info)
|
|
|
|
{
|
|
|
|
std::cout << "documents don't match" << std::endl;
|
2016-08-03 00:12:18 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
std::cout << "left:" << std::endl;
|
2016-12-24 10:04:57 -05:00
|
|
|
for (auto c : left)
|
|
|
|
{
|
|
|
|
std::cout << c << std::flush;
|
|
|
|
}
|
2018-07-09 15:24:03 +12:00
|
|
|
std::cout << std::endl;
|
2016-08-03 00:12:18 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
std::cout << "right:" << std::endl;
|
2016-12-24 10:04:57 -05:00
|
|
|
for (auto c : right)
|
|
|
|
{
|
|
|
|
std::cout << c << std::flush;
|
|
|
|
}
|
2018-07-09 15:24:03 +12:00
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
2016-08-03 00:12:18 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
return !difference;
|
2015-10-30 18:54:04 -04:00
|
|
|
}
|
2016-07-03 19:22:08 -04:00
|
|
|
|
2018-01-26 14:32:00 -05:00
|
|
|
static bool compare_relationships(const xlnt::manifest &left,
|
|
|
|
const xlnt::manifest &right)
|
|
|
|
{
|
|
|
|
std::unordered_set<std::string> parts;
|
|
|
|
|
|
|
|
for (const auto &part : left.parts())
|
|
|
|
{
|
|
|
|
parts.insert(part.string());
|
|
|
|
|
|
|
|
auto left_rels = left.relationships(part);
|
|
|
|
auto right_rels = right.relationships(part);
|
|
|
|
|
|
|
|
if (left_rels.size() != right_rels.size())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unordered_map<std::string, xlnt::relationship> left_rels_map;
|
|
|
|
|
|
|
|
for (const auto &rel : left_rels)
|
|
|
|
{
|
|
|
|
left_rels_map[rel.id()] = rel;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &right_rel : right_rels)
|
|
|
|
{
|
|
|
|
if (left_rels_map.count(right_rel.id()) != 1)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto &left_rel = left_rels_map.at(right_rel.id());
|
|
|
|
|
|
|
|
if (left_rel != right_rel)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto &part : right.parts())
|
|
|
|
{
|
|
|
|
if (parts.count(part.string()) != 1)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
static bool xlsx_archives_match(const std::vector<std::uint8_t> &left,
|
2017-01-04 19:02:31 -05:00
|
|
|
const std::vector<std::uint8_t> &right)
|
2018-07-09 15:24:03 +12:00
|
|
|
{
|
2016-10-30 15:48:40 -04:00
|
|
|
xlnt::detail::vector_istreambuf left_buffer(left);
|
|
|
|
std::istream left_stream(&left_buffer);
|
2017-01-04 19:02:31 -05:00
|
|
|
xlnt::detail::izstream left_archive(left_stream);
|
2016-10-30 15:48:40 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
const auto left_info = left_archive.files();
|
2016-10-30 15:48:40 -04:00
|
|
|
|
|
|
|
xlnt::detail::vector_istreambuf right_buffer(right);
|
|
|
|
std::istream right_stream(&right_buffer);
|
2017-01-04 19:02:31 -05:00
|
|
|
xlnt::detail::izstream right_archive(right_stream);
|
2016-10-30 15:48:40 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
const auto right_info = right_archive.files();
|
2016-08-06 10:40:17 -04:00
|
|
|
|
2017-05-08 11:40:01 -04:00
|
|
|
auto difference_is_missing_calc_chain = false;
|
|
|
|
|
|
|
|
if (std::abs(int(left_info.size()) - int(right_info.size())) == 1)
|
|
|
|
{
|
2018-07-09 15:24:03 +12:00
|
|
|
auto is_calc_chain = [](const xlnt::path &p) {
|
2017-05-08 11:40:01 -04:00
|
|
|
return p.filename() == "calcChain.xml";
|
|
|
|
};
|
|
|
|
|
|
|
|
auto left_has_calc_chain = std::find_if(left_info.begin(), left_info.end(), is_calc_chain)
|
|
|
|
!= left_info.end();
|
|
|
|
auto right_has_calc_chain = std::find_if(right_info.begin(), right_info.end(), is_calc_chain)
|
|
|
|
!= right_info.end();
|
|
|
|
|
|
|
|
if (left_has_calc_chain != right_has_calc_chain)
|
|
|
|
{
|
|
|
|
difference_is_missing_calc_chain = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
if (left_info.size() != right_info.size() && !difference_is_missing_calc_chain)
|
2016-08-12 00:22:14 -04:00
|
|
|
{
|
|
|
|
std::cout << "left has a different number of files than right" << std::endl;
|
|
|
|
|
|
|
|
std::cout << "left has: ";
|
|
|
|
for (auto &info : left_info)
|
|
|
|
{
|
2016-12-24 10:04:57 -05:00
|
|
|
std::cout << info.string() << ", ";
|
2016-08-12 00:22:14 -04:00
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
|
|
|
std::cout << "right has: ";
|
2016-08-16 00:23:49 -04:00
|
|
|
for (auto &info : right_info)
|
2016-08-12 00:22:14 -04:00
|
|
|
{
|
2016-12-24 10:04:57 -05:00
|
|
|
std::cout << info.string() << ", ";
|
2016-08-12 00:22:14 -04:00
|
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
2017-09-28 08:55:16 -04:00
|
|
|
|
2016-08-12 00:22:14 -04:00
|
|
|
bool match = true;
|
2016-10-30 15:48:40 -04:00
|
|
|
|
|
|
|
xlnt::workbook left_workbook;
|
|
|
|
left_workbook.load(left);
|
2016-12-02 14:37:50 +01:00
|
|
|
|
|
|
|
xlnt::workbook right_workbook;
|
|
|
|
right_workbook.load(right);
|
2017-09-28 08:55:16 -04:00
|
|
|
|
2016-12-02 14:37:50 +01:00
|
|
|
auto &left_manifest = left_workbook.manifest();
|
|
|
|
auto &right_manifest = right_workbook.manifest();
|
2016-08-06 10:40:17 -04:00
|
|
|
|
2018-01-26 14:32:00 -05:00
|
|
|
if (!compare_relationships(left_manifest, right_manifest))
|
|
|
|
{
|
2018-07-09 15:24:03 +12:00
|
|
|
std::cout << "relationship mismatch\n"
|
|
|
|
<< "Left:\n";
|
|
|
|
for (const auto &part : left_manifest.parts())
|
|
|
|
{
|
|
|
|
std::cout << "-part: " << part.string() << '\n';
|
|
|
|
auto rels = left_manifest.relationships(part);
|
|
|
|
for (auto &rel : rels)
|
|
|
|
{
|
|
|
|
std::cout << rel.id() << ':'
|
|
|
|
<< static_cast<int>(rel.type())
|
|
|
|
<< ':' << static_cast<int>(rel.target_mode())
|
|
|
|
<< ':' << rel.source().path().string()
|
|
|
|
<< ':' << rel.target().path().string() << '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::cout << "\nRight:\n";
|
|
|
|
for (const auto &part : right_manifest.parts())
|
|
|
|
{
|
|
|
|
std::cout << "-part: " << part.string() << '\n';
|
|
|
|
auto rels = right_manifest.relationships(part);
|
|
|
|
for (auto &rel : rels)
|
|
|
|
{
|
|
|
|
std::cout << rel.id()
|
|
|
|
<< ':' << static_cast<int>(rel.type())
|
|
|
|
<< ':' << static_cast<int>(rel.target_mode())
|
|
|
|
<< ':' << rel.source().path().string()
|
|
|
|
<< ':' << rel.target().path().string() << '\n';
|
|
|
|
}
|
|
|
|
}
|
2018-01-26 14:32:00 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
for (auto left_member : left_info)
|
|
|
|
{
|
|
|
|
if (!right_archive.has_file(left_member))
|
2016-08-12 00:22:14 -04:00
|
|
|
{
|
2017-05-08 11:40:01 -04:00
|
|
|
if (difference_is_missing_calc_chain)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-08-12 00:22:14 -04:00
|
|
|
match = false;
|
2016-12-24 10:04:57 -05:00
|
|
|
std::cout << "right is missing file: " << left_member.string() << std::endl;
|
2017-05-08 11:40:01 -04:00
|
|
|
|
2017-01-04 19:02:31 -05:00
|
|
|
break;
|
2016-08-12 00:22:14 -04:00
|
|
|
}
|
2017-09-28 08:55:16 -04:00
|
|
|
|
2017-01-04 19:02:31 -05:00
|
|
|
auto left_content_type = left_member.string() == "[Content_Types].xml"
|
2018-07-09 15:24:03 +12:00
|
|
|
? "[Content_Types].xml"
|
|
|
|
: left_manifest.content_type(left_member);
|
2017-01-04 19:02:31 -05:00
|
|
|
auto right_content_type = left_member.string() == "[Content_Types].xml"
|
2018-07-09 15:24:03 +12:00
|
|
|
? "[Content_Types].xml"
|
|
|
|
: right_manifest.content_type(left_member);
|
2017-01-04 19:02:31 -05:00
|
|
|
|
2016-09-05 19:17:36 -07:00
|
|
|
if (left_content_type != right_content_type)
|
|
|
|
{
|
|
|
|
std::cout << "content types differ: "
|
2018-07-09 15:24:03 +12:00
|
|
|
<< left_member.string()
|
|
|
|
<< " "
|
|
|
|
<< left_content_type
|
|
|
|
<< " "
|
|
|
|
<< right_content_type
|
|
|
|
<< std::endl;
|
2016-09-05 19:17:36 -07:00
|
|
|
match = false;
|
2017-01-04 19:02:31 -05:00
|
|
|
break;
|
2016-09-05 19:17:36 -07:00
|
|
|
}
|
2017-01-04 19:02:31 -05:00
|
|
|
|
|
|
|
if (!compare_files(left_archive.read(left_member),
|
2018-07-09 15:24:03 +12:00
|
|
|
right_archive.read(left_member), left_content_type))
|
|
|
|
{
|
|
|
|
std::cout << left_member.string() << std::endl;
|
2016-08-12 00:22:14 -04:00
|
|
|
match = false;
|
2017-01-04 19:02:31 -05:00
|
|
|
break;
|
2018-07-09 15:24:03 +12:00
|
|
|
}
|
|
|
|
}
|
2016-08-06 10:40:17 -04:00
|
|
|
|
2018-07-09 15:24:03 +12:00
|
|
|
return match;
|
|
|
|
}
|
2014-06-04 18:42:17 -04:00
|
|
|
};
|