LEFT | RIGHT |
1 # This Source Code Form is subject to the terms of the Mozilla Public | 1 # This Source Code Form is subject to the terms of the Mozilla Public |
2 # License, v. 2.0. If a copy of the MPL was not distributed with this | 2 # License, v. 2.0. If a copy of the MPL was not distributed with this |
3 # file, You can obtain one at http://mozilla.org/MPL/2.0/. | 3 # file, You can obtain one at http://mozilla.org/MPL/2.0/. |
4 | 4 |
5 import ConfigParser | 5 import ConfigParser |
| 6 import json |
6 import os | 7 import os |
7 import shutil | 8 import shutil |
8 import xml.etree.ElementTree as ET | 9 import xml.etree.ElementTree as ET |
9 import zipfile | 10 import zipfile |
10 | 11 |
11 import pytest | 12 import pytest |
12 | 13 |
13 from buildtools import packager, packagerEdge | 14 from buildtools import packager, packagerEdge |
14 | 15 |
15 TEST_DIR = os.path.dirname(__file__) | 16 TEST_DIR = os.path.dirname(__file__) |
16 TEST_METADATA = os.path.join(TEST_DIR, 'metadata.edge') | 17 TEST_METADATA = os.path.join(TEST_DIR, 'metadata.edge') |
17 CHARS = b''.join(chr(i % 200 + 30) for i in range(500)) | 18 CHARS = b''.join(chr(i % 200 + 30) for i in range(500)) |
| 19 MESSAGES_EN_US = json.dumps({ |
| 20 'name': {'message': 'Adblock Plus'}, |
| 21 'name_devbuild': {'message': 'devbuild-marker'}, |
| 22 'description': { |
| 23 'message': 'Adblock Plus is the most popular ad blocker ever, ' |
| 24 'and also supports websites by not blocking ' |
| 25 'unobstrusive ads by default (configurable).' |
| 26 }, |
| 27 }) |
18 | 28 |
19 | 29 |
20 @pytest.fixture | 30 @pytest.fixture |
21 def metadata(): | 31 def metadata(): |
22 """Loaded metadata config.""" | 32 """Loaded metadata config.""" |
23 conf_parser = ConfigParser.ConfigParser() | 33 conf_parser = ConfigParser.ConfigParser() |
24 conf_parser.read(TEST_METADATA) | 34 conf_parser.read(TEST_METADATA) |
25 return conf_parser | 35 return conf_parser |
26 | 36 |
27 | 37 |
28 @pytest.fixture | 38 @pytest.fixture |
| 39 def files(): |
| 40 """Minimal Files() for testing manifest and blockmap.""" |
| 41 files = packager.Files(set(), set()) |
| 42 for size in ['44', '50', '150']: |
| 43 files['Assets/logo_{}.png'.format(size)] = CHARS |
| 44 files['Extension/_locales/en_US/messages.json'] = MESSAGES_EN_US |
| 45 files['Extension/foo.xml'] = CHARS |
| 46 files['Extension/bar.png'] = CHARS * 200 |
| 47 return files |
| 48 |
| 49 |
| 50 @pytest.fixture |
29 def srcdir(tmpdir): | 51 def srcdir(tmpdir): |
30 """Source directory for building the package.""" | 52 """Source directory for building the package.""" |
31 srcdir = tmpdir.mkdir('src') | 53 srcdir = tmpdir.mkdir('src') |
32 shutil.copy(TEST_METADATA, str(srcdir.join('metadata.edge'))) | 54 shutil.copy(TEST_METADATA, str(srcdir.join('metadata.edge'))) |
33 for size in ['44', '50', '150']: | 55 for size in ['44', '50', '150']: |
34 path = srcdir.join('chrome', 'icons', 'abp-{}.png'.format(size)) | 56 path = srcdir.join('chrome', 'icons', 'abp-{}.png'.format(size)) |
35 path.write(size, ensure=True) | 57 path.write(size, ensure=True) |
| 58 localedir = srcdir.mkdir('_locales') |
| 59 en_us_dir = localedir.mkdir('en_US') |
| 60 en_us_dir.join('messages.json').write(MESSAGES_EN_US) |
36 return srcdir | 61 return srcdir |
37 | 62 |
38 | 63 |
39 def blockmap2dict(xml_data): | 64 def blockmap2dict(xml_data): |
40 """Convert AppxBlockMap.xml to a dict of dicts easier to inspect.""" | 65 """Convert AppxBlockMap.xml to a dict of dicts easier to inspect.""" |
41 return { | 66 return { |
42 file.get('Name'): { | 67 file.get('Name'): { |
43 'size': file.get('Size'), | 68 'size': file.get('Size'), |
44 'lfhsize': file.get('LfhSize'), | 69 'lfhsize': file.get('LfhSize'), |
45 'blocks': [ | 70 'blocks': [b.get('Hash') for b in file] |
46 {'hash': b.get('Hash'), 'size': b.get('Size', None)} | |
47 for b in file | |
48 ] | |
49 } | 71 } |
50 for file in ET.fromstring(xml_data) | 72 for file in ET.fromstring(xml_data) |
51 } | 73 } |
52 | 74 |
53 | 75 |
54 def test_create_appx_blockmap(): | 76 def test_create_appx_blockmap(files): |
55 files = packager.Files(set(), set()) | |
56 files['foo.xml'] = CHARS | |
57 files['foo/bar.png'] = CHARS * 200 | |
58 blockmap = blockmap2dict(packagerEdge.create_appx_blockmap(files)) | 77 blockmap = blockmap2dict(packagerEdge.create_appx_blockmap(files)) |
59 assert blockmap['foo.xml'] == { | 78 assert blockmap['Extension\\foo.xml'] == { |
60 'size': '500', | 79 'size': '500', |
61 'lfhsize': '37', | 80 'lfhsize': '47', |
| 81 'blocks': ['Vhwfmzss1Ney+j/ssR2QVISvFyMNBQeS2P+UjeE/di0='] |
| 82 } |
| 83 assert blockmap['Extension\\bar.png'] == { |
| 84 'size': '100000', |
| 85 'lfhsize': '47', |
62 'blocks': [ | 86 'blocks': [ |
63 {'hash': 'Vhwfmzss1Ney+j/ssR2QVISvFyMNBQeS2P+UjeE/di0=', | 87 'KPW2SxeEikUEGhoKmKxruUSexKun0bGXMppOqUFrX5E=', |
64 'size': None} | 88 'KQHnov1SZ1z34ttdDUjX2leYtpIIGndUVoUteieS2cw=', |
65 ] | |
66 } | |
67 assert blockmap['foo\\bar.png'] == { | |
68 'size': '100000', | |
69 'lfhsize': '41', | |
70 'blocks': [ | |
71 {'hash': 'KPW2SxeEikUEGhoKmKxruUSexKun0bGXMppOqUFrX5E=', | |
72 'size': None}, | |
73 {'hash': 'KQHnov1SZ1z34ttdDUjX2leYtpIIGndUVoUteieS2cw=', | |
74 'size': None} | |
75 ] | 89 ] |
76 } | 90 } |
77 | 91 |
78 | 92 |
79 def ctm2dict(content_types_map): | 93 def ctm2dict(content_types_map): |
80 """Convert content type map to a dict.""" | 94 """Convert content type map to a dict.""" |
81 ret = {'defaults': {}, 'overrides': {}} | 95 ret = {'defaults': {}, 'overrides': {}} |
82 for node in ET.fromstring(content_types_map): | 96 for node in ET.fromstring(content_types_map): |
83 ct = node.get('ContentType') | 97 ct = node.get('ContentType') |
84 if node.tag.endswith('Default'): | 98 if node.tag.endswith('Default'): |
85 ret['defaults'][node.get('Extension')] = ct | 99 ret['defaults'][node.get('Extension')] = ct |
86 elif node.tag.endswith('Override'): | 100 elif node.tag.endswith('Override'): |
87 ret['overrides'][node.get('PartName')] = ct | 101 ret['overrides'][node.get('PartName')] = ct |
88 else: | 102 else: |
89 raise ValueError('Unrecognised tag in content map: ' + node.tag) | 103 raise ValueError('Unrecognised tag in content map: ' + node.tag) |
90 return ret | 104 return ret |
91 | 105 |
92 | 106 |
93 def test_empty_content_types_map(): | 107 def test_empty_content_types_map(): |
94 files = packager.Files(set(), set()) | 108 ctm_dict = ctm2dict(packagerEdge.create_content_types_map([])) |
95 ctm_dict = ctm2dict(packagerEdge.create_content_types_map(files)) | 109 assert ctm_dict['defaults'] == {} |
96 assert ctm_dict['defaults'] == {'xml': 'text/xml'} | 110 assert ctm_dict['overrides'] == {} |
97 assert ctm_dict['overrides'] == { | |
98 '/AppxBlockMap.xml': 'application/vnd.ms-appx.blockmap+xml', | |
99 '/AppxManifest.xml': 'application/vnd.ms-appx.manifest+xml' | |
100 } | |
101 | 111 |
102 | 112 |
103 def test_full_content_types_map(): | 113 def test_full_content_types_map(): |
104 files = packager.Files(set(), set()) | 114 filenames = ['no-extension', packagerEdge.MANIFEST, packagerEdge.BLOCKMAP] |
105 for ext in 'json html js png css git otf'.split(): | 115 filenames += ['file.' + x for x in 'json html js png css git otf'.split()] |
106 files['file.' + ext] = 'data' | 116 ctm_dict = ctm2dict(packagerEdge.create_content_types_map(filenames)) |
107 ctm_dict = ctm2dict(packagerEdge.create_content_types_map(files)) | |
108 import pprint | |
109 pprint.pprint(ctm_dict['defaults']) | |
110 assert ctm_dict['defaults'] == { | 117 assert ctm_dict['defaults'] == { |
111 'css': 'text/css', | 118 'css': 'text/css', |
112 'html': 'text/html', | 119 'html': 'text/html', |
113 'js': 'application/javascript', | 120 'js': 'application/javascript', |
114 'json': 'application/json', | 121 'json': 'application/json', |
115 'otf': 'application/x-font-otf', | 122 'otf': 'application/x-font-otf', |
116 'png': 'image/png', | 123 'png': 'image/png', |
117 'xml': 'text/xml' | 124 'xml': 'application/xml' |
| 125 } |
| 126 assert ctm_dict['overrides'] == { |
| 127 '/AppxBlockMap.xml': 'application/vnd.ms-appx.blockmap+xml', |
| 128 '/AppxManifest.xml': 'application/vnd.ms-appx.manifest+xml' |
118 } | 129 } |
119 | 130 |
120 | 131 |
121 def test_create_appx_manifest(metadata): | 132 def test_create_appx_manifest(metadata, files): |
122 files = packager.Files(set(), set()) | 133 manifest = packagerEdge.create_appx_manifest( |
123 for size in ['44', '50', '150']: | 134 {'metadata': metadata}, files, release_build=True, |
124 files['Assets/logo_{}.png'.format(size)] = CHARS | 135 ) |
125 manifest = packagerEdge.create_appx_manifest({'metadata': metadata}, files) | |
126 with open(os.path.join(TEST_DIR, 'AppManifest.xml.expect')) as fp: | 136 with open(os.path.join(TEST_DIR, 'AppManifest.xml.expect')) as fp: |
127 manifest_expect = fp.read() | 137 manifest_expect = fp.read() |
128 assert manifest.strip() == manifest_expect.strip() | 138 assert manifest.strip() == manifest_expect.strip() |
| 139 |
| 140 |
| 141 def test_create_devbuild_appx_manifest(metadata, files): |
| 142 manifest = packagerEdge.create_appx_manifest( |
| 143 {'metadata': metadata}, files, release_build=False, |
| 144 ) |
| 145 assert 'devbuild-marker' in manifest |
129 | 146 |
130 | 147 |
131 def test_move_files_to_extension(): | 148 def test_move_files_to_extension(): |
132 files = packager.Files(set(), set()) | 149 files = packager.Files(set(), set()) |
133 files['foo.xml'] = CHARS | 150 files['foo.xml'] = CHARS |
134 files['foo/bar.xml'] = CHARS | 151 files['foo/bar.xml'] = CHARS |
135 files['Extension/foo.xml'] = CHARS | 152 files['Extension/foo.xml'] = CHARS |
136 packagerEdge.move_files_to_extension(files) | 153 packagerEdge.move_files_to_extension(files) |
137 assert set(files.keys()) == { | 154 assert set(files.keys()) == { |
138 'Extension/foo.xml', | 155 'Extension/foo.xml', |
139 'Extension/foo/bar.xml', | 156 'Extension/foo/bar.xml', |
140 'Extension/Extension/foo.xml' | 157 'Extension/Extension/foo.xml' |
141 } | 158 } |
142 | 159 |
143 | 160 |
144 def test_create_build(tmpdir, srcdir): | 161 def test_create_build(tmpdir, srcdir): |
145 out_file = str(tmpdir.join('abp.appx')) | 162 out_file = str(tmpdir.join('abp.appx')) |
146 packagerEdge.createBuild(str(srcdir), outFile=out_file) | 163 packagerEdge.createBuild(str(srcdir), outFile=out_file, releaseBuild=True) |
147 appx = zipfile.ZipFile(out_file) | 164 appx = zipfile.ZipFile(out_file) |
148 | 165 |
149 names = set(appx.namelist()) | 166 names = set(appx.namelist()) |
150 assert 'AppxManifest.xml' in names | 167 assert 'AppxManifest.xml' in names |
151 assert 'AppxBlockMap.xml' in names | 168 assert 'AppxBlockMap.xml' in names |
152 assert '[Content_Types].xml' in names | 169 assert '[Content_Types].xml' in names |
153 | 170 |
| 171 assert 'devbuild-marker' not in appx.read('AppxManifest.xml') |
154 assert appx.read('Assets/logo_44.png') == '44' | 172 assert appx.read('Assets/logo_44.png') == '44' |
155 assert appx.read('Extension/icons/abp-44.png') == '44' | 173 assert appx.read('Extension/icons/abp-44.png') == '44' |
| 174 |
| 175 |
| 176 def test_create_devbuild(tmpdir, srcdir): |
| 177 out_file = str(tmpdir.join('abp.appx')) |
| 178 packagerEdge.createBuild(str(srcdir), outFile=out_file, releaseBuild=False) |
| 179 appx = zipfile.ZipFile(out_file) |
| 180 assert 'devbuild-marker' in appx.read('AppxManifest.xml') |
LEFT | RIGHT |