10cae806f46f13bb075ee0d1f98f53090fee0e74
[dcpomatic.git] / cscript
1 # -*- mode: python -*-
2 #
3 #    Copyright (C) 2012-2022 Carl Hetherington <cth@carlh.net>
4 #
5 #    This file is part of DCP-o-matic.
6 #
7 #    DCP-o-matic is free software; you can redistribute it and/or modify
8 #    it under the terms of the GNU General Public License as published by
9 #    the Free Software Foundation; either version 2 of the License, or
10 #    (at your option) any later version.
11 #
12 #    DCP-o-matic is distributed in the hope that it will be useful,
13 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
14 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 #    GNU General Public License for more details.
16 #
17 #    You should have received a copy of the GNU General Public License
18 #    along with DCP-o-matic.  If not, see <http://www.gnu.org/licenses/>.
19 #
20
21 from __future__ import print_function
22 import glob
23 import shutil
24 import os
25 import copy
26 import json
27
28 deb_build_depends = dict()
29
30 deb_build_depends_base = ['debhelper', 'g++', 'pkg-config', 'libsndfile1-dev', 'libgtk2.0-dev', 'libx264-dev']
31
32 for v in ['16.04', '18.04', '20.04', '20.10', '21.04', '21.10']:
33     deb_build_depends[v] = copy.deepcopy(deb_build_depends_base)
34     deb_build_depends[v].extend(['libssh-dev', 'python'])
35 for v in ['9', '10']:
36     deb_build_depends[v] = copy.deepcopy(deb_build_depends_base)
37     deb_build_depends[v].extend(['libssh-gcrypt-dev', 'python'])
38 for v in ['11']:
39     deb_build_depends[v] = copy.deepcopy(deb_build_depends_base)
40     deb_build_depends[v].extend(['libssh-gcrypt-dev', 'python3.9'])
41 deb_build_depends['unstable'] = copy.deepcopy(deb_build_depends_base)
42 deb_build_depends['unstable'].extend(['python'])
43
44 deb_depends = dict()
45 deb_depends_gui = dict()
46
47 deb_depends_base = ['libc6', 'libsndfile1', 'libsamplerate0', 'libxmlsec1', 'libxmlsec1-openssl', 'libgtk2.0-0']
48
49 deb_depends['16.04'] = copy.deepcopy(deb_depends_base)
50 deb_depends['16.04'].extend(['libboost-filesystem1.58.0',
51                              'libboost-thread1.58.0',
52                              'libboost-regex1.58.0',
53                              'libxml++2.6-2v5',
54                              'libboost-date-time1.58.0',
55                              'libzip4',
56                              'libcairomm-1.0-1v5',
57                              'libpangomm-1.4-1v5',
58                              'libicu55',
59                              'libnettle6',
60                              'libssh-4',
61                              'libx264-148',
62                              'libcurl3',
63                              'libxerces-c3.1'])
64
65 deb_depends['18.04'] = copy.deepcopy(deb_depends_base)
66 deb_depends['18.04'].extend(['libboost-filesystem1.65.1',
67                              'libboost-thread1.65.1',
68                              'libboost-regex1.65.1',
69                              'libboost-date-time1.65.1',
70                              'libcairomm-1.0-1v5',
71                              'libpangomm-1.4-1v5',
72                              'libxml++2.6-2v5',
73                              'libzip4',
74                              'libicu60',
75                              'libnettle6',
76                              'libssh-4',
77                              'libx264-152',
78                              'libcurl4',
79                              'libpulse0',
80                              'libxerces-c3.2',
81                              'libnanomsg4'])
82
83 deb_depends['20.04'] = copy.deepcopy(deb_depends_base)
84 deb_depends['20.04'].extend(['libboost-filesystem1.71.0',
85                              'libboost-thread1.71.0',
86                              'libboost-regex1.71.0',
87                              'libboost-date-time1.71.0',
88                              'libcairomm-1.0-1v5',
89                              'libpangomm-1.4-1v5',
90                              'libxml++2.6-2v5',
91                              'libzip5',
92                              'libicu66',
93                              'libnettle7',
94                              'libssh-4',
95                              'libx264-155',
96                              'libcurl4',
97                              'libpulse0',
98                              'libxerces-c3.2',
99                              'libnanomsg5'])
100
101 deb_depends['20.10'] = copy.deepcopy(deb_depends_base)
102 deb_depends['20.10'].extend(['libboost-filesystem1.71.0',
103                              'libboost-thread1.71.0',
104                              'libboost-regex1.71.0',
105                              'libboost-date-time1.71.0',
106                              'libcairomm-1.0-1v5',
107                              'libpangomm-1.4-1v5',
108                              'libxml++2.6-2v5',
109                              'libzip5',
110                              'libicu67',
111                              'libnettle8',
112                              'libssh-4',
113                              'libx264-160',
114                              'libcurl4',
115                              'libpulse0',
116                              'libxerces-c3.2',
117                              'libnanomsg5'])
118
119 deb_depends['21.04'] = copy.deepcopy(deb_depends_base)
120 deb_depends['21.04'].extend(['libboost-filesystem1.74.0',
121                              'libboost-thread1.74.0',
122                              'libboost-regex1.74.0',
123                              'libboost-date-time1.74.0',
124                              'libcairomm-1.0-1v5',
125                              'libpangomm-1.4-1v5',
126                              'libxml++2.6-2v5',
127                              'libzip4',
128                              'libicu67',
129                              'libnettle8',
130                              'libssh-4',
131                              'libx264-160',
132                              'libcurl4',
133                              'libpulse0',
134                              'libxerces-c3.2',
135                              'libnanomsg5'])
136
137 deb_depends['21.10'] = copy.deepcopy(deb_depends_base)
138 deb_depends['21.10'].extend(['libboost-filesystem1.74.0',
139                              'libboost-thread1.74.0',
140                              'libboost-regex1.74.0',
141                              'libboost-date-time1.74.0',
142                              'libcairomm-1.0-1v5',
143                              'libpangomm-1.4-1v5',
144                              'libxml++2.6-2v5',
145                              'libzip4',
146                              'libicu67',
147                              'libnettle8',
148                              'libssh-4',
149                              'libx264-160',
150                              'libcurl4',
151                              'libpulse0',
152                              'libxerces-c3.2',
153                              'libnanomsg5',
154                              'libdav1d4'])
155
156 deb_depends['9'] = copy.deepcopy(deb_depends_base)
157 deb_depends['9'].extend(['libboost-filesystem1.62.0',
158                          'libboost-thread1.62.0',
159                          'libboost-regex1.62.0',
160                          'libboost-date-time1.62.0',
161                          'libxml++2.6-2v5',
162                          'libgtk2.0-0',
163                          'libzip4',
164                          'libcairomm-1.0-1v5',
165                          'libpangomm-1.4-1v5',
166                          'libicu57',
167                          'libssh-4',
168                          'libssh-gcrypt-4',
169                          'libnettle6',
170                          'libx264-148',
171                          'libcurl3',
172                          'libxerces-c3.1'])
173
174 deb_depends_gui['9'] = [ 'libxcb-xfixes0',
175                          'libxcb-shape0',
176                          'libasound2',
177                          'libpulse0' ]
178
179 deb_depends['10'] = copy.deepcopy(deb_depends_base)
180 deb_depends['10'].extend(['libboost-filesystem1.67.0',
181                           'libboost-thread1.67.0',
182                           'libboost-regex1.67.0',
183                           'libboost-date-time1.67.0',
184                           'libxml++2.6-2v5',
185                           'libgtk2.0-0',
186                           'libzip4',
187                           'libcairomm-1.0-1v5',
188                           'libpangomm-1.4-1v5',
189                           'libicu63',
190                           'libssh-4',
191                           'libssh-gcrypt-4',
192                           'libnettle6',
193                           'libx264-155',
194                           'libcurl4',
195                           'libxerces-c3.2',
196                           'libnanomsg5'])
197
198 deb_depends_gui['10'] = [ 'libxcb-xfixes0',
199                           'libxcb-shape0',
200                           'libasound2',
201                           'libpulse0' ]
202
203 deb_depends['11'] = copy.deepcopy(deb_depends_base)
204 deb_depends['11'].extend(['libboost-filesystem1.74.0',
205                           'libboost-thread1.74.0',
206                           'libboost-regex1.74.0',
207                           'libboost-date-time1.74.0',
208                           'libxml++2.6-2v5',
209                           'libzip4',
210                           'libcairomm-1.0-1v5',
211                           'libpangomm-1.4-1v5',
212                           'libicu67',
213                           'libssh-4',
214                           'libssh-gcrypt-4',
215                           'libnettle8',
216                           'libx264-160',
217                           'libcurl4',
218                           'libxerces-c3.2',
219                           'libnanomsg5',
220                           'libdav1d4'])
221
222 deb_depends_gui['11'] = [ 'libxcb-xfixes0',
223                           'libxcb-shape0',
224                           'libasound2',
225                           'libpulse0' ]
226
227 deb_depends['unstable'] = copy.deepcopy(deb_depends_base)
228 deb_depends['unstable'].extend(['libboost-filesystem1.67.0',
229                                 'libboost-thread1.67.0',
230                                 'libboost-regex1.67.0',
231                                 'libboost-date-time1.67.0',
232                                 'libxml++2.6-2v5',
233                                 'libgtk2.0-0',
234                                 'libzip4',
235                                 'libicu63',
236                                 'libnettle6',
237                                 'libx264-155',
238                                 'libcurl4',
239                                 'libxerces-c3.2',
240                                 'libdav1d4'])
241
242 def can_build_disk(target):
243     # We can build dcpomatic2_disk on platforms that have Boost process and can build the lwext4
244     # library.  For now, just whitelist good ones here.
245     #
246     # - Lots of Linux distros (including Ubuntu 16.04) don't have a new enough boost (1.64 or above)
247     # - On Centos 6 we can't build lwext4 because it needs a new CMake which Centos 6's g++ is not new enough to build.
248     # - On Centos 7 there is a build error in lwext4 related to __unused
249     if target.platform == 'windows':
250         return True
251     if target.platform == 'osx':
252         return True
253     if target.platform == 'linux':
254         if target.distro == 'ubuntu' and target.version != '16.04':
255             return True
256         if target.distro == 'debian' and target.version != '9':
257             return True
258         if target.detail == 'appimage':
259             return True 
260         if target.distro == 'fedora' and int(target.version) >= 31:
261             return True
262         if target.distro == 'centos' and target.version != '7':
263             return True
264         if target.distro == 'mageia':
265             return True
266     return False
267
268 def packages(name, packages, f):
269     s = '%s: ' % name
270     for p in packages:
271         s += str(p) + ', '
272     print(s[:-2], file=f)
273
274 def make_control(debian_version, bits, filename, debug, gui):
275     f = open(filename, 'w')
276     print('Source: dcpomatic', file=f)
277     print('Section: video', file=f)
278     print('Priority: extra', file=f)
279     print('Maintainer: Carl Hetherington <carl@dcpomatic.com>', file=f)
280     packages('Build-Depends', deb_build_depends[debian_version], f)
281     print('Standards-Version: 3.9.3', file=f)
282     print('Homepage: https://dcpomatic.com/', file=f)
283     print('', file=f)
284     suffix = '' if gui else '-cli'
285     print(f'Package: dcpomatic{suffix}', file=f)
286     if bits == 32:
287         print('Architecture: i386', file=f)
288     else:
289         print('Architecture: amd64', file=f)
290
291     pkg = deb_depends[debian_version]
292     if gui and debian_version in deb_depends_gui:
293         pkg.extend(deb_depends_gui[debian_version])
294
295     packages('Depends', pkg, f)
296
297     print('Description: Generator of Digital Cinema Packages (DCPs)', file=f)
298     print('  DCP-o-matic generates Digital Cinema Packages (DCPs) from videos, images,', file=f)
299     print('  sound and subtitle files.  You can use it to make content for playback on DCI-compliant', file=f)
300     print('  cinema projectors.', file=f)
301     if not gui:
302         print('  This package contains the command-line tools only.', file=f)
303
304     if debug:
305         print('', file=f)
306         print(f'Package: dcpomatic{suffix}-dbg', file=f)
307         if bits == 32:
308             print('Architecture: i386', file=f)
309         else:
310             print('Architecture: amd64', file=f)
311         print('Section: debug', file=f)
312         print('Priority: extra', file=f)
313         packages('Depends', pkg, f)
314         print('Description: debugging symbols for dcpomatic', file=f)
315         print('  This package contains the debugging symbols for dcpomatic.', file=f)
316         print('', file=f)
317
318 def make_spec(filename, version, target, options, requires=None):
319     """Make a .spec file for a RPM build"""
320     tools = os.path.join(os.path.dirname(os.path.dirname(os.path.dirname(filename))), "src/tools")
321     f = open(filename, 'w')
322     print('Summary:A program that generates Digital Cinema Packages (DCPs) from video and audio files', file=f)
323     print('Name:dcpomatic2', file=f)
324     print('Version:%s' % version, file=f)
325     print('Release:1%{?dist}', file=f)
326     print('License:GPL', file=f)
327     print('Group:Applications/Multimedia', file=f)
328     print('URL:https://dcpomatic.com/', file=f)
329     if requires is not None:
330         print('Requires:%s' % requires, file=f)
331     print('', file=f)
332     print('%description', file=f)
333     print('DCP-o-matic generates Digital Cinema Packages (DCPs) from video and audio ', file=f)
334     print('files for presentation on DCI-compliant digital projectors.', file=f)
335     print('', file=f)
336     print('%files', file=f)
337     print('%{_bindir}/dcpomatic2', file=f)
338     print('%{_bindir}/dcpomatic2_batch', file=f)
339     print('%{_bindir}/dcpomatic2_cli', file=f)
340     print('%{_bindir}/dcpomatic2_create', file=f)
341     print('%{_bindir}/dcpomatic2_kdm', file=f)
342     print('%{_bindir}/dcpomatic2_server', file=f)
343     print('%{_bindir}/dcpomatic2_server_cli', file=f)
344     print('%{_bindir}/dcpomatic2_kdm_cli', file=f)
345     print('%{_bindir}/dcpomatic2_player', file=f)
346     print('%{_bindir}/dcpomatic2_playlist', file=f)
347     print('%{_bindir}/dcpomatic2_openssl', file=f)
348     print('%{_bindir}/dcpomatic2_combiner', file=f)
349     print('%{_bindir}/dcpomatic2_verify', file=f)
350     if os.path.exists(os.path.join(tools, "dcpomatic2_disk")):
351         print('%{_bindir}/dcpomatic2_disk', file=f)
352         print('%caps(cap_dac_override=ep) %{_bindir}/dcpomatic2_disk_writer', file=f)
353     print('%{_datadir}/applications/dcpomatic2.desktop', file=f)
354     print('%{_datadir}/applications/dcpomatic2_batch.desktop', file=f)
355     print('%{_datadir}/applications/dcpomatic2_server.desktop', file=f)
356     print('%{_datadir}/applications/dcpomatic2_kdm.desktop', file=f)
357     print('%{_datadir}/applications/dcpomatic2_player.desktop', file=f)
358     print('%{_datadir}/applications/dcpomatic2_playlist.desktop', file=f)
359     print('%{_datadir}/applications/dcpomatic2_combiner.desktop', file=f)
360     if os.path.exists(os.path.join(tools, "dcpomatic2_disk")):
361         print('%{_datadir}/applications/dcpomatic2_disk.desktop', file=f)
362     print('%{_datadir}/dcpomatic2/dcpomatic2_server_small.png', file=f)
363     print('%{_datadir}/dcpomatic2/select.png', file=f)
364     print('%{_datadir}/dcpomatic2/sequence.png', file=f)
365     print('%{_datadir}/dcpomatic2/snap.png', file=f)
366     print('%{_datadir}/dcpomatic2/zoom.png', file=f)
367     print('%{_datadir}/dcpomatic2/zoom_all.png', file=f)
368     print('%{_datadir}/dcpomatic2/tick.png', file=f)
369     print('%{_datadir}/dcpomatic2/no_tick.png', file=f)
370     print('%{_datadir}/dcpomatic2/link.png', file=f)
371     print('%{_datadir}/dcpomatic2/me.jpg', file=f)
372     print('%{_datadir}/dcpomatic2/LiberationSans-Regular.ttf', file=f)
373     print('%{_datadir}/dcpomatic2/LiberationSans-Italic.ttf', file=f)
374     print('%{_datadir}/dcpomatic2/LiberationSans-Bold.ttf', file=f)
375     print('%{_datadir}/dcpomatic2/splash.png', file=f)
376     for r in ['128x128', '16x16', '22x22', '256x256', '32x32', '48x48', '512x512', '64x64']:
377         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2.png' % r, file=f)
378         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_batch.png' % r, file=f)
379         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_kdm.png' % r, file=f)
380         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_server.png' % r, file=f)
381         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_player.png' % r, file=f)
382         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_playlist.png' % r, file=f)
383         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_disk.png' % r, file=f)
384         print('%%{_datadir}/icons/hicolor/%s/apps/dcpomatic2_combiner.png' % r, file=f)
385     for l in ['de_DE', 'es_ES', 'fr_FR', 'it_IT', 'sv_SE', 'nl_NL', 'ru_RU', 'pl_PL', 'da_DK',
386               'pt_PT', 'pt_BR', 'sk_SK', 'cs_CZ', 'uk_UA', 'zh_CN', 'tr_TR']:
387         print('%%{_datadir}/locale/%s/LC_MESSAGES/dcpomatic2.mo' % l, file=f)
388         print('%%{_datadir}/locale/%s/LC_MESSAGES/libdcpomatic2-wx.mo' % l, file=f)
389         print('%%{_datadir}/locale/%s/LC_MESSAGES/libdcpomatic2.mo' % l, file=f)
390     print('%{_datadir}/libdcp/tags/*', file=f)
391     print('%{_datadir}/libdcp/xsd/*', file=f)
392     print('%{_datadir}/polkit-1/actions/com.dcpomatic.write-drive.policy', file=f)
393     print('', file=f)
394     print('%prep', file=f)
395     print('rm -rf $RPM_BUILD_DIR/dcpomatic-%s' % version, file=f)
396     print('tar xjf $RPM_SOURCE_DIR/dcpomatic-%s.tar.bz2' % version, file=f)
397     print('%build', file=f)
398     print('cd dcpomatic-%s' % version, file=f)
399     print('export PKG_CONFIG_PATH=%s/lib/pkgconfig:%s/lib64/pkgconfig:/usr/local/lib/pkgconfig:/usr/local/lib64/pkgconfig' % (target.directory, target.directory), file=f)
400     print('CXXFLAGS="-I%s/include" LDFLAGS="-L%s/lib" ./waf configure --prefix=%%{buildroot}/usr --destdir=/usr %s' %
401           (target.directory, target.directory, configure_options(target, options)), file=f)
402     print('./waf', file=f)
403     print('%install', file=f)
404     print('cd dcpomatic-%s' % version, file=f)
405     print('./waf install', file=f)
406     print('/bin/cp %s/src/openssl/apps/openssl %%{buildroot}/usr/bin/dcpomatic2_openssl' % target.directory, file=f)
407     print('/bin/mkdir -p %{buildroot}/usr/share/libdcp', file=f)
408     print('/bin/cp -r %s/src/libdcp/tags %%{buildroot}/usr/share/libdcp' % target.directory, file=f)
409     print('/bin/cp -r %s/src/libdcp/xsd %%{buildroot}/usr/share/libdcp' % target.directory, file=f)
410     print('/bin/mv %s/bin/dcpverify %%{buildroot}/usr/bin/dcpomatic2_verify' % target.directory, file=f)
411     print('', file=f)
412     print('%post', file=f)
413     print('/bin/touch --no-create %{_datadir}/icons/hicolor &>/dev/null || :', file=f)
414     print('setcap "cap_dac_override+ep cap_sys_admin+ep" /usr/bin/dcpomatic2_disk_writer', file=f)
415     print('', file=f)
416     print('%postun', file=f)
417     print('if [ $1 -eq 0 ] ; then', file=f)
418     print('    /bin/touch --no-create %{_datadir}/icons/hicolor &>/dev/null', file=f)
419     print('    /usr/bin/gtk-update-icon-cache %{_datadir}/icons/hicolor &>/dev/null || :', file=f)
420     print('fi', file=f)
421     print('', file=f)
422     print('%posttrans', file=f)
423     print('/usr/bin/gtk-update-icon-cache %{_datadir}/icons/hicolor &>/dev/null || :', file=f)
424
425 def dependencies(target, options):
426
427     if target.platform == 'linux':
428         ffmpeg_options = { 'shared': False }
429     else:
430         ffmpeg_options = {}
431
432     if target.platform != 'linux' or target.distro != 'arch':
433         deps = [('ffmpeg-cdist', 'cb2b073d4f88230fca1d1d74e45235f5268fd825', ffmpeg_options)]
434     else:
435         # Use distro-provided FFmpeg on Arch
436         deps = []
437
438     deps.append(('libdcp', 'v1.8.7'))
439     deps.append(('libsub', 'v1.6.7'))
440     deps.append(('leqm-nrt', '93ae9e6'))
441     deps.append(('rtaudio', 'f619b76'))
442     # We get our OpenSSL libraries from the environment, but we
443     # also need a patched openssl binary to make certificates.
444     # This dependency is to get that binary, which is added into
445     # the appropriate place later
446     deps.append(('openssl', '7f29dd5'))
447     if can_build_disk(target):
448         deps.append(('lwext4', 'cce3730'))
449     deps.append(('ffcmp', '10934f1a9cd9770ef0b38da153f9576e77e7e925'))
450
451     return deps
452
453 option_defaults = { "gui": True, "variant": None }
454
455 def configure_options(target, options):
456     opt = ' --warnings-are-errors'
457
458     if not ((target.platform == 'linux' and target.distro == 'ubuntu' and target.version == '18.04') or
459             (target.platform == 'osx') or
460             (target.platform == 'windows')):
461         # Currently we only build tests on Ubuntu 18.04, macOS and Windows
462         opt += ' --disable-tests'
463
464     if target.debug:
465         opt += ' --enable-debug'
466     if target.platform == 'windows':
467         opt += ' --target-windows'
468     elif target.platform == 'linux':
469         opt += ' --static-dcpomatic --static-wxwidgets --static-ffmpeg --static-dcp --static-sub --static-cxml'
470         if target.distro == 'centos':
471             if target.version == '6.5':
472                 opt += ' --static-boost --static-xmlpp'
473             elif target.version == '7':
474                 opt += ' --workaround-gssapi'
475
476     if not options['gui']:
477         opt += ' --disable-gui'
478
479     if options['variant'] is not None:
480         opt += ' --variant=%s' % options['variant']
481
482     # Build Windows debug versions with static linking as I think gdb works better then
483     if target.debug and target.platform == 'windows':
484         opt += ' --static-dcpomatic'
485
486     if can_build_disk(target):
487         opt += ' --enable-disk'
488
489     if target.platform == 'osx' and target.arch == 'arm64':
490         opt += ' --target-macos-arm64 --wx-config=%s/wx-config' % target.bin
491
492     return opt
493
494 def build(target, options):
495     if target.platform == 'flatpak':
496         target.checkout_dependencies()
497         prefix = 'https://dcpomatic.com/deps'
498         modules = []
499         modules.append({'name': 'libzip',
500                         'buildsystem': 'cmake',
501                         'sources': [{'type': 'archive',
502                                      'url': '%s/libzip-1.4.0.tar.xz' % prefix,
503                                      'sha256': 'e508aba025f5f94b267d5120fc33761bcd98440ebe49dbfe2ed3df3afeacc7b1'}]})
504         modules.append({'name': 'libsigc++',
505                         'sources': [{'type': 'archive',
506                                      'url': '%s/libsigc++-2.10.0.tar.xz' % prefix,
507                                      'sha256': 'f843d6346260bfcb4426259e314512b99e296e8ca241d771d21ac64f28298d81'}]})
508         modules.append({'name': 'glibmm',
509                         'sources': [{'type': 'archive',
510                                      'url': '%s/glibmm-2.48.1.tar.xz' % prefix,
511                                      'sha256': 'dc225f7d2f466479766332483ea78f82dc349d59399d30c00de50e5073157cdf'}]})
512         modules.append({'name': 'cairomm',
513                         'sources': [{'type': 'archive',
514                                      'url': '%s/cairomm-1.12.2.tar.gz' % prefix,
515                                      'sha256': '45c47fd4d0aa77464a75cdca011143fea3ef795c4753f6e860057da5fb8bd599'}]})
516         modules.append({'name': 'pangomm',
517                         'sources': [{'type': 'archive',
518                                      'url': '%s/pangomm-2.40.1.tar.xz' % prefix,
519                                      'sha256': '9762ee2a2d5781be6797448d4dd2383ce14907159b30bc12bf6b08e7227be3af'}]})
520         modules.append({'name': 'libxml++',
521                         'sources': [{'type': 'archive',
522                                      'url': '%s/libxml++-2.40.1.tar.xz' % prefix,
523                                      'sha256': '4ad4abdd3258874f61c2e2a41d08e9930677976d303653cd1670d3e9f35463e9'}]})
524         modules.append({'name': 'xmlsec1',
525                         'sources': [{'type': 'archive',
526                                      'url': '%s/xmlsec1-1.2.25.tar.gz' % prefix,
527                                      'sha256': '967ca83edf25ccb5b48a3c4a09ad3405a63365576503bf34290a42de1b92fcd2'}]})
528         modules.append({'name': 'openjpeg2',
529                         'buildsystem': 'cmake',
530                         'sources': [{'type': 'dir', 'path': os.path.abspath('../openjpeg2-cdist')}]})
531         modules.append({'name': 'boost',
532                         'buildsystem': 'simple',
533                         'build-commands': [
534                             './bootstrap.sh --prefix=/app',
535                             './b2 install'
536                         ],
537                         'sources': [{'type': 'archive',
538                                      'url': '%s/boost_1_66_0.tar.bz2' % prefix,
539                                      'sha256': '5721818253e6a0989583192f96782c4a98eb6204965316df9f5ad75819225ca9'}]})
540         modules.append({'name': 'asdcplib',
541                         'buildsystem': 'simple',
542                         'build-commands': [
543                             './waf configure --prefix=/app  --libdir=/app/lib build install'
544                         ],
545                         'sources': [{'type': 'dir', 'path': os.path.abspath('../asdcplib-carl')}]})
546         modules.append({'name': 'locked_sstream',
547                         'buildsystem': 'simple',
548                         'build-commands': [
549                             './waf configure --prefix=/app build install'
550                         ],
551                         'sources': [{'type': 'dir', 'path': os.path.abspath('../locked_sstream')}]})
552         modules.append({'name': 'libcxml',
553                         'buildsystem': 'simple',
554                         'build-commands': [
555                             './waf configure --prefix=/app  --libdir=/app/lib build install'
556                         ],
557                         'sources': [{'type': 'dir', 'path': os.path.abspath('../libcxml')}]})
558         modules.append({'name': 'libdcp',
559                         'buildsystem': 'simple',
560                         'build-commands': [
561                             './waf configure --prefix=/app --libdir=/app/lib build install'
562                         ],
563                         'sources': [{'type': 'dir', 'path': os.path.abspath('../libdcp')}]})
564         modules.append({'name': 'libsub',
565                         'buildsystem': 'simple',
566                         'build-commands': [
567                             './waf configure --prefix=/app --libdir=/app/lib build install'
568                         ],
569                         'sources': [{'type': 'dir', 'path': os.path.abspath('../libsub')}]})
570         modules.append({'name': 'rtaudio',
571                         'build-options': {
572                         'config-opts': [
573                             '--prefix=/app',
574                             '--with-pulse',
575                             '--with-alsa'
576                         ]
577                         },
578                         'sources': [{'type': 'dir', 'path': os.path.abspath('../rtaudio-cdist')}]})
579         modules.append({'name': 'wxwidgets',
580                         'sources': [{'type': 'archive',
581                                      'url': '%s/wxWidgets-3.0.3.tar.bz2' % prefix,
582                                      'sha256': '08c8033f48ec1b23520f036cde37b5ae925a6a65f137ded665633ca159b9307b'}]})
583         modules.append({'name': 'libssh',
584                         'buildsystem': 'cmake',
585                         'builddir': True,
586                         'sources': [{'type': 'archive',
587                                      'url': '%s/libssh-0.7.5.tar.xz' % prefix,
588                                      'sha256': '54e86dd5dc20e5367e58f3caab337ce37675f863f80df85b6b1614966a337095'}]})
589         modules.append({'name': 'dcpomatic',
590                         'buildsystem': 'simple',
591                         'build-commands': [
592                             './waf configure --prefix=/app build install'
593                          ],
594                         'build-options': {
595                            'build-args': ['--share=network']
596                         },
597                         'sources': [{'type': 'dir', 'path': os.path.abspath('.')}]})
598         desc = {'app-id': 'com.dcpomatic.DCP-o-matic',
599                 'runtime': 'org.gnome.Sdk',
600                 'runtime-version': '3.26',
601                 'sdk': 'org.gnome.Sdk',
602                 'command': 'dcpomatic2',
603                 'finish-args': ['--socket=x11', '--share=ipc', '--share=network', '--socket=pulseaudio', '--filesystem=host'],
604                 'modules': modules}
605         os.makedirs('build/platform')
606         with open('build/com.dcpomatic.DCP-o-matic.json', 'w') as outfile:
607             json.dump(desc, outfile)
608         target.command('%s --repo=build/platform/repo build/platform/flatpak build/com.dcpomatic.DCP-o-matic.json' % target.flatpak_builder())
609     else:
610         target.command('./waf configure --prefix=%s %s' % (target.directory, configure_options(target, options)))
611         target.command('./waf')
612         target.command('./waf install')
613
614 def package_windows(target):
615     identifier = ''
616     if target.version is not None:
617         identifier = '%s.' % target.version
618     identifier += '%d' % target.bits
619     shutil.copyfile('build/platform/windows/installer.%s.nsi' % identifier, 'build/platform/windows/installer2.%s.nsi' % identifier)
620     target.command('sed -i "s~%%resources%%~%s/platform/windows~g" build/platform/windows/installer2.%s.nsi' % (os.getcwd(), identifier))
621     target.command('sed -i "s~%%graphics%%~%s/graphics~g" build/platform/windows/installer2.%s.nsi' % (os.getcwd(), identifier))
622     target.command('sed -i "s~%%static_deps%%~%s~g" build/platform/windows/installer2.%s.nsi' % (target.windows_prefix, identifier))
623     target.command('sed -i "s~%%cdist_deps%%~%s~g" build/platform/windows/installer2.%s.nsi' % (target.directory, identifier))
624     target.command('sed -i "s~%%mingw%%~%s~g" build/platform/windows/installer2.%s.nsi' % (target.environment_prefix, identifier))
625     target.command('sed -i "s~%%binaries%%~%s/build~g" build/platform/windows/installer2.%s.nsi' % (os.getcwd(), identifier))
626     target.command('sed -i "s~%%bits%%~32~g" build/platform/windows/installer2.%s.nsi' % identifier)
627     target.command('makensis build/platform/windows/installer2.%s.nsi' % identifier)
628     return os.path.abspath(glob.glob('build/platform/windows/*%s*.exe' % target.bits)[0])
629
630 def package_debian(target, cpu, version, options):
631     make_control(target.version, target.bits, 'debian/control', target.debug, options['gui'])
632     if target.version != '9' and target.version != '16.04' and options['gui']:
633         with open('debian/postinst', 'w') as f:
634             print('#!/bin/sh', file=f)
635             # Get the required capability to write to disks
636             print('setcap "cap_dac_override+ep cap_sys_admin+ep" /usr/bin/dcpomatic2_disk_writer', file=f)
637     target.command('./waf dist')
638     f = open('debian/files', 'w')
639     suffix = '' if options['gui'] else '-cli'
640     print(f'dcpomatic{suffix}_{version}-1_{cpu}.deb video extra', file=f)
641     shutil.rmtree('build/deb', ignore_errors=True)
642
643     os.makedirs('build/deb')
644     os.chdir('build/deb')
645     shutil.move('../../dcpomatic-%s.tar.bz2' % version, 'dcpomatic_%s.orig.tar.bz2' % version)
646     target.command('tar xjf dcpomatic_%s.orig.tar.bz2' % version)
647     os.chdir('dcpomatic-%s' % version)
648     target.set('EMAIL', 'carl@dcpomatic.com')
649     target.command('dch -b -v %s-1 "New upstream release."' % version)
650     target.set('CDIST_LINKFLAGS', target.get('LINKFLAGS'))
651     target.set('CDIST_CXXFLAGS', target.get('CXXFLAGS'))
652     target.set('CDIST_PKG_CONFIG_PATH', target.get('PKG_CONFIG_PATH'))
653     target.set('CDIST_DIRECTORY', target.directory)
654
655     target.set('CDIST_CONFIGURE', '"' + configure_options(target, options) + '"')
656     target.set('CDIST_PACKAGE', f'dcpomatic{suffix}')
657     if target.debug:
658         target.set('CDIST_DEBUG_PACKAGE_FLAG', f'--dbg-package=dcpomatic{suffix}-dbg')
659
660     target.command('dpkg-buildpackage -uc -us')
661
662     debs = []
663     for p in glob.glob('../*.deb'):
664         debs.append(os.path.abspath(p))
665
666     return debs
667
668 def package_rpm(target, cpu, version, options):
669     topdir = os.path.realpath('build/rpmbuild')
670     os.makedirs('%s/BUILD' % topdir)
671     os.makedirs('%s/RPMS' % topdir)
672     os.makedirs('%s/SOURCES' % topdir)
673     os.makedirs('%s/SPECS' % topdir)
674     os.makedirs('%s/SRPMS' % topdir)
675
676     target.command('./waf dist')
677     shutil.copyfile(
678         "%s/src/dcpomatic/dcpomatic-%s.tar.bz2" % (target.directory, version),
679         "%s/SOURCES/dcpomatic-%s.tar.bz2" % (topdir, version)
680         )
681
682     requires = None
683     if target.distro == 'mageia':
684         requires = "lib64xmlsec1-devel lib64canberra-gtk0 libcap-utils"
685
686     make_spec('build/platform/linux/dcpomatic2.spec', version, target, options, requires)
687     cmd = 'rpmbuild --define "_topdir %s" -bb build/platform/linux/dcpomatic2.spec' % topdir
688     target.command(cmd)
689     rpms = []
690
691     if cpu == "amd64":
692         cpu = "x86_64"
693     else:
694         cpu = "i686"
695
696     for p in glob.glob('%s/RPMS/%s/*.rpm' % (topdir, cpu)):
697         rpms.append(os.path.abspath(p))
698
699     return rpms
700
701 def make_appimage(target, nice_name, internal_name, version):
702     nice_filename = nice_name.replace(' ', '_')
703     appdir = f'build/{nice_filename}.AppDir'
704     os.makedirs(f'{appdir}/usr/bin')
705     target.command(f'cp {target.directory}/bin/{internal_name} {appdir}/usr/bin')
706     target.command(f'cp {target.directory}/src/openssl/apps/openssl {appdir}/usr/bin/dcpomatic2_openssl')
707     target.command(f'cp {target.directory}/bin/dcpverify {appdir}/usr/bin/dcpomatic2_verify')
708     target.command(f'mkdir -p {appdir}/usr/share/libdcp')
709     target.command(f'cp -r {target.directory}/share/dcpomatic2 {appdir}/usr/share/')
710     target.command(f'cp -r {target.directory}/share/libdcp/xsd {appdir}/usr/share/libdcp/')
711     target.command(f'cp -r {target.directory}/share/libdcp/tags {appdir}/usr/share/libdcp/')
712     lib = 'usr/lib/x86_64-linux-gnu'
713     target.command(f'mkdir -p build/{nice_filename}.AppDir/{lib}/gdk-pixbuf-2.0/2.10.0')
714     target.command(f'cp -a /{lib}/gdk-pixbuf-2.0 build/{nice_filename}.AppDir/usr/lib/x86_64-linux-gnu/')
715     target.command('apt update')
716     for package in ['libc6', 'libglib2.0-0', 'gnome-settings-daemon-schemas', 'librsvg2-common', 'libgdk-pixbuf2.0-0', 'libpango-1.0-0', 'libpangoft2-1.0-0', 'libpangocairo-1.0-0']:
717         target.command(f'apt download {package}')
718         target.command(f'dpkg-deb -x {package}*.deb {appdir}')
719     target.command(f'glib-compile-schemas {appdir}/usr/share/glib-2.0/schemas')
720     target.command(f'sed -i -e "s|/usr/lib/x86_64-linux-gnu/gdk-pixbuf-.*/.*/loaders/||g" {appdir}/usr/lib/x86_64-linux-gnu/gdk-pixbuf-*/*/loaders.cache')
721     # Stop anything loading from outside the AppImage
722     target.command(f'sed -i -e "s|/usr|/xxx|g" {appdir}/lib/x86_64-linux-gnu/ld-linux-x86-64.so.2')
723
724     if internal_name == 'dcpomatic2_disk':
725         target.command(f'mkdir -p {appdir}/usr/share/polkit-1/actions')
726         target.command(f'cp {target.directory}/share/polkit-1/actions/com.dcpomatic.write-drive.policy {appdir}/usr/share/polkit-1/actions')
727
728     with open(f'{appdir}/AppRun', 'w') as f:
729         print('#!/bin/bash', file=f)
730         print('export PATH=$APPDIR/usr/bin:$PATH', file=f)
731         print('export XDG_DATA_DIRS="$APPDIR/usr/share/:/usr/share/:$XDG_DATA_DIRS"', file=f)
732         print('export GDK_PIXBUF_MODULEDIR=$(readlink -f "$APPDIR"/usr/lib/x86_64-linux-gnu/gdk-pixbuf-*/*/loaders/ )', file=f)
733         print('export GDK_PIXBUF_MODULE_FILE=$(readlink -f "$APPDIR"/usr/lib/x86_64-linux-gnu/gdk-pixbuf-*/*/loaders.cache )', file=f)
734         print('export LD_LIBRARY_PATH=$GDK_PIXBUF_MODULEDIR:$APPDIR/usr/lib:$APPDIR/usr/lib/x86_64-linux-gnu', file=f)
735         print(f'"$APPDIR"/usr/bin/{internal_name} $@', file=f)
736     target.command(f'chmod a+rx {appdir}/AppRun')
737     with open(f'{appdir}/{internal_name}.desktop', 'w') as f:
738         print('[Desktop Entry]', file=f)
739         print('Type=Application', file=f)
740         print('Categories=AudioVideo;', file=f)
741         print(f'Name={nice_name}', file=f)
742         print(f'Icon={internal_name}', file=f)
743     target.command(f'cp graphics/linux/256/{internal_name}.png {appdir}')
744     target.command(f'linuxdeploy-x86_64.AppImage --appdir {appdir}')
745     target.command(f'appimagetool-x86_64.AppImage {appdir}')
746     target.command(f'mv {nice_filename}-x86_64.AppImage build/{nice_filename}-{version}-x86_64.AppImage')
747     return os.path.abspath(f'build/{nice_filename}-{version}-x86_64.AppImage')
748
749 def package(target, version, options):
750     """version: DCP-o-matic version string"""
751     if target.platform == 'windows':
752         return package_windows(target)
753     elif target.platform == 'linux':
754         if target.detail == 'appimage':
755             out = []
756             out.append(make_appimage(target, 'DCP-o-matic', 'dcpomatic2', version))
757             out.append(make_appimage(target, 'DCP-o-matic Player', 'dcpomatic2_player', version))
758             out.append(make_appimage(target, 'DCP-o-matic Playlist Editor', 'dcpomatic2_playlist', version))
759             out.append(make_appimage(target, 'DCP-o-matic KDM Creator', 'dcpomatic2_kdm', version))
760             out.append(make_appimage(target, 'DCP-o-matic Batch Converter', 'dcpomatic2_batch', version))
761             out.append(make_appimage(target, 'DCP-o-matic Encode Server', 'dcpomatic2_server', version))
762             out.append(make_appimage(target, 'DCP-o-matic Combiner', 'dcpomatic2_combiner', version))
763             return out
764         else:
765             if target.bits == 32:
766                 cpu = 'i386'
767             else:
768                 cpu = 'amd64'
769
770             if target.distro == 'debian' or target.distro == 'ubuntu':
771                 return package_debian(target, cpu, version, options)
772             elif target.distro == 'centos' or target.distro == 'fedora' or target.distro == 'mageia':
773                 return package_rpm(target, cpu, version, options)
774     elif target.platform == 'osx':
775         archs = ' '.join(f'{t.arch}/{t.deployment}' for t in target.sub_targets)
776         target.command('bash platform/osx/make_dmg.sh %s %s %s %s %s' % (target.environment_prefix, target.directory, target.apple_id, target.apple_password, archs))
777         packages = []
778         for x in glob.glob('build/platform/osx/DCP-o-matic*.dmg'):
779             a = os.path.abspath(x)
780             if x.find("Player") != -1:
781                 packages.append((a, "com.dcpomatic.player"))
782             elif x.find("Playlist Editor") != -1:
783                 packages.append((a, "com.dcpomatic.playlist"))
784             elif x.find("KDM Creator") != -1:
785                 packages.append((a, "com.dcpomatic.kdm"))
786             elif x.find("Batch Converter") != -1:
787                 packages.append((a, "com.dcpomatic.batch"))
788             elif x.find("Encode Server") != -1:
789                 packages.append((a, "com.dcpomatic.server"))
790             elif x.find("Disk Writer") != -1:
791                 packages.append((a, "com.dcpomatic.disk"))
792             elif x.find("Combiner") != -1:
793                 packages.append((a, "com.dcpomatic.combiner"))
794             else:
795                 packages.append((a, "com.dcpomatic"))
796         return packages
797     elif target.platform == 'docker':
798         shutil.copyfile(target.deb, 'build/platform/docker')
799         f = open('build/platform/docker/Dockerfile', 'w')
800         print('FROM debian:jessie', file=f)
801         print('MAINTAINER carl@dcpomatic.com', file=f)
802         print('ADD build/platform/docker/dcpomatic_%s-1_amd64.deb /tmp' % (version, version), file=f)
803         print('RUN apt-get -o Acquire:http::Timeout="5" update; exit 0', file=f)
804         print('RUN dpkg -i /tmp/dcpomatic_*.deb; exit 0', file=f)
805         print('RUN apt-get -y -f install', file=f)
806         print('RUN apt-get clean', file=f)
807         print('EXPOSE 6192', file=f)
808         print('CMD ["/usr/bin/dcpomatic2_server_cli", "--verbose"]', file=f)
809         f.close()
810         target.command('docker build build/platform/docker -t dcpomatic-server:%s' % version)
811         target.command('docker save dcpomatic-server:%s -o dcpomatic-server-%s-docker.tar' % (version, version))
812     elif target.platform == 'flatpak':
813         target.command('%s build-bundle build/platform/repo build/dcpomatic_%s.flatpak com.dcpomatic.DCP-o-matic' % (target.flatpak(), version))
814         return os.path.abspath('build/dcpomatic_%s.flatpak' % version)
815
816 def make_pot(target):
817     target.command('./waf pot')
818     return [os.path.abspath('build/src/lib/libdcpomatic.pot'),
819             os.path.abspath('build/src/wx/libdcpomatic-wx.pot'),
820             os.path.abspath('build/src/tools/dcpomatic.pot')]
821
822 def make_manual(target):
823     target.command('make -C doc/manual LIBDCP=../../../libdcp')
824     os.chdir('doc/manual')
825     target.command('pdflatex colour.tex')
826     return [os.path.abspath('pdf'), os.path.abspath('html'), os.path.abspath('colour.pdf')]
827
828 def test(target, options, test):
829     target.set('LC_ALL', 'C')
830     if target.platform == 'windows':
831         cmd = 'run\\tests '
832     else:
833         cmd = 'run/tests '
834     if target.debug:
835         cmd += '--backtrace '
836     if test is not None:
837         cmd += '-t %s' % test
838     target.command(cmd)