| | 1 | # -*- coding: utf-8 -*- |
| | 2 | # |
| | 3 | # Copyright (C)2005-2010 Edgewall Software |
| | 4 | # Copyright (C) 2010 Itamar Ostricher <itamarost@gmail.com> |
| | 5 | # All rights reserved. |
| | 6 | # |
| | 7 | # This software is licensed as described in the file COPYING, which |
| | 8 | # you should have received as part of this distribution. The terms |
| | 9 | # are also available at http://trac.edgewall.org/wiki/TracLicense. |
| | 10 | # |
| | 11 | # This software consists of voluntary contributions made by many |
| | 12 | # individuals. For the exact contribution history, see the revision |
| | 13 | # history and logs, available at http://trac.edgewall.org/log/. |
| | 14 | # |
| | 15 | # Author: Itamar Ostricher <itamarost@gmail.com> |
| | 16 | |
| | 17 | import unittest |
| | 18 | import random |
| | 19 | from urllib import splittype |
| | 20 | |
| | 21 | from trac.test import Mock |
| | 22 | from trac.core import TracError |
| | 23 | from trac.versioncontrol import svn_prop |
| | 24 | |
| | 25 | class SvnExternalsParserTests(unittest.TestCase): |
| | 26 | |
| | 27 | def test_no_externals(self): |
| | 28 | externals_prop = u'' |
| | 29 | externals_list = svn_prop.parse_externals(externals_prop, |
| | 30 | reponame=u'myrep', repopath=u'dev/trunk/src/my-module') |
| | 31 | self.assertTrue(isinstance(externals_list, list)) |
| | 32 | self.assertEqual(0, len(externals_list)) |
| | 33 | |
| | 34 | def test_old_syntax_parsing(self): |
| | 35 | externals_prop = u""" |
| | 36 | simple http://example.org/svn/repos/dir1 |
| | 37 | # this is a comment that should be ignored. |
| | 38 | spaceless_rev -r10 svn://example.org/svn/repos/dir2 |
| | 39 | spaced_rev -r 12 svn+ssh://example.org/svn/repos/dir3 |
| | 40 | """ |
| | 41 | externals_list = svn_prop.parse_externals(externals_prop, |
| | 42 | reponame=u'myrep', repopath=u'dev/trunk/src/my-module') |
| | 43 | self.assertTrue(isinstance(externals_list, list)) |
| | 44 | self.assertEqual(3, len(externals_list)) |
| | 45 | for ext_dict in externals_list: |
| | 46 | self.assertTrue(isinstance(ext_dict, dict)) |
| | 47 | self.assertEqual(ext_dict['reponame'], u'myrep') |
| | 48 | self.assertEqual(ext_dict['repopath'], |
| | 49 | u'dev/trunk/src/my-module') |
| | 50 | simple, spaceless, spaced = externals_list |
| | 51 | self.assertEqual(simple['dir'], u'simple') |
| | 52 | self.assertEqual(simple['url'], |
| | 53 | u'http://example.org/svn/repos/dir1') |
| | 54 | self.assertTrue(simple['rev'] is None) |
| | 55 | self.assertEqual(spaceless['dir'], u'spaceless_rev') |
| | 56 | self.assertEqual(spaceless['url'], |
| | 57 | u'svn://example.org/svn/repos/dir2') |
| | 58 | self.assertEqual(10, spaceless['rev']) |
| | 59 | self.assertEqual(spaced['dir'], u'spaced_rev') |
| | 60 | self.assertEqual(spaced['url'], |
| | 61 | u'svn+ssh://example.org/svn/repos/dir3') |
| | 62 | self.assertEqual(12, spaced['rev']) |
| | 63 | |
| | 64 | def test_new_syntax_parsing(self): |
| | 65 | externals_prop = u""" |
| | 66 | https://example.org/svn/repos/dir1 simple |
| | 67 | -r10 ^/repos/dir2 spaceless_rev |
| | 68 | -r 12 //example.org/svn/repos/dir3 spaced_rev |
| | 69 | # this is a comment that should be ignored. |
| | 70 | ../dir4@17 peg_rev |
| | 71 | """ |
| | 72 | externals_list = svn_prop.parse_externals(externals_prop, |
| | 73 | reponame=u'myrep', repopath=u'dev/trunk/src/my-module') |
| | 74 | self.assertTrue(isinstance(externals_list, list)) |
| | 75 | self.assertEqual(4, len(externals_list)) |
| | 76 | for ext_dict in externals_list: |
| | 77 | self.assertTrue(isinstance(ext_dict, dict)) |
| | 78 | self.assertEqual(ext_dict['reponame'], u'myrep') |
| | 79 | self.assertEqual(ext_dict['repopath'], |
| | 80 | u'dev/trunk/src/my-module') |
| | 81 | simple, spaceless, spaced, pegged = externals_list |
| | 82 | self.assertEqual(simple['dir'], u'simple') |
| | 83 | self.assertEqual(simple['url'], |
| | 84 | u'https://example.org/svn/repos/dir1') |
| | 85 | self.assertTrue(simple['rev'] is None) |
| | 86 | self.assertEqual(spaceless['dir'], u'spaceless_rev') |
| | 87 | self.assertEqual(spaceless['url'], |
| | 88 | u'^/repos/dir2') |
| | 89 | self.assertEqual(10, spaceless['rev']) |
| | 90 | self.assertEqual(spaced['dir'], u'spaced_rev') |
| | 91 | self.assertEqual(spaced['url'], |
| | 92 | u'//example.org/svn/repos/dir3') |
| | 93 | self.assertEqual(12, spaced['rev']) |
| | 94 | self.assertEqual(pegged['dir'], u'peg_rev') |
| | 95 | self.assertEqual(pegged['url'], |
| | 96 | u'../dir4') |
| | 97 | self.assertEqual(17, pegged['rev']) |
| | 98 | |
| | 99 | # Repositories filesystem layout for following tests: |
| | 100 | # /var/svn |
| | 101 | # | |
| | 102 | # |-repos (svn root of repository "repos", AKA "myrep") |
| | 103 | # | | |
| | 104 | # | |-dev |
| | 105 | # | | |
| | 106 | # | \-devel |
| | 107 | # | |-common |
| | 108 | # | | \-trunk |
| | 109 | # | | \-log |
| | 110 | # | \-my-prod |
| | 111 | # | \-trunk |
| | 112 | # | \-src |
| | 113 | # | \-my-module |
| | 114 | # | |
| | 115 | # \-repos-2 (svn root of repository "repos-2", AKA "otherrep") |
| | 116 | # \-dev |
| | 117 | |
| | 118 | class SvnExternalsRepositoryRelativeLinkTests(unittest.TestCase): |
| | 119 | |
| | 120 | def setUp(self): |
| | 121 | self.repo_base = u'svn:random-uuid:/var/svn/repos' |
| | 122 | self.other_repo_base = u'svn:random-uuid:/var/svn/repos-2' |
| | 123 | self.repositories = { |
| | 124 | u'myrep': Mock(reponame=u'myrep', scope=u'/', |
| | 125 | get_base=lambda: self.repo_base, |
| | 126 | get_path_url=lambda path, rev: None), |
| | 127 | u'my-devel-rep': Mock(reponame=u'my-devel-rep', scope=u'/devel/', |
| | 128 | get_base=lambda: self.repo_base, |
| | 129 | get_path_url=lambda path, rev: None), |
| | 130 | u'my-prod-rep': Mock(reponame=u'my-prod-rep', |
| | 131 | scope=u'/devel/my-prod/', |
| | 132 | get_base=lambda: self.repo_base, |
| | 133 | get_path_url=lambda path, rev: None), |
| | 134 | # Note 'my-dev-rep' purposefully has a scope that prefixes |
| | 135 | # 'my-devel-rep'. This tests cases of false matching scopes. |
| | 136 | u'my-dev-rep': Mock(reponame=u'my-dev-rep', scope=u'/dev/', |
| | 137 | get_base=lambda: self.repo_base, |
| | 138 | get_path_url=lambda path, rev: None), |
| | 139 | u'otherrep': Mock(reponame=u'otherrep', scope=u'/', |
| | 140 | get_base=lambda: self.other_repo_base, |
| | 141 | get_path_url=lambda path, rev: None), |
| | 142 | } |
| | 143 | self.rm = Mock(get_repository=lambda reponame: |
| | 144 | self.repositories[reponame]) |
| | 145 | |
| | 146 | def test_repository_relative_url(self): |
| | 147 | external = {u'reponame': 'myrep', |
| | 148 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 149 | u'dir': u'local', |
| | 150 | u'url': u'^/devel/common/trunk/log', |
| | 151 | u'rev': None} |
| | 152 | scoped_external = {u'reponame': 'my-devel-rep', |
| | 153 | # Note the missing 'devel/' in the repo-path |
| | 154 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 155 | u'dir': u'local', |
| | 156 | u'url': u'^/devel/common/trunk/log', |
| | 157 | u'rev': None} |
| | 158 | deep_external = {u'reponame': 'my-devel-rep', |
| | 159 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 160 | u'repopath': u'trunk/src/my-module', |
| | 161 | u'dir': u'local', |
| | 162 | u'url': u'^/devel/common/trunk/log', |
| | 163 | u'rev': None} |
| | 164 | |
| | 165 | for ext in [external, scoped_external, deep_external]: |
| | 166 | # Un-scoped repository asserts |
| | 167 | repo = self.rm.get_repository(u'myrep') |
| | 168 | for scope in [u'', u'/']: |
| | 169 | repo.scope = scope |
| | 170 | # External as seen from same repository |
| | 171 | self.assertEqual(u'myrep/devel/common/trunk/log?rev=%(rev)s', |
| | 172 | svn_prop.get_repo_relative_path(ext, u'myrep', rm=self.rm)) |
| | 173 | # Scoped repository asserts |
| | 174 | repo = self.rm.get_repository(u'my-devel-rep') |
| | 175 | for scope in [u'devel', u'/devel', u'devel/', u'/devel/']: |
| | 176 | repo.scope = scope |
| | 177 | # External as seen from the un-scoped repository |
| | 178 | # Note the missing '/devel/' in the expected href |
| | 179 | self.assertEqual(u'my-devel-rep/common/trunk/log?rev=%(rev)s', |
| | 180 | svn_prop.get_repo_relative_path(ext, u'my-devel-rep', |
| | 181 | rm=self.rm)) |
| | 182 | # Deep-scoped repository (too deep for common) |
| | 183 | self.assertTrue(svn_prop.get_repo_relative_path(ext, |
| | 184 | u'my-prod-rep', rm=self.rm) is None) |
| | 185 | # Differently-scoped repository (out of range) |
| | 186 | self.assertTrue(svn_prop.get_repo_relative_path(ext, |
| | 187 | u'my-dev-rep', rm=self.rm) is None) |
| | 188 | # Another repository, not related to external |
| | 189 | self.assertTrue(svn_prop.get_repo_relative_path(ext, u'otherrep', |
| | 190 | rm=self.rm) is None) |
| | 191 | |
| | 192 | def test_directory_relative_url(self): |
| | 193 | external = {u'reponame': u'myrep', |
| | 194 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 195 | u'dir': u'local', |
| | 196 | u'url': u'../../../../common/trunk/log', |
| | 197 | u'rev': None} |
| | 198 | scoped_external = {u'reponame': u'my-devel-rep', |
| | 199 | # Note the missing 'devel/' in the repo-path |
| | 200 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 201 | u'dir': u'local', |
| | 202 | u'url': u'../../../../common/trunk/log', |
| | 203 | u'rev': None} |
| | 204 | deep_external = {u'reponame': u'my-prod-rep', |
| | 205 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 206 | u'repopath': u'trunk/src/my-module', |
| | 207 | u'dir': u'local', |
| | 208 | # Going even deeper just for fun (and test of course) |
| | 209 | u'url': u'../../../../../devel/common/trunk/log', |
| | 210 | u'rev': None} |
| | 211 | |
| | 212 | for ext in [external, scoped_external, deep_external]: |
| | 213 | # Un-scoped repository asserts |
| | 214 | self.assertEqual(u'myrep/devel/common/trunk/log?rev=%(rev)s', |
| | 215 | svn_prop.get_repo_relative_path(ext, u'myrep', |
| | 216 | rm=self.rm)) |
| | 217 | # Scoped repository asserts |
| | 218 | self.assertEqual(u'my-devel-rep/common/trunk/log?rev=%(rev)s', |
| | 219 | svn_prop.get_repo_relative_path(ext, |
| | 220 | u'my-devel-rep', |
| | 221 | rm=self.rm)) |
| | 222 | # Deep-scoped repository asserts (out of reach) |
| | 223 | self.assertTrue(svn_prop.get_repo_relative_path(ext, |
| | 224 | u'my-prod-rep', rm=self.rm) is None) |
| | 225 | # Unrelated repository asserts |
| | 226 | self.assertTrue(svn_prop.get_repo_relative_path(ext, |
| | 227 | u'otherrep', rm=self.rm) is None) |
| | 228 | |
| | 229 | def test_invalid_directory_relative_url(self): |
| | 230 | external = {u'reponame': u'myrep', |
| | 231 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 232 | u'dir': u'local', |
| | 233 | # Out of repository base! aaah! |
| | 234 | u'url': u'../../../../../../common/trunk/log', |
| | 235 | u'rev': None} |
| | 236 | scoped_external = {u'reponame': u'my-devel-rep', |
| | 237 | # Note the missing 'devel/' in the repo-path |
| | 238 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 239 | u'dir': u'local', |
| | 240 | u'url': u'../../../../../../common/trunk/log', |
| | 241 | u'rev': None} |
| | 242 | deep_external = {u'reponame': u'my-prod-rep', |
| | 243 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 244 | u'repopath': u'trunk/src/my-module', |
| | 245 | u'dir': u'local', |
| | 246 | # Going even deeper just for fun (and test of course) |
| | 247 | u'url': u'../../../../../../../devel/common/trunk/log', |
| | 248 | u'rev': None} |
| | 249 | |
| | 250 | for ext in [external, scoped_external, deep_external]: |
| | 251 | self.assertRaises(TracError, svn_prop.get_repo_relative_path, |
| | 252 | ext, u'myrep', rm=self.rm) |
| | 253 | self.assertRaises(TracError, svn_prop.get_repo_relative_path, |
| | 254 | ext, u'my-devel-rep', rm=self.rm) |
| | 255 | self.assertRaises(TracError, svn_prop.get_repo_relative_path, |
| | 256 | ext, u'my-prod-rep', rm=self.rm) |
| | 257 | self.assertTrue(svn_prop.get_repo_relative_path(ext, u'otherrep', |
| | 258 | rm=self.rm) is None) |
| | 259 | |
| | 260 | class SvnExternalsUrlBasedLinkTests(unittest.TestCase): |
| | 261 | |
| | 262 | def setUp(self): |
| | 263 | self.repo_base = u'svn:random-uuid:/var/svn/repos' |
| | 264 | self.repo_host = u'example.org' |
| | 265 | self.repo_base_url = u'//%s/svn/myrep' % (self.repo_host) |
| | 266 | self.other_repo_base = u'svn:random-uuid:/var/svn/repos-2' |
| | 267 | self.other_repo_base_url = u'//%s/svn/otherrep' % (self.repo_host) |
| | 268 | self.repositories = { |
| | 269 | u'myrep': Mock(reponame=u'myrep', scope=u'/', |
| | 270 | get_base=lambda: self.repo_base, |
| | 271 | get_path_url=lambda path, rev: u'https:%s/%s' % |
| | 272 | (self.repo_base_url, path.lstrip(u'/'))), |
| | 273 | u'my-devel-rep': Mock(reponame=u'my-devel-rep', scope=u'/devel/', |
| | 274 | get_base=lambda: self.repo_base, |
| | 275 | get_path_url=lambda path, rev: |
| | 276 | u'http:%s/devel/%s' % |
| | 277 | (self.repo_base_url, path.lstrip(u'/'))), |
| | 278 | u'my-prod-rep': Mock(reponame=u'my-prod-rep', |
| | 279 | scope=u'/devel/my-prod/', |
| | 280 | get_base=lambda: self.repo_base, |
| | 281 | get_path_url=lambda path, rev: |
| | 282 | u'svn+ssh:%s/devel/my-prod/%s' % |
| | 283 | (self.repo_base_url, path.lstrip(u'/'))), |
| | 284 | # Note 'my-dev-rep' purposefully has a scope that prefixes |
| | 285 | # 'my-devel-rep'. This tests cases of false matching scopes. |
| | 286 | u'my-dev-rep': Mock(reponame=u'my-dev-rep', scope=u'/dev/', |
| | 287 | get_base=lambda: self.repo_base, |
| | 288 | get_path_url=lambda path, rev: u'%s/dev/%s' % |
| | 289 | (self.repo_base_url, path.lstrip(u'/'))), |
| | 290 | # Extra repository, same base as above, without URL definition |
| | 291 | u'my-nourl-rep': Mock(reponame=u'my-nourl-rep', scope=u'/devel/', |
| | 292 | get_base=lambda: self.repo_base, |
| | 293 | get_path_url=lambda path, rev: None), |
| | 294 | u'otherrep': Mock(reponame=u'otherrep', scope=u'/', |
| | 295 | get_base=lambda: self.other_repo_base, |
| | 296 | get_path_url=lambda path, rev: u'svn:%s/%s' % |
| | 297 | (self.other_repo_base_url, path.lstrip(u'/'))), |
| | 298 | } |
| | 299 | self.rm = Mock(get_repository=lambda reponame: |
| | 300 | self.repositories[reponame]) |
| | 301 | |
| | 302 | def test_fully_qualified_and_scheme_relative_url(self): |
| | 303 | external = {u'reponame': 'myrep', |
| | 304 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 305 | u'dir': u'local', |
| | 306 | u'rev': None} |
| | 307 | req_scheme = random.choice([u'http', u'https']) |
| | 308 | req_host = u'local.server' |
| | 309 | req_href = u'%s://%s/trac/env/browser/myrep/devel/my-prod/' \ |
| | 310 | 'trunk/src/my-module' % (req_scheme, req_host) |
| | 311 | use_req_href = random.choice([req_href] * 7 + [None, u'', u'/']) |
| | 312 | |
| | 313 | ext_url = u'//remote.server.org/third/party/library' |
| | 314 | for scheme in [u'', u'http:', u'https:', u'svn:', u'svn+ssh:']: |
| | 315 | external[u'url'] = u'%s%s' % (scheme, ext_url) |
| | 316 | for reponame in self.repositories.keys(): |
| | 317 | expected_scheme = scheme |
| | 318 | # Twists: |
| | 319 | # - If URL is scheme-relative, and repository URL has scheme, |
| | 320 | # then take the scheme from the URL. |
| | 321 | # - If URL is scheme-relative, and repository URL does not |
| | 322 | # have scheme, then take the scheme from the request. |
| | 323 | if not scheme: |
| | 324 | repo_url = self.rm.get_repository(reponame) \ |
| | 325 | .get_path_url(u'/', None) or u'' |
| | 326 | if u'://' in repo_url: |
| | 327 | # Inherit scheme from Repository URL |
| | 328 | expected_scheme = u'%s:' % (splittype(repo_url)[0]) |
| | 329 | elif use_req_href and u'://' in use_req_href: |
| | 330 | # Inherit scheme from request |
| | 331 | expected_scheme = u'%s:' % (req_scheme) |
| | 332 | self.assertEqual(u'%s%s' % (expected_scheme, ext_url), |
| | 333 | svn_prop.get_external_url(external, reponame, |
| | 334 | rm=self.rm, req_href=use_req_href)) |
| | 335 | |
| | 336 | def test_server_relative_url(self): |
| | 337 | external = {u'reponame': 'myrep', |
| | 338 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 339 | u'dir': u'local', |
| | 340 | # External from myrep points to otherrep via server root! |
| | 341 | u'url': u'/svn/otherrep/dev', |
| | 342 | u'rev': None} |
| | 343 | req_scheme = random.choice([u'http', u'https']) |
| | 344 | req_host = u'local.server' |
| | 345 | req_href = u'%s://%s/trac/env/browser/myrep/devel/my-prod/' \ |
| | 346 | 'trunk/src/my-module' % (req_scheme, req_host) |
| | 347 | use_req_href = random.choice([req_href] * 7 + [None, u'', u'/']) |
| | 348 | use_req_href = req_href |
| | 349 | |
| | 350 | for reponame in self.repositories.keys(): |
| | 351 | repo_url = self.rm.get_repository(reponame) \ |
| | 352 | .get_path_url(u'/', None) or u'' |
| | 353 | expected_link = None |
| | 354 | if repo_url: |
| | 355 | expected_link = u'//%s%s' % (self.repo_host, external[u'url']) |
| | 356 | if u'://' in repo_url: |
| | 357 | expected_link = u'%s:%s' % (splittype(repo_url)[0], |
| | 358 | expected_link) |
| | 359 | elif (use_req_href or u'').rstrip(u'/'): |
| | 360 | expected_link = u'%s://%s%s' % (req_scheme, req_host, |
| | 361 | external[u'url']) |
| | 362 | if expected_link: |
| | 363 | self.assertEqual(expected_link, |
| | 364 | svn_prop.get_external_url(external, reponame, |
| | 365 | rm=self.rm, req_href=use_req_href)) |
| | 366 | else: |
| | 367 | self.assertTrue(svn_prop.get_external_url(external, reponame, |
| | 368 | rm=self.rm, req_href=use_req_href) is None) |
| | 369 | |
| | 370 | def test_repository_relative_url(self): |
| | 371 | external = {u'reponame': 'myrep', |
| | 372 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 373 | u'dir': u'local', |
| | 374 | u'url': u'^/devel/common/trunk/log', |
| | 375 | u'rev': None} |
| | 376 | scoped_external = {u'reponame': 'my-devel-rep', |
| | 377 | # Note the missing 'devel/' in the repo-path |
| | 378 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 379 | u'dir': u'local', |
| | 380 | u'url': u'^/devel/common/trunk/log', |
| | 381 | u'rev': None} |
| | 382 | deep_external = {u'reponame': 'my-devel-rep', |
| | 383 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 384 | u'repopath': u'trunk/src/my-module', |
| | 385 | u'dir': u'local', |
| | 386 | u'url': u'^/devel/common/trunk/log', |
| | 387 | u'rev': None} |
| | 388 | |
| | 389 | expected_link = u'%s/devel/common/trunk/log' % (self.repo_base_url) |
| | 390 | for ext in [external, scoped_external, deep_external]: |
| | 391 | self.assertEqual(u'https:%s' % (expected_link), |
| | 392 | svn_prop.get_external_url(ext, |
| | 393 | u'myrep', rm=self.rm)) |
| | 394 | self.assertEqual(u'http:%s' % (expected_link), |
| | 395 | svn_prop.get_external_url(ext, |
| | 396 | u'my-devel-rep', rm=self.rm)) |
| | 397 | self.assertEqual(u'svn+ssh:%s' % (expected_link), |
| | 398 | svn_prop.get_external_url(ext, |
| | 399 | u'my-prod-rep', rm=self.rm)) |
| | 400 | self.assertEqual(u'%s' % (expected_link), |
| | 401 | svn_prop.get_external_url(ext, |
| | 402 | u'my-dev-rep', rm=self.rm)) |
| | 403 | self.assertTrue(svn_prop.get_external_url(ext, u'my-nourl-rep', |
| | 404 | rm=self.rm) is None) |
| | 405 | self.assertTrue(svn_prop.get_external_url(ext, u'otherrep', |
| | 406 | rm=self.rm) is None) |
| | 407 | |
| | 408 | def test_directory_relative_url(self): |
| | 409 | external = {u'reponame': u'myrep', |
| | 410 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 411 | u'dir': u'local', |
| | 412 | u'url': u'../../../../common/trunk/log', |
| | 413 | u'rev': None} |
| | 414 | scoped_external = {u'reponame': u'my-devel-rep', |
| | 415 | # Note the missing 'devel/' in the repo-path |
| | 416 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 417 | u'dir': u'local', |
| | 418 | u'url': u'../../../../common/trunk/log', |
| | 419 | u'rev': None} |
| | 420 | deep_external = {u'reponame': u'my-prod-rep', |
| | 421 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 422 | u'repopath': u'trunk/src/my-module', |
| | 423 | u'dir': u'local', |
| | 424 | # Going even deeper just for fun (and test of course) |
| | 425 | u'url': u'../../../../../devel/common/trunk/log', |
| | 426 | u'rev': None} |
| | 427 | |
| | 428 | expected_link = u'%s/devel/common/trunk/log' % (self.repo_base_url) |
| | 429 | for ext in [external, scoped_external, deep_external]: |
| | 430 | self.assertEqual(u'https:%s' % (expected_link), |
| | 431 | svn_prop.get_external_url(ext, |
| | 432 | u'myrep', rm=self.rm)) |
| | 433 | self.assertEqual(u'http:%s' % (expected_link), |
| | 434 | svn_prop.get_external_url(ext, |
| | 435 | u'my-devel-rep', rm=self.rm)) |
| | 436 | self.assertEqual(u'svn+ssh:%s' % (expected_link), |
| | 437 | svn_prop.get_external_url(ext, |
| | 438 | u'my-prod-rep', rm=self.rm)) |
| | 439 | self.assertEqual(u'%s' % (expected_link), |
| | 440 | svn_prop.get_external_url(ext, |
| | 441 | u'my-dev-rep', rm=self.rm)) |
| | 442 | self.assertTrue(svn_prop.get_external_url(ext, u'my-nourl-rep', |
| | 443 | rm=self.rm) is None) |
| | 444 | self.assertTrue(svn_prop.get_external_url(ext, u'otherrep', |
| | 445 | rm=self.rm) is None) |
| | 446 | |
| | 447 | def test_invalid_directory_relative_url(self): |
| | 448 | external = {u'reponame': u'myrep', |
| | 449 | u'repopath': u'devel/my-prod/trunk/src/my-module', |
| | 450 | u'dir': u'local', |
| | 451 | # Out of repository base! aaah! |
| | 452 | u'url': u'../../../../../../common/trunk/log', |
| | 453 | u'rev': None} |
| | 454 | scoped_external = {u'reponame': u'my-devel-rep', |
| | 455 | # Note the missing 'devel/' in the repo-path |
| | 456 | u'repopath': u'my-prod/trunk/src/my-module', |
| | 457 | u'dir': u'local', |
| | 458 | u'url': u'../../../../../../common/trunk/log', |
| | 459 | u'rev': None} |
| | 460 | deep_external = {u'reponame': u'my-prod-rep', |
| | 461 | # Note the missing 'devel/my-prod/' in the repo-path |
| | 462 | u'repopath': u'trunk/src/my-module', |
| | 463 | u'dir': u'local', |
| | 464 | # Going even deeper just for fun (and test of course) |
| | 465 | u'url': u'../../../../../../../devel/common/trunk/log', |
| | 466 | u'rev': None} |
| | 467 | |
| | 468 | for ext in [external, scoped_external, deep_external]: |
| | 469 | self.assertRaises(TracError, svn_prop.get_external_url, |
| | 470 | ext, u'myrep', rm=self.rm) |
| | 471 | self.assertRaises(TracError, svn_prop.get_external_url, |
| | 472 | ext, u'my-devel-rep', rm=self.rm) |
| | 473 | self.assertRaises(TracError, svn_prop.get_external_url, |
| | 474 | ext, u'my-prod-rep', rm=self.rm) |
| | 475 | self.assertTrue(svn_prop.get_external_url(ext, u'otherrep', |
| | 476 | rm=self.rm) is None) |
| | 477 | |
| | 478 | class SvnExternalsMapTests(unittest.TestCase): |
| | 479 | |
| | 480 | def setUp(self): |
| | 481 | self.ini_section = { |
| | 482 | '1': u'//server/repos1 ' |
| | 483 | u'http://trac/proj/browser/Rep1/$path?rev=$rev', |
| | 484 | '2': u'svn://server/repos2/ ' |
| | 485 | u'http://trac/proj/browser/Rep2/$path?rev=$rev', |
| | 486 | '3': u'http://theirserver.org/svn/eng-soft ' |
| | 487 | u'http://ourserver/viewvc/svn/$path/?pathrev=25914', |
| | 488 | '4': u'svn://anotherserver.com/tools_repository ' |
| | 489 | u'//ourserver/tracs/tools/browser/$path?rev=$rev', |
| | 490 | # Testing existing partial match to 1 & 2 |
| | 491 | # and also Trac-env-relative pattern. |
| | 492 | # Note that need to specify only repository name (no "/browser") |
| | 493 | '5': u'//server/repo Repo/$path?rev=$rev', |
| | 494 | # But should work fine also with "/browser" specified |
| | 495 | # Also testing here for a better (more specific) match than 5 |
| | 496 | '6': u'//server/repo/sub-proj-a ' |
| | 497 | u'/browser/SubProjA/$path?rev=$rev' |
| | 498 | } |
| | 499 | self.map_from_ini = { |
| | 500 | u'//server/repos1': |
| | 501 | u'http://trac/proj/browser/Rep1/%(path)s?rev=%(rev)s', |
| | 502 | u'svn://server/repos2': |
| | 503 | u'http://trac/proj/browser/Rep2/%(path)s?rev=%(rev)s', |
| | 504 | u'http://theirserver.org/svn/eng-soft': |
| | 505 | u'http://ourserver/viewvc/svn/%(path)s/?pathrev=25914', |
| | 506 | u'svn://anotherserver.com/tools_repository': |
| | 507 | u'//ourserver/tracs/tools/browser/%(path)s?rev=%(rev)s', |
| | 508 | u'//server/repo': u'Repo/%(path)s?rev=%(rev)s', |
| | 509 | u'//server/repo/sub-proj-a': |
| | 510 | u'SubProjA/%(path)s?rev=%(rev)s' |
| | 511 | } |
| | 512 | self.log_messages = [] |
| | 513 | self.log = Mock(debug=lambda msg: self.log_messages.append(msg)) |
| | 514 | |
| | 515 | def test_parse_ini_externals_map(self): |
| | 516 | ext_map = {} |
| | 517 | svn_prop.parse_externals_map_ini(ext_map, |
| | 518 | self.ini_section.iteritems(), |
| | 519 | self.log) |
| | 520 | # No skips |
| | 521 | self.assertEqual([], self.log_messages) |
| | 522 | # Could be done better using Python 3.1 assertDictEqual... |
| | 523 | self.assertEqual(len(self.map_from_ini), len(ext_map)) |
| | 524 | for key, value in self.map_from_ini.iteritems(): |
| | 525 | self.assertTrue(key in ext_map) |
| | 526 | self.assertEqual(value, ext_map[key]) |
| | 527 | |
| | 528 | def test_null_pattern_link_expansion(self): |
| | 529 | self.assertTrue(svn_prop.format_external_link( |
| | 530 | ext_map={}, |
| | 531 | link=u'http://server/repos1', |
| | 532 | rev=None) is None) |
| | 533 | |
| | 534 | def test_fully_qualified_pattern_link_expansion(self): |
| | 535 | self.assertEqual(u'http://trac/proj/browser/Rep2/?rev=%(rev)s', |
| | 536 | svn_prop.format_external_link( |
| | 537 | ext_map=self.map_from_ini, |
| | 538 | link=u'svn://server/repos2')) |
| | 539 | self.assertEqual(u'http://ourserver/viewvc/svn/trunk/?pathrev=25914', |
| | 540 | svn_prop.format_external_link( |
| | 541 | ext_map=self.map_from_ini, |
| | 542 | link=u'http://theirserver.org/svn/eng-soft/trunk/')) |
| | 543 | self.assertEqual(u'//ourserver/tracs/tools/browser/branches/0.6-qa' |
| | 544 | '?rev=%(rev)s', |
| | 545 | svn_prop.format_external_link( |
| | 546 | ext_map=self.map_from_ini, |
| | 547 | link=u'svn://anotherserver.com/tools_repository/' |
| | 548 | 'branches/0.6-qa')) |
| | 549 | |
| | 550 | def test_trac_relative_link_expansions(self): |
| | 551 | self.assertEqual(u'SubProjA/trunk/dev?rev=%(rev)s', |
| | 552 | svn_prop.format_external_link( |
| | 553 | ext_map=self.map_from_ini, |
| | 554 | link=u'//server/repo/sub-proj-a/trunk/dev')) |
| | 555 | self.assertEqual(u'Repo/sub-proj-b/trunk?rev=%(rev)s', |
| | 556 | svn_prop.format_external_link( |
| | 557 | ext_map=self.map_from_ini, |
| | 558 | link=u'//server/repo/sub-proj-b/trunk')) |
| | 559 | |
| | 560 | def test_bad_partial_match(self): |
| | 561 | self.assertTrue(svn_prop.format_external_link( |
| | 562 | ext_map=self.map_from_ini, |
| | 563 | link=u'svn://server/rep', |
| | 564 | rev=None) is None) |
| | 565 | |
| | 566 | def test_bad_ini(self): |
| | 567 | bad_ini_sections = [ { '1': u'1-item-is-not-enough' }, |
| | 568 | { '1': u'3 items-is too-much' }, ] |
| | 569 | for section in bad_ini_sections: |
| | 570 | ext_map = {} |
| | 571 | self.log_messages = [] |
| | 572 | svn_prop.parse_externals_map_ini(ext_map, section.iteritems(), |
| | 573 | self.log) |
| | 574 | self.assertEqual(1, len(self.log_messages)) |
| | 575 | self.assertEqual(u'svn:externals entry 1 doesn\'t contain a ' |
| | 576 | 'space-separated key value pair, skipping.', |
| | 577 | self.log_messages[0]) |
| | 578 | |
| | 579 | def suite(): |
| | 580 | suite = unittest.TestSuite() |
| | 581 | suite.addTest(unittest.makeSuite(SvnExternalsParserTests)) |
| | 582 | suite.addTest(unittest.makeSuite(SvnExternalsRepositoryRelativeLinkTests)) |
| | 583 | suite.addTest(unittest.makeSuite(SvnExternalsUrlBasedLinkTests)) |
| | 584 | suite.addTest(unittest.makeSuite(SvnExternalsMapTests)) |
| | 585 | return suite |
| | 586 | |
| | 587 | if __name__ == '__main__': |
| | 588 | runner = unittest.TextTestRunner() |
| | 589 | runner.run(suite()) |