Fixed SSL certificate validation (again)
[zeroinstall.git] / tests / testmodel.py
blob20105c70d35453d456dd90ed4f632524d9605eb3
1 #!/usr/bin/env python
2 # -*- coding: utf-8 -*-
3 from __future__ import unicode_literals
4 import basetest
5 from basetest import BaseTest, empty_feed
6 import sys, os
7 from xml.dom import minidom
8 import unittest
9 from io import BytesIO
11 sys.path.insert(0, '..')
12 from zeroinstall.injector import model, qdom, namespaces
14 mydir = os.path.dirname(os.path.abspath(__file__))
16 class TestModel(BaseTest):
17 def testLevels(self):
18 assert model.network_offline in model.network_levels
19 assert model.network_minimal in model.network_levels
20 assert model.network_full in model.network_levels
22 def testStabilities(self):
23 assert 'insecure' in model.stability_levels
24 assert 'buggy' in model.stability_levels
25 assert 'developer' in model.stability_levels
26 assert 'testing' in model.stability_levels
27 assert 'stable' in model.stability_levels
28 assert 'preferred' in model.stability_levels
29 str(model.insecure)
31 def testEscape(self):
32 self.assertEqual("", model.escape(""))
33 self.assertEqual("hello", model.escape("hello"))
34 self.assertEqual("%20", model.escape(" "))
36 self.assertEqual("file%3a%2f%2ffoo%7ebar",
37 model.escape("file://foo~bar"))
38 self.assertEqual("file%3a%2f%2ffoo%25bar",
39 model.escape("file://foo%bar"))
41 self.assertEqual("file:##foo%7ebar",
42 model._pretty_escape("file://foo~bar"))
43 self.assertEqual("file:##foo%25bar",
44 model._pretty_escape("file://foo%bar"))
46 def testUnescape(self):
47 self.assertEqual("", model.unescape(""))
48 self.assertEqual("hello", model.unescape("hello"))
49 self.assertEqual(" ", model.unescape("%20"))
51 self.assertEqual("file://foo~bar",
52 model.unescape("file%3a%2f%2ffoo%7ebar"))
53 self.assertEqual("file://foo%bar",
54 model.unescape("file%3a%2f%2ffoo%25bar"))
56 self.assertEqual("file://foo",
57 model.unescape("file:##foo"))
58 self.assertEqual("file://foo~bar",
59 model.unescape("file:##foo%7ebar"))
60 self.assertEqual("file://foo%bar",
61 model.unescape("file:##foo%25bar"))
63 def testEscaping(self):
64 def check(str):
65 self.assertEqual(str, model.unescape(model.escape(str)))
66 self.assertEqual(str, model.unescape(model._pretty_escape(str)))
68 check('http://example.com')
69 check('http://example%46com')
70 check('http:##example#com')
71 check('http://example.com/foo/bar.xml')
72 check('%20%21~&!"£ :@;,./{}$%^&()')
74 def testBadInterface(self):
75 try:
76 model.Interface('foo')
77 assert 0
78 except model.SafeException:
79 pass
81 def testInterface(self):
82 i = model.Interface('http://foo')
83 self.assertEqual('(foo)', i.get_name())
84 feed = model.ZeroInstallFeed(empty_feed, local_path = '/foo')
85 self.assertEqual('Empty', feed.get_name())
86 repr(i)
88 def testMetadata(self):
89 main_feed = model.ZeroInstallFeed(empty_feed, local_path = '/foo')
90 e = qdom.parse(BytesIO(b'<ns:b xmlns:ns="a" foo="bar"/>'))
91 main_feed.metadata = [e]
92 assert main_feed.get_metadata('a', 'b') == [e]
93 assert main_feed.get_metadata('b', 'b') == []
94 assert main_feed.get_metadata('a', 'a') == []
95 assert e.getAttribute('foo') == 'bar'
97 def testLocale(self):
98 local_path = os.path.join(mydir, 'Local.xml')
99 dom = qdom.parse(open(local_path))
100 feed = model.ZeroInstallFeed(dom, local_path = local_path)
101 # (defaults to en-US if no language is set in the locale)
102 self.assertEqual("Local feed (English)", feed.summary)
103 self.assertEqual("English", feed.description)
105 self.assertEqual(4, len(feed.summaries))
106 self.assertEqual(2, len(feed.descriptions))
108 try:
109 basetest.test_locale = ('es_ES', 'UTF8')
111 self.assertEqual("Fuente local", feed.summary)
112 self.assertEqual("Español", feed.description)
114 basetest.test_locale = ('en_GB', 'UTF8')
116 self.assertEqual("Local feed (English GB)", feed.summary)
118 basetest.test_locale = ('fr_FR', 'UTF8')
120 self.assertEqual("Local feed (English)", feed.summary)
121 self.assertEqual("English", feed.description)
122 finally:
123 basetest.test_locale = (None, None)
125 def testCommand(self):
126 local_path = os.path.join(mydir, 'Command.xml')
127 dom = qdom.parse(open(local_path))
128 feed = model.ZeroInstallFeed(dom, local_path = local_path)
130 assert feed.implementations['a'].main == 'foo'
131 assert feed.implementations['a'].commands['run'].path == 'foo'
132 assert feed.implementations['a'].commands['test'].path == 'test-foo'
134 assert feed.implementations['b'].main == 'bar'
135 assert feed.implementations['b'].commands['run'].path == 'bar'
136 assert feed.implementations['b'].commands['test'].path == 'test-foo'
138 assert feed.implementations['c'].main == 'test-gui'
139 assert feed.implementations['c'].commands['run'].path == 'test-gui'
140 assert feed.implementations['c'].commands['test'].path == 'test-baz'
142 def testStabPolicy(self):
143 i = model.Interface('http://foo')
144 self.assertEqual(None, i.stability_policy)
145 i.set_stability_policy(model.buggy)
146 self.assertEqual(model.buggy, i.stability_policy)
148 def testImpl(self):
149 i = model.Interface('http://foo')
150 a = model.ZeroInstallImplementation(i, 'foo', None)
151 assert a.id == 'foo'
152 assert a.size == a.version == a.user_stability == None
153 assert a.arch == a.upstream_stability == None
154 assert a.dependencies == {}
155 assert a.download_sources == []
156 assert a.get_stability() is model.testing
157 a.upstream_stability = model.stable
158 assert a.get_stability() is model.stable
159 a.user_stability = model.buggy
160 assert a.get_stability() is model.buggy
161 a.version = model.parse_version('1.2.3')
162 self.assertEqual('1.2.3', a.get_version())
163 a.version = model.parse_version('1.2.3-rc2-post')
164 self.assertEqual('1.2.3-rc2-post', a.get_version())
165 assert str(a) == 'foo'
167 b = model.ZeroInstallImplementation(i, 'foo', None)
168 b.version = model.parse_version("1.2.1")
169 assert b > a
171 def testDownloadSource(self):
172 f = model.ZeroInstallFeed(empty_feed, local_path = '/foo')
173 a = model.ZeroInstallImplementation(f, 'foo', None)
174 a.add_download_source('ftp://foo', 1024, None)
175 a.add_download_source('ftp://foo.tgz', 1025, 'foo')
176 assert a.download_sources[0].url == 'ftp://foo'
177 assert a.download_sources[0].size == 1024
178 assert a.download_sources[0].extract == None
179 assert a.feed is f
181 def testEnvBind(self):
182 a = model.EnvironmentBinding('PYTHONPATH', 'path')
183 assert a.name == 'PYTHONPATH'
184 assert a.insert == 'path'
185 str(a)
187 def testEnvModes(self):
188 prepend = model.EnvironmentBinding('PYTHONPATH', 'lib', None, model.EnvironmentBinding.PREPEND)
189 assert prepend.name == 'PYTHONPATH'
190 assert prepend.insert == 'lib'
191 assert prepend.mode is model.EnvironmentBinding.PREPEND
193 self.assertEqual('/impl/lib:/usr/lib', prepend.get_value('/impl', '/usr/lib'))
194 self.assertEqual('/impl/lib', prepend.get_value('/impl', None))
196 append = model.EnvironmentBinding('PYTHONPATH', 'lib', '/opt/lib', model.EnvironmentBinding.APPEND)
197 assert append.name == 'PYTHONPATH'
198 assert append.insert == 'lib'
199 assert append.mode is model.EnvironmentBinding.APPEND
201 self.assertEqual('/usr/lib:/impl/lib', append.get_value('/impl', '/usr/lib'))
202 self.assertEqual('/opt/lib:/impl/lib', append.get_value('/impl', None))
204 append = model.EnvironmentBinding('PYTHONPATH', 'lib', None, model.EnvironmentBinding.REPLACE)
205 assert append.name == 'PYTHONPATH'
206 assert append.insert == 'lib'
207 assert append.mode is model.EnvironmentBinding.REPLACE
209 self.assertEqual('/impl/lib', append.get_value('/impl', '/usr/lib'))
210 self.assertEqual('/impl/lib', append.get_value('/impl', None))
212 assert model.EnvironmentBinding('PYTHONPATH', 'lib').mode == model.EnvironmentBinding.PREPEND
214 def testOverlay(self):
215 for xml, expected in [(b'<overlay/>', '<overlay . on />'),
216 (b'<overlay src="usr"/>', '<overlay usr on />'),
217 (b'<overlay src="package" mount-point="/usr/games"/>', '<overlay package on /usr/games>')]:
218 e = qdom.parse(BytesIO(xml))
219 ol = model.process_binding(e)
220 self.assertEqual(expected, str(ol))
222 doc = minidom.parseString('<doc/>')
223 new_xml = str(ol._toxml(doc, None).toxml())
224 new_e = qdom.parse(BytesIO(new_xml))
225 new_ol = model.process_binding(new_e)
226 self.assertEqual(expected, str(new_ol))
228 def testDep(self):
229 b = model.InterfaceDependency('http://foo', element = qdom.Element(namespaces.XMLNS_IFACE, 'requires', {}))
230 assert not b.restrictions
231 assert not b.bindings
232 str(b)
234 def testFeed(self):
235 f = model.Feed('http://feed', arch = None, user_override = False)
236 assert f.uri == 'http://feed'
237 assert f.os == None
238 assert f.machine == None
239 assert f.arch == None
240 assert f.user_override == False
242 f = model.Feed('http://feed', arch = 'Linux-*', user_override = True)
243 assert f.uri == 'http://feed'
244 assert f.os == 'Linux'
245 assert f.machine == None
246 assert f.arch == 'Linux-*'
247 assert f.user_override == True
249 f = model.Feed('http://feed', arch = '*-i386', user_override = True)
250 assert f.uri == 'http://feed'
251 assert f.os == None
252 assert f.machine == 'i386'
253 assert f.arch == '*-i386'
254 assert f.user_override == True
255 assert str(f).startswith('<Feed from')
257 try:
258 f = model.Feed('http://feed', arch = 'i386', user_override = True)
259 assert False
260 except model.SafeException as ex:
261 assert 'Malformed arch' in str(ex)
263 def testCanonical(self):
264 # HTTP
265 try:
266 model.canonical_iface_uri('http://foo')
267 assert False
268 except model.SafeException as ex:
269 assert 'Missing /' in str(ex)
271 self.assertEqual('http://foo/',
272 model.canonical_iface_uri('http://foo/'))
273 try:
274 model.canonical_iface_uri('bad-name')
275 assert False
276 except model.SafeException as ex:
277 assert 'Bad interface name' in str(ex)
279 # Bare relative path
280 model.canonical_iface_uri('Command.xml')
281 try:
282 model.canonical_iface_uri('CommandMissing.xml')
283 assert False
284 except model.SafeException as ex:
285 assert "Bad interface name 'CommandMissing.xml'" in str(ex), ex
287 # file:absolute
288 model.canonical_iface_uri('file://{path}/Command.xml'.format(path = mydir))
289 try:
290 print model.canonical_iface_uri('file://{path}/CommandMissing.xml'.format(path = mydir))
291 assert False
292 except model.SafeException as ex:
293 assert "Bad interface name 'file://" in str(ex), ex
295 # file:relative
296 model.canonical_iface_uri('file:Command.xml')
297 try:
298 model.canonical_iface_uri('file:CommandMissing.xml')
299 assert False
300 except model.SafeException as ex:
301 assert "Bad interface name 'file:CommandMissing.xml'" in str(ex), ex
304 def testVersions(self):
305 def pv(v):
306 parsed = model.parse_version(v)
307 assert model.format_version(parsed) == v
308 return parsed
310 assert pv('1.0') > pv('0.9')
311 assert pv('1.0') > pv('1')
312 assert pv('1.0') == pv('1.0')
313 assert pv('0.9.9') < pv('1.0')
314 assert pv('10') > pv('2')
316 def invalid(v):
317 try:
318 pv(v)
319 assert False
320 except model.SafeException:
321 pass
322 invalid('.')
323 invalid('hello')
324 invalid('2./1')
325 invalid('.1')
326 invalid('')
328 # Check parsing
329 assert pv('1') == [[1], 0]
330 assert pv('1.0') == [[1,0], 0]
331 assert pv('1.0-pre5') == [[1,0], -2, [5], 0]
332 assert pv('1.0-rc5') == [[1,0], -1, [5], 0]
333 assert pv('1.0-5') == [[1,0], 0, [5], 0]
334 assert pv('1.0-post5') == [[1,0], 1, [5], 0]
335 assert pv('1.0-post') == [[1,0], 1]
336 assert pv('1-rc2.0-pre2-post') == [[1], -1, [2,0], -2, [2], 1]
337 assert pv('1-rc2.0-pre-post') == [[1], -1, [2,0], -2, [], 1]
339 assert pv('1.0-0') > pv('1.0')
340 assert pv('1.0-1') > pv('1.0-0')
341 assert pv('1.0-0') < pv('1.0-1')
343 assert pv('1.0-pre99') > pv('1.0-pre1')
344 assert pv('1.0-pre99') < pv('1.0-rc1')
345 assert pv('1.0-rc1') < pv('1.0')
346 assert pv('1.0') < pv('1.0-0')
347 assert pv('1.0-0') < pv('1.0-post')
348 assert pv('2.1.9-pre-1') > pv('2.1.9-pre')
350 assert pv('2-post999') < pv('3-pre1')
352 if __name__ == '__main__':
353 unittest.main()