#include <fstream>
#include <iomanip>
#include <cassert>
+#include <iostream>
#include <boost/filesystem.hpp>
+#include <libxml++/libxml++.h>
#include "dcp.h"
#include "asset.h"
#include "sound_asset.h"
#include "picture_asset.h"
+#include "subtitle_asset.h"
#include "util.h"
#include "metadata.h"
#include "exceptions.h"
DCP::DCP (string directory)
: _directory (directory)
{
- string cpl_file;
- string pkl_file;
- string asset_map_file;
+ Files files;
+ scan (files, directory);
- for (filesystem::directory_iterator i = filesystem::directory_iterator(directory); i != filesystem::directory_iterator(); ++i) {
- string const t = i->path().string ();
- if (ends_with (t, "_cpl.xml")) {
- if (cpl_file.empty ()) {
- cpl_file = t;
- } else {
- throw DCPReadError ("duplicate CPLs found");
- }
- } else if (ends_with (t, "_pkl.xml")) {
- if (pkl_file.empty ()) {
- pkl_file = t;
- } else {
- throw DCPReadError ("duplicate PKLs found");
- }
- } else if (ends_with (t, "ASSETMAP.xml")) {
- if (asset_map_file.empty ()) {
- asset_map_file = t;
- } else {
- throw DCPReadError ("duplicate AssetMaps found");
- }
- }
+ if (files.cpl.empty ()) {
+ throw FileError ("no CPL file found", "");
+ }
+
+ if (files.pkl.empty ()) {
+ throw FileError ("no PKL file found", "");
+ }
+
+ if (files.asset_map.empty ()) {
+ throw FileError ("no AssetMap file found", "");
}
/* Read the XML */
- CPL cpl (cpl_file);
- PKL pkl (pkl_file);
- AssetMap asset_map (asset_map_file);
+ shared_ptr<CPL> cpl;
+ try {
+ cpl.reset (new CPL (files.cpl));
+ } catch (FileError& e) {
+ throw FileError ("could not load CPL file", files.cpl);
+ }
+
+ shared_ptr<PKL> pkl;
+ try {
+ pkl.reset (new PKL (files.pkl));
+ } catch (FileError& e) {
+ throw FileError ("could not load PKL file", files.pkl);
+ }
+
+ shared_ptr<AssetMap> asset_map;
+ try {
+ asset_map.reset (new AssetMap (files.asset_map));
+ } catch (FileError& e) {
+ throw FileError ("could not load AssetMap file", files.asset_map);
+ }
/* Cross-check */
/* XXX */
/* Now cherry-pick the required bits into our own data structure */
- _name = cpl.annotation_text;
- _content_kind = cpl.content_kind;
+ _name = cpl->annotation_text;
+ _content_kind = cpl->content_kind;
+
+ shared_ptr<CPLAssetList> cpl_assets = cpl->reels.front()->asset_list;
- shared_ptr<CPLAssetList> cpl_assets = cpl.reels.front()->asset_list;
-
/* XXX */
_fps = cpl_assets->main_picture->frame_rate.numerator;
_length = cpl_assets->main_picture->duration;
+ string n = pkl->asset_from_id(cpl_assets->main_picture->id)->original_file_name;
+ if (n.empty ()) {
+ n = cpl_assets->main_picture->annotation_text;
+ }
+
_assets.push_back (shared_ptr<PictureAsset> (
new PictureAsset (
_directory,
- cpl_assets->main_picture->annotation_text,
+ n,
_fps,
- _length,
- cpl_assets->main_picture->screen_aspect_ratio.numerator,
- cpl_assets->main_picture->screen_aspect_ratio.denominator
+ _length
)
));
if (cpl_assets->main_sound) {
+
+ n = pkl->asset_from_id(cpl_assets->main_sound->id)->original_file_name;
+ if (n.empty ()) {
+ n = cpl_assets->main_sound->annotation_text;
+ }
+
_assets.push_back (shared_ptr<SoundAsset> (
new SoundAsset (
_directory,
- cpl_assets->main_picture->annotation_text,
+ n,
_fps,
_length
)
));
}
+
+ for (list<string>::iterator i = files.subtitles.begin(); i != files.subtitles.end(); ++i) {
+ string const l = i->substr (_directory.length ());
+ _assets.push_back (shared_ptr<SubtitleAsset> (new SubtitleAsset (_directory, l)));
+ }
}
+
+void
+DCP::scan (Files& files, string directory) const
+{
+ for (filesystem::directory_iterator i = filesystem::directory_iterator(directory); i != filesystem::directory_iterator(); ++i) {
+
+ string const t = i->path().string ();
+
+ if (filesystem::is_directory (*i)) {
+ scan (files, t);
+ continue;
+ }
+
+ if (ends_with (t, ".mxf") || ends_with (t, ".ttf")) {
+ continue;
+ }
+
+ xmlpp::DomParser* p = new xmlpp::DomParser;
+
+ try {
+ p->parse_file (t);
+ } catch (std::exception& e) {
+ delete p;
+ continue;
+ }
+
+ if (!p) {
+ delete p;
+ continue;
+ }
+
+ string const root = p->get_document()->get_root_node()->get_name ();
+ delete p;
+
+ if (root == "CompositionPlaylist") {
+ if (files.cpl.empty ()) {
+ files.cpl = t;
+ } else {
+ throw DCPReadError ("duplicate CPLs found");
+ }
+ } else if (root == "PackingList") {
+ if (files.pkl.empty ()) {
+ files.pkl = t;
+ } else {
+ throw DCPReadError ("duplicate PKLs found");
+ }
+ } else if (root == "AssetMap") {
+ if (files.asset_map.empty ()) {
+ files.asset_map = t;
+ } else {
+ throw DCPReadError ("duplicate AssetMaps found");
+ }
+ files.asset_map = t;
+ } else if (root == "DCSubtitle") {
+ files.subtitles.push_back (t);
+ }
+ }
+}
+
+
list<string>
-DCP::equals (DCP const & other, EqualityFlags flags) const
+DCP::equals (DCP const & other, EqualityOptions opt) const
{
list<string> notes;
- if (flags & LIBDCP_METADATA) {
+ if (opt.flags & LIBDCP_METADATA) {
if (_name != other._name) {
notes.push_back ("names differ");
}
list<shared_ptr<Asset> >::const_iterator b = other._assets.begin ();
while (a != _assets.end ()) {
- list<string> n = (*a)->equals (*b->get(), flags);
+ list<string> n = (*a)->equals (*b, opt);
notes.merge (n);
++a;
++b;
return notes;
}
+
+shared_ptr<const PictureAsset>
+DCP::picture_asset () const
+{
+ for (list<shared_ptr<Asset> >::const_iterator i = _assets.begin(); i != _assets.end(); ++i) {
+ shared_ptr<PictureAsset> p = dynamic_pointer_cast<PictureAsset> (*i);
+ if (p) {
+ return p;
+ }
+ }
+
+ return shared_ptr<const PictureAsset> ();
+}
+
+shared_ptr<const SoundAsset>
+DCP::sound_asset () const
+{
+ for (list<shared_ptr<Asset> >::const_iterator i = _assets.begin(); i != _assets.end(); ++i) {
+ shared_ptr<SoundAsset> s = dynamic_pointer_cast<SoundAsset> (*i);
+ if (s) {
+ return s;
+ }
+ }
+
+ return shared_ptr<const SoundAsset> ();
+}
+
+shared_ptr<const SubtitleAsset>
+DCP::subtitle_asset () const
+{
+ for (list<shared_ptr<Asset> >::const_iterator i = _assets.begin(); i != _assets.end(); ++i) {
+ shared_ptr<SubtitleAsset> s = dynamic_pointer_cast<SubtitleAsset> (*i);
+ if (s) {
+ return s;
+ }
+ }
+
+ return shared_ptr<const SubtitleAsset> ();
+}