Include film title in KDM filenames.
[dcpomatic.git] / src / lib / kdm.cc
1 /*
2     Copyright (C) 2013 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <list>
21 #include <boost/shared_ptr.hpp>
22 #include <quickmail.h>
23 #include <zip.h>
24 #include <libdcp/kdm.h>
25 #include "kdm.h"
26 #include "cinema.h"
27 #include "exceptions.h"
28 #include "util.h"
29 #include "film.h"
30 #include "config.h"
31
32 using std::list;
33 using std::string;
34 using boost::shared_ptr;
35
36 struct ScreenKDM
37 {
38         ScreenKDM (shared_ptr<Screen> s, libdcp::KDM k)
39                 : screen (s)
40                 , kdm (k)
41         {}
42         
43         shared_ptr<Screen> screen;
44         libdcp::KDM kdm;
45 };
46
47 static string
48 kdm_filename (shared_ptr<Film> film, ScreenKDM kdm)
49 {
50         return tidy_for_filename (film->name()) + "_" + tidy_for_filename (kdm.screen->cinema->name) + "_" + tidy_for_filename (kdm.screen->name) + ".kdm.xml";
51 }
52
53 struct CinemaKDMs
54 {
55         shared_ptr<Cinema> cinema;
56         list<ScreenKDM> screen_kdms;
57
58         void make_zip_file (shared_ptr<Film> film, boost::filesystem::path zip_file) const
59         {
60                 int error;
61                 struct zip* zip = zip_open (zip_file.string().c_str(), ZIP_CREATE | ZIP_EXCL, &error);
62                 if (!zip) {
63                         if (error == ZIP_ER_EXISTS) {
64                                 throw FileError ("ZIP file already exists", zip_file);
65                         }
66                         throw FileError ("could not create ZIP file", zip_file);
67                 }
68                 
69                 list<shared_ptr<string> > kdm_strings;
70                 
71                 for (list<ScreenKDM>::const_iterator i = screen_kdms.begin(); i != screen_kdms.end(); ++i) {
72                         shared_ptr<string> kdm (new string (i->kdm.as_xml ()));
73                         kdm_strings.push_back (kdm);
74                         
75                         struct zip_source* source = zip_source_buffer (zip, kdm->c_str(), kdm->length(), 0);
76                         if (!source) {
77                                 throw StringError ("could not create ZIP source");
78                         }
79                         
80                         if (zip_add (zip, kdm_filename (film, *i).c_str(), source) == -1) {
81                                 throw StringError ("failed to add KDM to ZIP archive");
82                         }
83                 }
84                 
85                 if (zip_close (zip) == -1) {
86                         throw StringError ("failed to close ZIP archive");
87                 }
88         }
89 };
90
91 /* Not complete but sufficient for our purposes (we're using
92    ScreenKDM in a list where all the screens will be unique).
93 */
94 bool
95 operator== (ScreenKDM const & a, ScreenKDM const & b)
96 {
97         return a.screen == b.screen;
98 }
99
100 static list<ScreenKDM>
101 make_screen_kdms (shared_ptr<Film> film, list<shared_ptr<Screen> > screens, boost::posix_time::ptime from, boost::posix_time::ptime to)
102 {
103         list<libdcp::KDM> kdms = film->make_kdms (screens, from, to);
104            
105         list<ScreenKDM> screen_kdms;
106         
107         list<shared_ptr<Screen> >::iterator i = screens.begin ();
108         list<libdcp::KDM>::iterator j = kdms.begin ();
109         while (i != screens.end() && j != kdms.end ()) {
110                 screen_kdms.push_back (ScreenKDM (*i, *j));
111                 ++i;
112                 ++j;
113         }
114
115         return screen_kdms;
116 }
117
118 static list<CinemaKDMs>
119 make_cinema_kdms (shared_ptr<Film> film, list<shared_ptr<Screen> > screens, boost::posix_time::ptime from, boost::posix_time::ptime to)
120 {
121         list<ScreenKDM> screen_kdms = make_screen_kdms (film, screens, from, to);
122         list<CinemaKDMs> cinema_kdms;
123
124         while (!screen_kdms.empty ()) {
125                 
126                 /* Get all the screens from a single cinema */
127
128                 CinemaKDMs ck;
129                 
130                 list<ScreenKDM>::iterator i = screen_kdms.begin ();
131                 ck.cinema = i->screen->cinema;
132                 ck.screen_kdms.push_back (*i);
133                 list<ScreenKDM>::iterator j = i;
134                 ++i;
135                 screen_kdms.remove (*j);
136                 
137                 while (i != screen_kdms.end ()) {
138                         if (i->screen->cinema == ck.cinema) {
139                                 ck.screen_kdms.push_back (*i);
140                                 list<ScreenKDM>::iterator j = i;
141                                 ++i;
142                                 screen_kdms.remove (*j);
143                         } else {
144                                 ++i;
145                         }
146                 }
147
148                 cinema_kdms.push_back (ck);
149         }
150
151         return cinema_kdms;
152 }
153
154 void
155 write_kdm_files (
156         shared_ptr<Film> film, list<shared_ptr<Screen> > screens, boost::posix_time::ptime from, boost::posix_time::ptime to, boost::filesystem::path directory
157         )
158 {
159         list<ScreenKDM> screen_kdms = make_screen_kdms (film, screens, from, to);
160
161         /* Write KDMs to the specified directory */
162         for (list<ScreenKDM>::iterator i = screen_kdms.begin(); i != screen_kdms.end(); ++i) {
163                 boost::filesystem::path out = directory;
164                 out /= kdm_filename (film, *i);
165                 i->kdm.as_xml (out);
166         }
167 }
168
169 void
170 write_kdm_zip_files (
171         shared_ptr<Film> film, list<shared_ptr<Screen> > screens, boost::posix_time::ptime from, boost::posix_time::ptime to, boost::filesystem::path directory
172         )
173 {
174         list<CinemaKDMs> cinema_kdms = make_cinema_kdms (film, screens, from, to);
175
176         for (list<CinemaKDMs>::const_iterator i = cinema_kdms.begin(); i != cinema_kdms.end(); ++i) {
177                 boost::filesystem::path path = directory;
178                 path /= tidy_for_filename (i->cinema->name) + ".zip";
179                 i->make_zip_file (film, path);
180         }
181 }
182
183 void
184 email_kdms (shared_ptr<Film> film, list<shared_ptr<Screen> > screens, boost::posix_time::ptime from, boost::posix_time::ptime to)
185 {
186         list<CinemaKDMs> cinema_kdms = make_cinema_kdms (film, screens, from, to);
187
188         for (list<CinemaKDMs>::const_iterator i = cinema_kdms.begin(); i != cinema_kdms.end(); ++i) {
189                 
190                 boost::filesystem::path zip_file = boost::filesystem::temp_directory_path ();
191                 zip_file /= boost::filesystem::unique_path().string() + ".zip";
192                 i->make_zip_file (film, zip_file);
193                 
194                 /* Send email */
195                 
196                 quickmail_initialize ();
197                 quickmail mail = quickmail_create (Config::instance()->kdm_from().c_str(), "KDM delivery");
198                 quickmail_add_to (mail, i->cinema->email.c_str ());
199                 
200                 string body = Config::instance()->kdm_email().c_str();
201                 boost::algorithm::replace_all (body, "$DCP_NAME", film->dcp_name ());
202                 
203                 quickmail_set_body (mail, body.c_str());
204                 quickmail_add_attachment_file (mail, zip_file.string().c_str());
205                 char const* error = quickmail_send (mail, Config::instance()->mail_server().c_str(), 25, "", "");
206                 if (error) {
207                         quickmail_destroy (mail);
208                         throw StringError (String::compose ("Failed to send KDM email (%1)", error));
209                 }
210                 quickmail_destroy (mail);
211         }
212 }