##// END OF EJS Templates
release: merge back stable branch into default
marcink -
r2553:09501d59 merge default
parent child Browse files
Show More
@@ -0,0 +1,47
1 |RCE| 4.11.2 |RNS|
2 ------------------
3
4 Release Date
5 ^^^^^^^^^^^^
6
7 - 2018-02-08
8
9
10 New Features
11 ^^^^^^^^^^^^
12
13
14
15 General
16 ^^^^^^^
17
18 - git: stabilize long running git processes to prevent potential worker freezes.
19
20
21 Security
22 ^^^^^^^^
23
24 - Repo Checks(low): prevent from html injection on repo-check page. It could
25 be prepared to fake the displayed text fooling users into unwanted actions.
26
27
28 Performance
29 ^^^^^^^^^^^
30
31
32
33 Fixes
34 ^^^^^
35
36 - Diffs: in case of text lexers don't do any HL because of pygments problems.
37 Fixed newline rendering for certain very unique diffs using Pygments.
38 - Webhook: fixed extra variable replacement in webhook url.
39 - Subprocessio: use safe b.read() and prevent potential valueErrors
40 - Subprocess: use subprocessio helper to run various subprocess commands.
41 - Git: use subprocessio for hooks execution to use non-blocking subprocess.
42
43
44 Upgrade notes
45 ^^^^^^^^^^^^^
46
47 - Scheduled bugfix release fixing certain requested problems with diffs and, Git.
@@ -1,32 +1,33
1 1bd3e92b7e2e2d2024152b34bb88dff1db544a71 v4.0.0
1 1bd3e92b7e2e2d2024152b34bb88dff1db544a71 v4.0.0
2 170c5398320ea6cddd50955e88d408794c21d43a v4.0.1
2 170c5398320ea6cddd50955e88d408794c21d43a v4.0.1
3 c3fe200198f5aa34cf2e4066df2881a9cefe3704 v4.1.0
3 c3fe200198f5aa34cf2e4066df2881a9cefe3704 v4.1.0
4 7fd5c850745e2ea821fb4406af5f4bff9b0a7526 v4.1.1
4 7fd5c850745e2ea821fb4406af5f4bff9b0a7526 v4.1.1
5 41c87da28a179953df86061d817bc35533c66dd2 v4.1.2
5 41c87da28a179953df86061d817bc35533c66dd2 v4.1.2
6 baaf9f5bcea3bae0ef12ae20c8b270482e62abb6 v4.2.0
6 baaf9f5bcea3bae0ef12ae20c8b270482e62abb6 v4.2.0
7 32a70c7e56844a825f61df496ee5eaf8c3c4e189 v4.2.1
7 32a70c7e56844a825f61df496ee5eaf8c3c4e189 v4.2.1
8 fa695cdb411d294679ac081d595ac654e5613b03 v4.3.0
8 fa695cdb411d294679ac081d595ac654e5613b03 v4.3.0
9 0e4dc11b58cad833c513fe17bac39e6850edf959 v4.3.1
9 0e4dc11b58cad833c513fe17bac39e6850edf959 v4.3.1
10 8a876f48f5cb1d018b837db28ff928500cb32cfb v4.4.0
10 8a876f48f5cb1d018b837db28ff928500cb32cfb v4.4.0
11 8dd86b410b1aac086ffdfc524ef300f896af5047 v4.4.1
11 8dd86b410b1aac086ffdfc524ef300f896af5047 v4.4.1
12 d2514226abc8d3b4f6fb57765f47d1b6fb360a05 v4.4.2
12 d2514226abc8d3b4f6fb57765f47d1b6fb360a05 v4.4.2
13 27d783325930af6dad2741476c0d0b1b7c8415c2 v4.5.0
13 27d783325930af6dad2741476c0d0b1b7c8415c2 v4.5.0
14 7f2016f352abcbdba4a19d4039c386e9629449da v4.5.1
14 7f2016f352abcbdba4a19d4039c386e9629449da v4.5.1
15 416fec799314c70a5c780fb28b3357b08869333a v4.5.2
15 416fec799314c70a5c780fb28b3357b08869333a v4.5.2
16 27c3b85fafc83143e6678fbc3da69e1615bcac55 v4.6.0
16 27c3b85fafc83143e6678fbc3da69e1615bcac55 v4.6.0
17 5ad13deb9118c2a5243d4032d4d9cc174e5872db v4.6.1
17 5ad13deb9118c2a5243d4032d4d9cc174e5872db v4.6.1
18 2be921e01fa24bb102696ada596f87464c3666f6 v4.7.0
18 2be921e01fa24bb102696ada596f87464c3666f6 v4.7.0
19 7198bdec29c2872c974431d55200d0398354cdb1 v4.7.1
19 7198bdec29c2872c974431d55200d0398354cdb1 v4.7.1
20 bd1c8d230fe741c2dfd7100a0ef39fd0774fd581 v4.7.2
20 bd1c8d230fe741c2dfd7100a0ef39fd0774fd581 v4.7.2
21 9731914f89765d9628dc4dddc84bc9402aa124c8 v4.8.0
21 9731914f89765d9628dc4dddc84bc9402aa124c8 v4.8.0
22 c5a2b7d0e4bbdebc4a62d7b624befe375207b659 v4.9.0
22 c5a2b7d0e4bbdebc4a62d7b624befe375207b659 v4.9.0
23 d9aa3b27ac9f7e78359775c75fedf7bfece232f1 v4.9.1
23 d9aa3b27ac9f7e78359775c75fedf7bfece232f1 v4.9.1
24 4ba4d74981cec5d6b28b158f875a2540952c2f74 v4.10.0
24 4ba4d74981cec5d6b28b158f875a2540952c2f74 v4.10.0
25 0a6821cbd6b0b3c21503002f88800679fa35ab63 v4.10.1
25 0a6821cbd6b0b3c21503002f88800679fa35ab63 v4.10.1
26 434ad90ec8d621f4416074b84f6e9ce03964defb v4.10.2
26 434ad90ec8d621f4416074b84f6e9ce03964defb v4.10.2
27 68baee10e698da2724c6e0f698c03a6abb993bf2 v4.10.3
27 68baee10e698da2724c6e0f698c03a6abb993bf2 v4.10.3
28 00821d3afd1dce3f4767cc353f84a17f7d5218a1 v4.10.4
28 00821d3afd1dce3f4767cc353f84a17f7d5218a1 v4.10.4
29 22f6744ad8cc274311825f63f953e4dee2ea5cb9 v4.10.5
29 22f6744ad8cc274311825f63f953e4dee2ea5cb9 v4.10.5
30 96eb24bea2f5f9258775245e3f09f6fa0a4dda01 v4.10.6
30 96eb24bea2f5f9258775245e3f09f6fa0a4dda01 v4.10.6
31 3121217a812c956d7dd5a5875821bd73e8002a32 v4.11.0
31 3121217a812c956d7dd5a5875821bd73e8002a32 v4.11.0
32 fa98b454715ac5b912f39e84af54345909a2a805 v4.11.1
32 fa98b454715ac5b912f39e84af54345909a2a805 v4.11.1
33 3982abcfdcc229a723cebe52d3a9bcff10bba08e v4.11.2
@@ -1,109 +1,110
1 .. _rhodecode-release-notes-ref:
1 .. _rhodecode-release-notes-ref:
2
2
3 Release Notes
3 Release Notes
4 =============
4 =============
5
5
6 |RCE| 4.x Versions
6 |RCE| 4.x Versions
7 ------------------
7 ------------------
8
8
9 .. toctree::
9 .. toctree::
10 :maxdepth: 1
10 :maxdepth: 1
11
11
12 release-notes-4.11.2.rst
12 release-notes-4.11.1.rst
13 release-notes-4.11.1.rst
13 release-notes-4.11.0.rst
14 release-notes-4.11.0.rst
14 release-notes-4.10.6.rst
15 release-notes-4.10.6.rst
15 release-notes-4.10.5.rst
16 release-notes-4.10.5.rst
16 release-notes-4.10.4.rst
17 release-notes-4.10.4.rst
17 release-notes-4.10.3.rst
18 release-notes-4.10.3.rst
18 release-notes-4.10.2.rst
19 release-notes-4.10.2.rst
19 release-notes-4.10.1.rst
20 release-notes-4.10.1.rst
20 release-notes-4.10.0.rst
21 release-notes-4.10.0.rst
21 release-notes-4.9.1.rst
22 release-notes-4.9.1.rst
22 release-notes-4.9.0.rst
23 release-notes-4.9.0.rst
23 release-notes-4.8.0.rst
24 release-notes-4.8.0.rst
24 release-notes-4.7.2.rst
25 release-notes-4.7.2.rst
25 release-notes-4.7.1.rst
26 release-notes-4.7.1.rst
26 release-notes-4.7.0.rst
27 release-notes-4.7.0.rst
27 release-notes-4.6.1.rst
28 release-notes-4.6.1.rst
28 release-notes-4.6.0.rst
29 release-notes-4.6.0.rst
29 release-notes-4.5.2.rst
30 release-notes-4.5.2.rst
30 release-notes-4.5.1.rst
31 release-notes-4.5.1.rst
31 release-notes-4.5.0.rst
32 release-notes-4.5.0.rst
32 release-notes-4.4.2.rst
33 release-notes-4.4.2.rst
33 release-notes-4.4.1.rst
34 release-notes-4.4.1.rst
34 release-notes-4.4.0.rst
35 release-notes-4.4.0.rst
35 release-notes-4.3.1.rst
36 release-notes-4.3.1.rst
36 release-notes-4.3.0.rst
37 release-notes-4.3.0.rst
37 release-notes-4.2.1.rst
38 release-notes-4.2.1.rst
38 release-notes-4.2.0.rst
39 release-notes-4.2.0.rst
39 release-notes-4.1.2.rst
40 release-notes-4.1.2.rst
40 release-notes-4.1.1.rst
41 release-notes-4.1.1.rst
41 release-notes-4.1.0.rst
42 release-notes-4.1.0.rst
42 release-notes-4.0.1.rst
43 release-notes-4.0.1.rst
43 release-notes-4.0.0.rst
44 release-notes-4.0.0.rst
44
45
45 |RCE| 3.x Versions
46 |RCE| 3.x Versions
46 ------------------
47 ------------------
47
48
48 .. toctree::
49 .. toctree::
49 :maxdepth: 1
50 :maxdepth: 1
50
51
51 release-notes-3.8.4.rst
52 release-notes-3.8.4.rst
52 release-notes-3.8.3.rst
53 release-notes-3.8.3.rst
53 release-notes-3.8.2.rst
54 release-notes-3.8.2.rst
54 release-notes-3.8.1.rst
55 release-notes-3.8.1.rst
55 release-notes-3.8.0.rst
56 release-notes-3.8.0.rst
56 release-notes-3.7.1.rst
57 release-notes-3.7.1.rst
57 release-notes-3.7.0.rst
58 release-notes-3.7.0.rst
58 release-notes-3.6.1.rst
59 release-notes-3.6.1.rst
59 release-notes-3.6.0.rst
60 release-notes-3.6.0.rst
60 release-notes-3.5.2.rst
61 release-notes-3.5.2.rst
61 release-notes-3.5.1.rst
62 release-notes-3.5.1.rst
62 release-notes-3.5.0.rst
63 release-notes-3.5.0.rst
63 release-notes-3.4.1.rst
64 release-notes-3.4.1.rst
64 release-notes-3.4.0.rst
65 release-notes-3.4.0.rst
65 release-notes-3.3.4.rst
66 release-notes-3.3.4.rst
66 release-notes-3.3.3.rst
67 release-notes-3.3.3.rst
67 release-notes-3.3.2.rst
68 release-notes-3.3.2.rst
68 release-notes-3.3.1.rst
69 release-notes-3.3.1.rst
69 release-notes-3.3.0.rst
70 release-notes-3.3.0.rst
70 release-notes-3.2.3.rst
71 release-notes-3.2.3.rst
71 release-notes-3.2.2.rst
72 release-notes-3.2.2.rst
72 release-notes-3.2.1.rst
73 release-notes-3.2.1.rst
73 release-notes-3.2.0.rst
74 release-notes-3.2.0.rst
74 release-notes-3.1.1.rst
75 release-notes-3.1.1.rst
75 release-notes-3.1.0.rst
76 release-notes-3.1.0.rst
76 release-notes-3.0.2.rst
77 release-notes-3.0.2.rst
77 release-notes-3.0.1.rst
78 release-notes-3.0.1.rst
78 release-notes-3.0.0.rst
79 release-notes-3.0.0.rst
79
80
80 |RCE| 2.x Versions
81 |RCE| 2.x Versions
81 ------------------
82 ------------------
82
83
83 .. toctree::
84 .. toctree::
84 :maxdepth: 1
85 :maxdepth: 1
85
86
86 release-notes-2.2.8.rst
87 release-notes-2.2.8.rst
87 release-notes-2.2.7.rst
88 release-notes-2.2.7.rst
88 release-notes-2.2.6.rst
89 release-notes-2.2.6.rst
89 release-notes-2.2.5.rst
90 release-notes-2.2.5.rst
90 release-notes-2.2.4.rst
91 release-notes-2.2.4.rst
91 release-notes-2.2.3.rst
92 release-notes-2.2.3.rst
92 release-notes-2.2.2.rst
93 release-notes-2.2.2.rst
93 release-notes-2.2.1.rst
94 release-notes-2.2.1.rst
94 release-notes-2.2.0.rst
95 release-notes-2.2.0.rst
95 release-notes-2.1.0.rst
96 release-notes-2.1.0.rst
96 release-notes-2.0.2.rst
97 release-notes-2.0.2.rst
97 release-notes-2.0.1.rst
98 release-notes-2.0.1.rst
98 release-notes-2.0.0.rst
99 release-notes-2.0.0.rst
99
100
100 |RCE| 1.x Versions
101 |RCE| 1.x Versions
101 ------------------
102 ------------------
102
103
103 .. toctree::
104 .. toctree::
104 :maxdepth: 1
105 :maxdepth: 1
105
106
106 release-notes-1.7.2.rst
107 release-notes-1.7.2.rst
107 release-notes-1.7.1.rst
108 release-notes-1.7.1.rst
108 release-notes-1.7.0.rst
109 release-notes-1.7.0.rst
109 release-notes-1.6.0.rst
110 release-notes-1.6.0.rst
@@ -1,113 +1,114
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2011-2018 RhodeCode GmbH
3 # Copyright (C) 2011-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21 import logging
21 import logging
22
22
23 from pyramid.view import view_config
23 from pyramid.view import view_config
24 from pyramid.httpexceptions import HTTPFound, HTTPNotFound
24 from pyramid.httpexceptions import HTTPFound, HTTPNotFound
25
25
26 from rhodecode.apps._base import BaseAppView
26 from rhodecode.apps._base import BaseAppView
27 from rhodecode.lib import helpers as h
27 from rhodecode.lib import helpers as h
28 from rhodecode.lib.auth import (NotAnonymous, HasRepoPermissionAny)
28 from rhodecode.lib.auth import (NotAnonymous, HasRepoPermissionAny)
29 from rhodecode.model.db import Repository
29 from rhodecode.model.db import Repository
30 from rhodecode.model.validation_schema.types import RepoNameType
30
31
31 log = logging.getLogger(__name__)
32 log = logging.getLogger(__name__)
32
33
33
34
34 class RepoChecksView(BaseAppView):
35 class RepoChecksView(BaseAppView):
35 def load_default_context(self):
36 def load_default_context(self):
36 c = self._get_local_tmpl_context()
37 c = self._get_local_tmpl_context()
37
38
38 return c
39 return c
39
40
40 @NotAnonymous()
41 @NotAnonymous()
41 @view_config(
42 @view_config(
42 route_name='repo_creating', request_method='GET',
43 route_name='repo_creating', request_method='GET',
43 renderer='rhodecode:templates/admin/repos/repo_creating.mako')
44 renderer='rhodecode:templates/admin/repos/repo_creating.mako')
44 def repo_creating(self):
45 def repo_creating(self):
45 c = self.load_default_context()
46 c = self.load_default_context()
46
47 repo_name = self.request.matchdict['repo_name']
47 repo_name = self.request.matchdict['repo_name']
48 repo_name = RepoNameType().deserialize(None, repo_name)
48 db_repo = Repository.get_by_repo_name(repo_name)
49 db_repo = Repository.get_by_repo_name(repo_name)
49
50
50 # check if maybe repo is already created
51 # check if maybe repo is already created
51 if db_repo and db_repo.repo_state in [Repository.STATE_CREATED]:
52 if db_repo and db_repo.repo_state in [Repository.STATE_CREATED]:
52 # re-check permissions before redirecting to prevent resource
53 # re-check permissions before redirecting to prevent resource
53 # discovery by checking the 302 code
54 # discovery by checking the 302 code
54 perm_set = ['repository.read', 'repository.write', 'repository.admin']
55 perm_set = ['repository.read', 'repository.write', 'repository.admin']
55 has_perm = HasRepoPermissionAny(*perm_set)(
56 has_perm = HasRepoPermissionAny(*perm_set)(
56 db_repo.repo_name, 'Repo Creating check')
57 db_repo.repo_name, 'Repo Creating check')
57 if not has_perm:
58 if not has_perm:
58 raise HTTPNotFound()
59 raise HTTPNotFound()
59
60
60 raise HTTPFound(h.route_path(
61 raise HTTPFound(h.route_path(
61 'repo_summary', repo_name=db_repo.repo_name))
62 'repo_summary', repo_name=db_repo.repo_name))
62
63
63 c.task_id = self.request.GET.get('task_id')
64 c.task_id = self.request.GET.get('task_id')
64 c.repo_name = repo_name
65 c.repo_name = repo_name
65
66
66 return self._get_template_context(c)
67 return self._get_template_context(c)
67
68
68 @NotAnonymous()
69 @NotAnonymous()
69 @view_config(
70 @view_config(
70 route_name='repo_creating_check', request_method='GET',
71 route_name='repo_creating_check', request_method='GET',
71 renderer='json_ext')
72 renderer='json_ext')
72 def repo_creating_check(self):
73 def repo_creating_check(self):
73 _ = self.request.translate
74 _ = self.request.translate
74 task_id = self.request.GET.get('task_id')
75 task_id = self.request.GET.get('task_id')
75 self.load_default_context()
76 self.load_default_context()
76
77
77 repo_name = self.request.matchdict['repo_name']
78 repo_name = self.request.matchdict['repo_name']
78
79
79 if task_id and task_id not in ['None']:
80 if task_id and task_id not in ['None']:
80 import rhodecode
81 import rhodecode
81 from rhodecode.lib.celerylib.loader import celery_app, exceptions
82 from rhodecode.lib.celerylib.loader import celery_app, exceptions
82 if rhodecode.CELERY_ENABLED:
83 if rhodecode.CELERY_ENABLED:
83 log.debug('celery: checking result for task:%s', task_id)
84 log.debug('celery: checking result for task:%s', task_id)
84 task = celery_app.AsyncResult(task_id)
85 task = celery_app.AsyncResult(task_id)
85 try:
86 try:
86 task.get(timeout=10)
87 task.get(timeout=10)
87 except exceptions.TimeoutError:
88 except exceptions.TimeoutError:
88 task = None
89 task = None
89 if task and task.failed():
90 if task and task.failed():
90 msg = self._log_creation_exception(task.result, repo_name)
91 msg = self._log_creation_exception(task.result, repo_name)
91 h.flash(msg, category='error')
92 h.flash(msg, category='error')
92 raise HTTPFound(h.route_path('home'), code=501)
93 raise HTTPFound(h.route_path('home'), code=501)
93
94
94 db_repo = Repository.get_by_repo_name(repo_name)
95 db_repo = Repository.get_by_repo_name(repo_name)
95 if db_repo and db_repo.repo_state == Repository.STATE_CREATED:
96 if db_repo and db_repo.repo_state == Repository.STATE_CREATED:
96 if db_repo.clone_uri:
97 if db_repo.clone_uri:
97 clone_uri = db_repo.clone_uri_hidden
98 clone_uri = db_repo.clone_uri_hidden
98 h.flash(_('Created repository %s from %s')
99 h.flash(_('Created repository %s from %s')
99 % (db_repo.repo_name, clone_uri), category='success')
100 % (db_repo.repo_name, clone_uri), category='success')
100 else:
101 else:
101 repo_url = h.link_to(
102 repo_url = h.link_to(
102 db_repo.repo_name,
103 db_repo.repo_name,
103 h.route_path('repo_summary', repo_name=db_repo.repo_name))
104 h.route_path('repo_summary', repo_name=db_repo.repo_name))
104 fork = db_repo.fork
105 fork = db_repo.fork
105 if fork:
106 if fork:
106 fork_name = fork.repo_name
107 fork_name = fork.repo_name
107 h.flash(h.literal(_('Forked repository %s as %s')
108 h.flash(h.literal(_('Forked repository %s as %s')
108 % (fork_name, repo_url)), category='success')
109 % (fork_name, repo_url)), category='success')
109 else:
110 else:
110 h.flash(h.literal(_('Created repository %s') % repo_url),
111 h.flash(h.literal(_('Created repository %s') % repo_url),
111 category='success')
112 category='success')
112 return {'result': True}
113 return {'result': True}
113 return {'result': False}
114 return {'result': False}
@@ -1,394 +1,395
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2012-2018 RhodeCode GmbH
3 # Copyright (C) 2012-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21 from __future__ import unicode_literals
21 from __future__ import unicode_literals
22 import string
22 import string
23 from collections import OrderedDict
23 from collections import OrderedDict
24
24
25 import deform
25 import deform
26 import deform.widget
26 import deform.widget
27 import logging
27 import logging
28 import requests
28 import requests
29 import requests.adapters
29 import requests.adapters
30 import colander
30 import colander
31 from requests.packages.urllib3.util.retry import Retry
31 from requests.packages.urllib3.util.retry import Retry
32
32
33 import rhodecode
33 import rhodecode
34 from rhodecode import events
34 from rhodecode import events
35 from rhodecode.translation import _
35 from rhodecode.translation import _
36 from rhodecode.integrations.types.base import IntegrationTypeBase
36 from rhodecode.integrations.types.base import IntegrationTypeBase
37 from rhodecode.lib.celerylib import run_task, async_task, RequestContextTask
37 from rhodecode.lib.celerylib import run_task, async_task, RequestContextTask
38
38
39 log = logging.getLogger(__name__)
39 log = logging.getLogger(__name__)
40
40
41
41
42 # updating this required to update the `common_vars` passed in url calling func
42 # updating this required to update the `common_vars` passed in url calling func
43 WEBHOOK_URL_VARS = [
43 WEBHOOK_URL_VARS = [
44 'repo_name',
44 'repo_name',
45 'repo_type',
45 'repo_type',
46 'repo_id',
46 'repo_id',
47 'repo_url',
47 'repo_url',
48 # extra repo fields
48 # extra repo fields
49 'extra:<extra_key_name>',
49 'extra:<extra_key_name>',
50
50
51 # special attrs below that we handle, using multi-call
51 # special attrs below that we handle, using multi-call
52 'branch',
52 'branch',
53 'commit_id',
53 'commit_id',
54
54
55 # pr events vars
55 # pr events vars
56 'pull_request_id',
56 'pull_request_id',
57 'pull_request_url',
57 'pull_request_url',
58
58
59 # user who triggers the call
59 # user who triggers the call
60 'username',
60 'username',
61 'user_id',
61 'user_id',
62
62
63 ]
63 ]
64 URL_VARS = ', '.join('${' + x + '}' for x in WEBHOOK_URL_VARS)
64 URL_VARS = ', '.join('${' + x + '}' for x in WEBHOOK_URL_VARS)
65
65
66
66
67 def get_auth(settings):
67 def get_auth(settings):
68 from requests.auth import HTTPBasicAuth
68 from requests.auth import HTTPBasicAuth
69 username = settings.get('username')
69 username = settings.get('username')
70 password = settings.get('password')
70 password = settings.get('password')
71 if username and password:
71 if username and password:
72 return HTTPBasicAuth(username, password)
72 return HTTPBasicAuth(username, password)
73 return None
73 return None
74
74
75
75
76 class WebhookHandler(object):
76 class WebhookHandler(object):
77 def __init__(self, template_url, secret_token, headers):
77 def __init__(self, template_url, secret_token, headers):
78 self.template_url = template_url
78 self.template_url = template_url
79 self.secret_token = secret_token
79 self.secret_token = secret_token
80 self.headers = headers
80 self.headers = headers
81
81
82 def get_base_parsed_template(self, data):
82 def get_base_parsed_template(self, data):
83 """
83 """
84 initially parses the passed in template with some common variables
84 initially parses the passed in template with some common variables
85 available on ALL calls
85 available on ALL calls
86 """
86 """
87 # note: make sure to update the `WEBHOOK_URL_VARS` if this changes
87 # note: make sure to update the `WEBHOOK_URL_VARS` if this changes
88 common_vars = {
88 common_vars = {
89 'repo_name': data['repo']['repo_name'],
89 'repo_name': data['repo']['repo_name'],
90 'repo_type': data['repo']['repo_type'],
90 'repo_type': data['repo']['repo_type'],
91 'repo_id': data['repo']['repo_id'],
91 'repo_id': data['repo']['repo_id'],
92 'repo_url': data['repo']['url'],
92 'repo_url': data['repo']['url'],
93 'username': data['actor']['username'],
93 'username': data['actor']['username'],
94 'user_id': data['actor']['user_id']
94 'user_id': data['actor']['user_id']
95 }
95 }
96
96 extra_vars = {}
97 extra_vars = {}
97 for extra_key, extra_val in data['repo']['extra_fields'].items():
98 for extra_key, extra_val in data['repo']['extra_fields'].items():
98 extra_vars['extra:{}'.format(extra_key)] = extra_val
99 extra_vars['extra__{}'.format(extra_key)] = extra_val
99 common_vars.update(extra_vars)
100 common_vars.update(extra_vars)
100
101
101 return string.Template(
102 template_url = self.template_url.replace('${extra:', '${extra__')
102 self.template_url).safe_substitute(**common_vars)
103 return string.Template(template_url).safe_substitute(**common_vars)
103
104
104 def repo_push_event_handler(self, event, data):
105 def repo_push_event_handler(self, event, data):
105 url = self.get_base_parsed_template(data)
106 url = self.get_base_parsed_template(data)
106 url_cals = []
107 url_cals = []
107 branch_data = OrderedDict()
108 branch_data = OrderedDict()
108 for obj in data['push']['branches']:
109 for obj in data['push']['branches']:
109 branch_data[obj['name']] = obj
110 branch_data[obj['name']] = obj
110
111
111 branches_commits = OrderedDict()
112 branches_commits = OrderedDict()
112 for commit in data['push']['commits']:
113 for commit in data['push']['commits']:
113 if commit.get('git_ref_change'):
114 if commit.get('git_ref_change'):
114 # special case for GIT that allows creating tags,
115 # special case for GIT that allows creating tags,
115 # deleting branches without associated commit
116 # deleting branches without associated commit
116 continue
117 continue
117
118
118 if commit['branch'] not in branches_commits:
119 if commit['branch'] not in branches_commits:
119 branch_commits = {'branch': branch_data[commit['branch']],
120 branch_commits = {'branch': branch_data[commit['branch']],
120 'commits': []}
121 'commits': []}
121 branches_commits[commit['branch']] = branch_commits
122 branches_commits[commit['branch']] = branch_commits
122
123
123 branch_commits = branches_commits[commit['branch']]
124 branch_commits = branches_commits[commit['branch']]
124 branch_commits['commits'].append(commit)
125 branch_commits['commits'].append(commit)
125
126
126 if '${branch}' in url:
127 if '${branch}' in url:
127 # call it multiple times, for each branch if used in variables
128 # call it multiple times, for each branch if used in variables
128 for branch, commit_ids in branches_commits.items():
129 for branch, commit_ids in branches_commits.items():
129 branch_url = string.Template(url).safe_substitute(branch=branch)
130 branch_url = string.Template(url).safe_substitute(branch=branch)
130 # call further down for each commit if used
131 # call further down for each commit if used
131 if '${commit_id}' in branch_url:
132 if '${commit_id}' in branch_url:
132 for commit_data in commit_ids['commits']:
133 for commit_data in commit_ids['commits']:
133 commit_id = commit_data['raw_id']
134 commit_id = commit_data['raw_id']
134 commit_url = string.Template(branch_url).safe_substitute(
135 commit_url = string.Template(branch_url).safe_substitute(
135 commit_id=commit_id)
136 commit_id=commit_id)
136 # register per-commit call
137 # register per-commit call
137 log.debug(
138 log.debug(
138 'register webhook call(%s) to url %s', event, commit_url)
139 'register webhook call(%s) to url %s', event, commit_url)
139 url_cals.append((commit_url, self.secret_token, self.headers, data))
140 url_cals.append((commit_url, self.secret_token, self.headers, data))
140
141
141 else:
142 else:
142 # register per-branch call
143 # register per-branch call
143 log.debug(
144 log.debug(
144 'register webhook call(%s) to url %s', event, branch_url)
145 'register webhook call(%s) to url %s', event, branch_url)
145 url_cals.append((branch_url, self.secret_token, self.headers, data))
146 url_cals.append((branch_url, self.secret_token, self.headers, data))
146
147
147 else:
148 else:
148 log.debug(
149 log.debug(
149 'register webhook call(%s) to url %s', event, url)
150 'register webhook call(%s) to url %s', event, url)
150 url_cals.append((url, self.secret_token, self.headers, data))
151 url_cals.append((url, self.secret_token, self.headers, data))
151
152
152 return url_cals
153 return url_cals
153
154
154 def repo_create_event_handler(self, event, data):
155 def repo_create_event_handler(self, event, data):
155 url = self.get_base_parsed_template(data)
156 url = self.get_base_parsed_template(data)
156 log.debug(
157 log.debug(
157 'register webhook call(%s) to url %s', event, url)
158 'register webhook call(%s) to url %s', event, url)
158 return [(url, self.secret_token, self.headers, data)]
159 return [(url, self.secret_token, self.headers, data)]
159
160
160 def pull_request_event_handler(self, event, data):
161 def pull_request_event_handler(self, event, data):
161 url = self.get_base_parsed_template(data)
162 url = self.get_base_parsed_template(data)
162 log.debug(
163 log.debug(
163 'register webhook call(%s) to url %s', event, url)
164 'register webhook call(%s) to url %s', event, url)
164 url = string.Template(url).safe_substitute(
165 url = string.Template(url).safe_substitute(
165 pull_request_id=data['pullrequest']['pull_request_id'],
166 pull_request_id=data['pullrequest']['pull_request_id'],
166 pull_request_url=data['pullrequest']['url'])
167 pull_request_url=data['pullrequest']['url'])
167 return [(url, self.secret_token, self.headers, data)]
168 return [(url, self.secret_token, self.headers, data)]
168
169
169 def __call__(self, event, data):
170 def __call__(self, event, data):
170 if isinstance(event, events.RepoPushEvent):
171 if isinstance(event, events.RepoPushEvent):
171 return self.repo_push_event_handler(event, data)
172 return self.repo_push_event_handler(event, data)
172 elif isinstance(event, events.RepoCreateEvent):
173 elif isinstance(event, events.RepoCreateEvent):
173 return self.repo_create_event_handler(event, data)
174 return self.repo_create_event_handler(event, data)
174 elif isinstance(event, events.PullRequestEvent):
175 elif isinstance(event, events.PullRequestEvent):
175 return self.pull_request_event_handler(event, data)
176 return self.pull_request_event_handler(event, data)
176 else:
177 else:
177 raise ValueError('event type not supported: %s' % events)
178 raise ValueError('event type not supported: %s' % events)
178
179
179
180
180 class WebhookSettingsSchema(colander.Schema):
181 class WebhookSettingsSchema(colander.Schema):
181 url = colander.SchemaNode(
182 url = colander.SchemaNode(
182 colander.String(),
183 colander.String(),
183 title=_('Webhook URL'),
184 title=_('Webhook URL'),
184 description=
185 description=
185 _('URL to which Webhook should submit data. Following variables '
186 _('URL to which Webhook should submit data. Following variables '
186 'are allowed to be used: {vars}. Some of the variables would '
187 'are allowed to be used: {vars}. Some of the variables would '
187 'trigger multiple calls, like ${{branch}} or ${{commit_id}}. '
188 'trigger multiple calls, like ${{branch}} or ${{commit_id}}. '
188 'Webhook will be called as many times as unique objects in '
189 'Webhook will be called as many times as unique objects in '
189 'data in such cases.').format(vars=URL_VARS),
190 'data in such cases.').format(vars=URL_VARS),
190 missing=colander.required,
191 missing=colander.required,
191 required=True,
192 required=True,
192 validator=colander.url,
193 validator=colander.url,
193 widget=deform.widget.TextInputWidget(
194 widget=deform.widget.TextInputWidget(
194 placeholder='https://www.example.com/webhook'
195 placeholder='https://www.example.com/webhook'
195 ),
196 ),
196 )
197 )
197 secret_token = colander.SchemaNode(
198 secret_token = colander.SchemaNode(
198 colander.String(),
199 colander.String(),
199 title=_('Secret Token'),
200 title=_('Secret Token'),
200 description=_('Optional string used to validate received payloads. '
201 description=_('Optional string used to validate received payloads. '
201 'It will be sent together with event data in JSON'),
202 'It will be sent together with event data in JSON'),
202 default='',
203 default='',
203 missing='',
204 missing='',
204 widget=deform.widget.TextInputWidget(
205 widget=deform.widget.TextInputWidget(
205 placeholder='e.g. secret_token'
206 placeholder='e.g. secret_token'
206 ),
207 ),
207 )
208 )
208 username = colander.SchemaNode(
209 username = colander.SchemaNode(
209 colander.String(),
210 colander.String(),
210 title=_('Username'),
211 title=_('Username'),
211 description=_('Optional username to authenticate the call.'),
212 description=_('Optional username to authenticate the call.'),
212 default='',
213 default='',
213 missing='',
214 missing='',
214 widget=deform.widget.TextInputWidget(
215 widget=deform.widget.TextInputWidget(
215 placeholder='e.g. admin'
216 placeholder='e.g. admin'
216 ),
217 ),
217 )
218 )
218 password = colander.SchemaNode(
219 password = colander.SchemaNode(
219 colander.String(),
220 colander.String(),
220 title=_('Password'),
221 title=_('Password'),
221 description=_('Optional password to authenticate the call.'),
222 description=_('Optional password to authenticate the call.'),
222 default='',
223 default='',
223 missing='',
224 missing='',
224 widget=deform.widget.PasswordWidget(
225 widget=deform.widget.PasswordWidget(
225 placeholder='e.g. secret.',
226 placeholder='e.g. secret.',
226 redisplay=True,
227 redisplay=True,
227 ),
228 ),
228 )
229 )
229 custom_header_key = colander.SchemaNode(
230 custom_header_key = colander.SchemaNode(
230 colander.String(),
231 colander.String(),
231 title=_('Custom Header Key'),
232 title=_('Custom Header Key'),
232 description=_('Custom Header name to be set when calling endpoint.'),
233 description=_('Custom Header name to be set when calling endpoint.'),
233 default='',
234 default='',
234 missing='',
235 missing='',
235 widget=deform.widget.TextInputWidget(
236 widget=deform.widget.TextInputWidget(
236 placeholder='e.g.Authorization'
237 placeholder='e.g.Authorization'
237 ),
238 ),
238 )
239 )
239 custom_header_val = colander.SchemaNode(
240 custom_header_val = colander.SchemaNode(
240 colander.String(),
241 colander.String(),
241 title=_('Custom Header Value'),
242 title=_('Custom Header Value'),
242 description=_('Custom Header value to be set when calling endpoint.'),
243 description=_('Custom Header value to be set when calling endpoint.'),
243 default='',
244 default='',
244 missing='',
245 missing='',
245 widget=deform.widget.TextInputWidget(
246 widget=deform.widget.TextInputWidget(
246 placeholder='e.g. RcLogin auth=xxxx'
247 placeholder='e.g. RcLogin auth=xxxx'
247 ),
248 ),
248 )
249 )
249 method_type = colander.SchemaNode(
250 method_type = colander.SchemaNode(
250 colander.String(),
251 colander.String(),
251 title=_('Call Method'),
252 title=_('Call Method'),
252 description=_('Select if the Webhook call should be made '
253 description=_('Select if the Webhook call should be made '
253 'with POST or GET.'),
254 'with POST or GET.'),
254 default='post',
255 default='post',
255 missing='',
256 missing='',
256 widget=deform.widget.RadioChoiceWidget(
257 widget=deform.widget.RadioChoiceWidget(
257 values=[('get', 'GET'), ('post', 'POST')],
258 values=[('get', 'GET'), ('post', 'POST')],
258 inline=True
259 inline=True
259 ),
260 ),
260 )
261 )
261
262
262
263
263 class WebhookIntegrationType(IntegrationTypeBase):
264 class WebhookIntegrationType(IntegrationTypeBase):
264 key = 'webhook'
265 key = 'webhook'
265 display_name = _('Webhook')
266 display_name = _('Webhook')
266 description = _('Post json events to a Webhook endpoint')
267 description = _('Post json events to a Webhook endpoint')
267 icon = '''<?xml version="1.0" encoding="UTF-8" standalone="no"?><svg viewBox="0 0 256 239" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" preserveAspectRatio="xMidYMid"><g><path d="M119.540432,100.502743 C108.930124,118.338815 98.7646301,135.611455 88.3876025,152.753617 C85.7226696,157.154315 84.4040417,160.738531 86.5332204,166.333309 C92.4107024,181.787152 84.1193605,196.825836 68.5350381,200.908244 C53.8383677,204.759349 39.5192953,195.099955 36.6032893,179.365384 C34.0194114,165.437749 44.8274148,151.78491 60.1824106,149.608284 C61.4694072,149.424428 62.7821041,149.402681 64.944891,149.240571 C72.469175,136.623655 80.1773157,123.700312 88.3025935,110.073173 C73.611854,95.4654658 64.8677898,78.3885437 66.803227,57.2292132 C68.1712787,42.2715849 74.0527146,29.3462646 84.8033863,18.7517722 C105.393354,-1.53572199 136.805164,-4.82141828 161.048542,10.7510424 C184.333097,25.7086706 194.996783,54.8450075 185.906752,79.7822957 C179.052655,77.9239597 172.151111,76.049808 164.563565,73.9917997 C167.418285,60.1274266 165.306899,47.6765751 155.95591,37.0109123 C149.777932,29.9690049 141.850349,26.2780332 132.835442,24.9178894 C114.764113,22.1877169 97.0209573,33.7983633 91.7563309,51.5355878 C85.7800012,71.6669027 94.8245623,88.1111998 119.540432,100.502743 L119.540432,100.502743 Z" fill="#C73A63"></path><path d="M149.841194,79.4106285 C157.316054,92.5969067 164.905578,105.982857 172.427885,119.246236 C210.44865,107.483365 239.114472,128.530009 249.398582,151.063322 C261.81978,178.282014 253.328765,210.520191 228.933162,227.312431 C203.893073,244.551464 172.226236,241.605803 150.040866,219.46195 C155.694953,214.729124 161.376716,209.974552 167.44794,204.895759 C189.360489,219.088306 208.525074,218.420096 222.753207,201.614016 C234.885769,187.277151 234.622834,165.900356 222.138374,151.863988 C207.730339,135.66681 188.431321,135.172572 165.103273,150.721309 C155.426087,133.553447 145.58086,116.521995 136.210101,99.2295848 C133.05093,93.4015266 129.561608,90.0209366 122.440622,88.7873178 C110.547271,86.7253555 102.868785,76.5124151 102.408155,65.0698097 C101.955433,53.7537294 108.621719,43.5249733 119.04224,39.5394355 C129.363912,35.5914599 141.476705,38.7783085 148.419765,47.554004 C154.093621,54.7244134 155.896602,62.7943365 152.911402,71.6372484 C152.081082,74.1025091 151.00562,76.4886916 149.841194,79.4106285 L149.841194,79.4106285 Z" fill="#4B4B4B"></path><path d="M167.706921,187.209935 L121.936499,187.209935 C117.54964,205.253587 108.074103,219.821756 91.7464461,229.085759 C79.0544063,236.285822 65.3738898,238.72736 50.8136292,236.376762 C24.0061432,232.053165 2.08568567,207.920497 0.156179306,180.745298 C-2.02835403,149.962159 19.1309765,122.599149 47.3341915,116.452801 C49.2814904,123.524363 51.2485589,130.663141 53.1958579,137.716911 C27.3195169,150.919004 18.3639187,167.553089 25.6054984,188.352614 C31.9811726,206.657224 50.0900643,216.690262 69.7528413,212.809503 C89.8327554,208.847688 99.9567329,192.160226 98.7211371,165.37844 C117.75722,165.37844 136.809118,165.180745 155.847178,165.475311 C163.280522,165.591951 169.019617,164.820939 174.620326,158.267339 C183.840836,147.48306 200.811003,148.455721 210.741239,158.640984 C220.88894,169.049642 220.402609,185.79839 209.663799,195.768166 C199.302587,205.38802 182.933414,204.874012 173.240413,194.508846 C171.247644,192.37176 169.677943,189.835329 167.706921,187.209935 L167.706921,187.209935 Z" fill="#4A4A4A"></path></g></svg>'''
268 icon = '''<?xml version="1.0" encoding="UTF-8" standalone="no"?><svg viewBox="0 0 256 239" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" preserveAspectRatio="xMidYMid"><g><path d="M119.540432,100.502743 C108.930124,118.338815 98.7646301,135.611455 88.3876025,152.753617 C85.7226696,157.154315 84.4040417,160.738531 86.5332204,166.333309 C92.4107024,181.787152 84.1193605,196.825836 68.5350381,200.908244 C53.8383677,204.759349 39.5192953,195.099955 36.6032893,179.365384 C34.0194114,165.437749 44.8274148,151.78491 60.1824106,149.608284 C61.4694072,149.424428 62.7821041,149.402681 64.944891,149.240571 C72.469175,136.623655 80.1773157,123.700312 88.3025935,110.073173 C73.611854,95.4654658 64.8677898,78.3885437 66.803227,57.2292132 C68.1712787,42.2715849 74.0527146,29.3462646 84.8033863,18.7517722 C105.393354,-1.53572199 136.805164,-4.82141828 161.048542,10.7510424 C184.333097,25.7086706 194.996783,54.8450075 185.906752,79.7822957 C179.052655,77.9239597 172.151111,76.049808 164.563565,73.9917997 C167.418285,60.1274266 165.306899,47.6765751 155.95591,37.0109123 C149.777932,29.9690049 141.850349,26.2780332 132.835442,24.9178894 C114.764113,22.1877169 97.0209573,33.7983633 91.7563309,51.5355878 C85.7800012,71.6669027 94.8245623,88.1111998 119.540432,100.502743 L119.540432,100.502743 Z" fill="#C73A63"></path><path d="M149.841194,79.4106285 C157.316054,92.5969067 164.905578,105.982857 172.427885,119.246236 C210.44865,107.483365 239.114472,128.530009 249.398582,151.063322 C261.81978,178.282014 253.328765,210.520191 228.933162,227.312431 C203.893073,244.551464 172.226236,241.605803 150.040866,219.46195 C155.694953,214.729124 161.376716,209.974552 167.44794,204.895759 C189.360489,219.088306 208.525074,218.420096 222.753207,201.614016 C234.885769,187.277151 234.622834,165.900356 222.138374,151.863988 C207.730339,135.66681 188.431321,135.172572 165.103273,150.721309 C155.426087,133.553447 145.58086,116.521995 136.210101,99.2295848 C133.05093,93.4015266 129.561608,90.0209366 122.440622,88.7873178 C110.547271,86.7253555 102.868785,76.5124151 102.408155,65.0698097 C101.955433,53.7537294 108.621719,43.5249733 119.04224,39.5394355 C129.363912,35.5914599 141.476705,38.7783085 148.419765,47.554004 C154.093621,54.7244134 155.896602,62.7943365 152.911402,71.6372484 C152.081082,74.1025091 151.00562,76.4886916 149.841194,79.4106285 L149.841194,79.4106285 Z" fill="#4B4B4B"></path><path d="M167.706921,187.209935 L121.936499,187.209935 C117.54964,205.253587 108.074103,219.821756 91.7464461,229.085759 C79.0544063,236.285822 65.3738898,238.72736 50.8136292,236.376762 C24.0061432,232.053165 2.08568567,207.920497 0.156179306,180.745298 C-2.02835403,149.962159 19.1309765,122.599149 47.3341915,116.452801 C49.2814904,123.524363 51.2485589,130.663141 53.1958579,137.716911 C27.3195169,150.919004 18.3639187,167.553089 25.6054984,188.352614 C31.9811726,206.657224 50.0900643,216.690262 69.7528413,212.809503 C89.8327554,208.847688 99.9567329,192.160226 98.7211371,165.37844 C117.75722,165.37844 136.809118,165.180745 155.847178,165.475311 C163.280522,165.591951 169.019617,164.820939 174.620326,158.267339 C183.840836,147.48306 200.811003,148.455721 210.741239,158.640984 C220.88894,169.049642 220.402609,185.79839 209.663799,195.768166 C199.302587,205.38802 182.933414,204.874012 173.240413,194.508846 C171.247644,192.37176 169.677943,189.835329 167.706921,187.209935 L167.706921,187.209935 Z" fill="#4A4A4A"></path></g></svg>'''
268
269
269 valid_events = [
270 valid_events = [
270 events.PullRequestCloseEvent,
271 events.PullRequestCloseEvent,
271 events.PullRequestMergeEvent,
272 events.PullRequestMergeEvent,
272 events.PullRequestUpdateEvent,
273 events.PullRequestUpdateEvent,
273 events.PullRequestCommentEvent,
274 events.PullRequestCommentEvent,
274 events.PullRequestReviewEvent,
275 events.PullRequestReviewEvent,
275 events.PullRequestCreateEvent,
276 events.PullRequestCreateEvent,
276 events.RepoPushEvent,
277 events.RepoPushEvent,
277 events.RepoCreateEvent,
278 events.RepoCreateEvent,
278 ]
279 ]
279
280
280 def settings_schema(self):
281 def settings_schema(self):
281 schema = WebhookSettingsSchema()
282 schema = WebhookSettingsSchema()
282 schema.add(colander.SchemaNode(
283 schema.add(colander.SchemaNode(
283 colander.Set(),
284 colander.Set(),
284 widget=deform.widget.CheckboxChoiceWidget(
285 widget=deform.widget.CheckboxChoiceWidget(
285 values=sorted(
286 values=sorted(
286 [(e.name, e.display_name) for e in self.valid_events]
287 [(e.name, e.display_name) for e in self.valid_events]
287 )
288 )
288 ),
289 ),
289 description="Events activated for this integration",
290 description="Events activated for this integration",
290 name='events'
291 name='events'
291 ))
292 ))
292 return schema
293 return schema
293
294
294 def send_event(self, event):
295 def send_event(self, event):
295 log.debug('handling event %s with Webhook integration %s',
296 log.debug('handling event %s with Webhook integration %s',
296 event.name, self)
297 event.name, self)
297
298
298 if event.__class__ not in self.valid_events:
299 if event.__class__ not in self.valid_events:
299 log.debug('event not valid: %r' % event)
300 log.debug('event not valid: %r' % event)
300 return
301 return
301
302
302 if event.name not in self.settings['events']:
303 if event.name not in self.settings['events']:
303 log.debug('event ignored: %r' % event)
304 log.debug('event ignored: %r' % event)
304 return
305 return
305
306
306 data = event.as_dict()
307 data = event.as_dict()
307 template_url = self.settings['url']
308 template_url = self.settings['url']
308
309
309 headers = {}
310 headers = {}
310 head_key = self.settings.get('custom_header_key')
311 head_key = self.settings.get('custom_header_key')
311 head_val = self.settings.get('custom_header_val')
312 head_val = self.settings.get('custom_header_val')
312 if head_key and head_val:
313 if head_key and head_val:
313 headers = {head_key: head_val}
314 headers = {head_key: head_val}
314
315
315 handler = WebhookHandler(
316 handler = WebhookHandler(
316 template_url, self.settings['secret_token'], headers)
317 template_url, self.settings['secret_token'], headers)
317
318
318 url_calls = handler(event, data)
319 url_calls = handler(event, data)
319 log.debug('webhook: calling following urls: %s',
320 log.debug('webhook: calling following urls: %s',
320 [x[0] for x in url_calls])
321 [x[0] for x in url_calls])
321
322
322 run_task(post_to_webhook, url_calls, self.settings)
323 run_task(post_to_webhook, url_calls, self.settings)
323
324
324
325
325 @async_task(ignore_result=True, base=RequestContextTask)
326 @async_task(ignore_result=True, base=RequestContextTask)
326 def post_to_webhook(url_calls, settings):
327 def post_to_webhook(url_calls, settings):
327 """
328 """
328 Example data::
329 Example data::
329
330
330 {'actor': {'user_id': 2, 'username': u'admin'},
331 {'actor': {'user_id': 2, 'username': u'admin'},
331 'actor_ip': u'192.168.157.1',
332 'actor_ip': u'192.168.157.1',
332 'name': 'repo-push',
333 'name': 'repo-push',
333 'push': {'branches': [{'name': u'default',
334 'push': {'branches': [{'name': u'default',
334 'url': 'http://rc.local:8080/hg-repo/changelog?branch=default'}],
335 'url': 'http://rc.local:8080/hg-repo/changelog?branch=default'}],
335 'commits': [{'author': u'Marcin Kuzminski <marcin@rhodecode.com>',
336 'commits': [{'author': u'Marcin Kuzminski <marcin@rhodecode.com>',
336 'branch': u'default',
337 'branch': u'default',
337 'date': datetime.datetime(2017, 11, 30, 12, 59, 48),
338 'date': datetime.datetime(2017, 11, 30, 12, 59, 48),
338 'issues': [],
339 'issues': [],
339 'mentions': [],
340 'mentions': [],
340 'message': u'commit Thu 30 Nov 2017 13:59:48 CET',
341 'message': u'commit Thu 30 Nov 2017 13:59:48 CET',
341 'message_html': u'commit Thu 30 Nov 2017 13:59:48 CET',
342 'message_html': u'commit Thu 30 Nov 2017 13:59:48 CET',
342 'message_html_title': u'commit Thu 30 Nov 2017 13:59:48 CET',
343 'message_html_title': u'commit Thu 30 Nov 2017 13:59:48 CET',
343 'parents': [{'raw_id': '431b772a5353dad9974b810dd3707d79e3a7f6e0'}],
344 'parents': [{'raw_id': '431b772a5353dad9974b810dd3707d79e3a7f6e0'}],
344 'permalink_url': u'http://rc.local:8080/_7/changeset/a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf',
345 'permalink_url': u'http://rc.local:8080/_7/changeset/a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf',
345 'raw_id': 'a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf',
346 'raw_id': 'a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf',
346 'refs': {'bookmarks': [], 'branches': [u'default'], 'tags': [u'tip']},
347 'refs': {'bookmarks': [], 'branches': [u'default'], 'tags': [u'tip']},
347 'reviewers': [],
348 'reviewers': [],
348 'revision': 9L,
349 'revision': 9L,
349 'short_id': 'a815cc738b96',
350 'short_id': 'a815cc738b96',
350 'url': u'http://rc.local:8080/hg-repo/changeset/a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf'}],
351 'url': u'http://rc.local:8080/hg-repo/changeset/a815cc738b9651eb5ffbcfb1ce6ccd7c701a5ddf'}],
351 'issues': {}},
352 'issues': {}},
352 'repo': {'extra_fields': '',
353 'repo': {'extra_fields': '',
353 'permalink_url': u'http://rc.local:8080/_7',
354 'permalink_url': u'http://rc.local:8080/_7',
354 'repo_id': 7,
355 'repo_id': 7,
355 'repo_name': u'hg-repo',
356 'repo_name': u'hg-repo',
356 'repo_type': u'hg',
357 'repo_type': u'hg',
357 'url': u'http://rc.local:8080/hg-repo'},
358 'url': u'http://rc.local:8080/hg-repo'},
358 'server_url': u'http://rc.local:8080',
359 'server_url': u'http://rc.local:8080',
359 'utc_timestamp': datetime.datetime(2017, 11, 30, 13, 0, 1, 569276)
360 'utc_timestamp': datetime.datetime(2017, 11, 30, 13, 0, 1, 569276)
360
361
361 """
362 """
362 max_retries = 3
363 max_retries = 3
363 retries = Retry(
364 retries = Retry(
364 total=max_retries,
365 total=max_retries,
365 backoff_factor=0.15,
366 backoff_factor=0.15,
366 status_forcelist=[500, 502, 503, 504])
367 status_forcelist=[500, 502, 503, 504])
367 call_headers = {
368 call_headers = {
368 'User-Agent': 'RhodeCode-webhook-caller/{}'.format(
369 'User-Agent': 'RhodeCode-webhook-caller/{}'.format(
369 rhodecode.__version__)
370 rhodecode.__version__)
370 } # updated below with custom ones, allows override
371 } # updated below with custom ones, allows override
371
372
372 for url, token, headers, data in url_calls:
373 for url, token, headers, data in url_calls:
373 req_session = requests.Session()
374 req_session = requests.Session()
374 req_session.mount( # retry max N times
375 req_session.mount( # retry max N times
375 'http://', requests.adapters.HTTPAdapter(max_retries=retries))
376 'http://', requests.adapters.HTTPAdapter(max_retries=retries))
376
377
377 method = settings.get('method_type') or 'post'
378 method = settings.get('method_type') or 'post'
378 call_method = getattr(req_session, method)
379 call_method = getattr(req_session, method)
379
380
380 headers = headers or {}
381 headers = headers or {}
381 call_headers.update(headers)
382 call_headers.update(headers)
382 auth = get_auth(settings)
383 auth = get_auth(settings)
383
384
384 log.debug('calling Webhook with method: %s, and auth:%s',
385 log.debug('calling Webhook with method: %s, and auth:%s',
385 call_method, auth)
386 call_method, auth)
386 if settings.get('log_data'):
387 if settings.get('log_data'):
387 log.debug('calling webhook with data: %s', data)
388 log.debug('calling webhook with data: %s', data)
388 resp = call_method(url, json={
389 resp = call_method(url, json={
389 'token': token,
390 'token': token,
390 'event': data
391 'event': data
391 }, headers=call_headers, auth=auth)
392 }, headers=call_headers, auth=auth)
392 log.debug('Got Webhook response: %s', resp)
393 log.debug('Got Webhook response: %s', resp)
393
394
394 resp.raise_for_status() # raise exception on a failed request
395 resp.raise_for_status() # raise exception on a failed request
@@ -1,735 +1,743
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2011-2018 RhodeCode GmbH
3 # Copyright (C) 2011-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21 import logging
21 import logging
22 import difflib
22 import difflib
23 from itertools import groupby
23 from itertools import groupby
24
24
25 from pygments import lex
25 from pygments import lex
26 from pygments.formatters.html import _get_ttype_class as pygment_token_class
26 from pygments.formatters.html import _get_ttype_class as pygment_token_class
27 from pygments.lexers.special import TextLexer, Token
28
27 from rhodecode.lib.helpers import (
29 from rhodecode.lib.helpers import (
28 get_lexer_for_filenode, html_escape, get_custom_lexer)
30 get_lexer_for_filenode, html_escape, get_custom_lexer)
29 from rhodecode.lib.utils2 import AttributeDict
31 from rhodecode.lib.utils2 import AttributeDict
30 from rhodecode.lib.vcs.nodes import FileNode
32 from rhodecode.lib.vcs.nodes import FileNode
31 from rhodecode.lib.diff_match_patch import diff_match_patch
33 from rhodecode.lib.diff_match_patch import diff_match_patch
32 from rhodecode.lib.diffs import LimitedDiffContainer
34 from rhodecode.lib.diffs import LimitedDiffContainer
33 from pygments.lexers import get_lexer_by_name
35 from pygments.lexers import get_lexer_by_name
34
36
35 plain_text_lexer = get_lexer_by_name(
37 plain_text_lexer = get_lexer_by_name(
36 'text', stripall=False, stripnl=False, ensurenl=False)
38 'text', stripall=False, stripnl=False, ensurenl=False)
37
39
38
40
39 log = logging.getLogger()
41 log = logging.getLogger()
40
42
41
43
42 def filenode_as_lines_tokens(filenode, lexer=None):
44 def filenode_as_lines_tokens(filenode, lexer=None):
43 org_lexer = lexer
45 org_lexer = lexer
44 lexer = lexer or get_lexer_for_filenode(filenode)
46 lexer = lexer or get_lexer_for_filenode(filenode)
45 log.debug('Generating file node pygment tokens for %s, %s, org_lexer:%s',
47 log.debug('Generating file node pygment tokens for %s, %s, org_lexer:%s',
46 lexer, filenode, org_lexer)
48 lexer, filenode, org_lexer)
47 tokens = tokenize_string(filenode.content, lexer)
49 tokens = tokenize_string(filenode.content, lexer)
48 lines = split_token_stream(tokens, split_string='\n')
50 lines = split_token_stream(tokens)
49 rv = list(lines)
51 rv = list(lines)
50 return rv
52 return rv
51
53
52
54
53 def tokenize_string(content, lexer):
55 def tokenize_string(content, lexer):
54 """
56 """
55 Use pygments to tokenize some content based on a lexer
57 Use pygments to tokenize some content based on a lexer
56 ensuring all original new lines and whitespace is preserved
58 ensuring all original new lines and whitespace is preserved
57 """
59 """
58
60
59 lexer.stripall = False
61 lexer.stripall = False
60 lexer.stripnl = False
62 lexer.stripnl = False
61 lexer.ensurenl = False
63 lexer.ensurenl = False
62 for token_type, token_text in lex(content, lexer):
64
65 if isinstance(lexer, TextLexer):
66 lexed = [(Token.Text, content)]
67 else:
68 lexed = lex(content, lexer)
69
70 for token_type, token_text in lexed:
63 yield pygment_token_class(token_type), token_text
71 yield pygment_token_class(token_type), token_text
64
72
65
73
66 def split_token_stream(tokens, split_string=u'\n'):
74 def split_token_stream(tokens):
67 """
75 """
68 Take a list of (TokenType, text) tuples and split them by a string
76 Take a list of (TokenType, text) tuples and split them by a string
69
77
70 >>> split_token_stream([(TEXT, 'some\ntext'), (TEXT, 'more\n')])
78 split_token_stream([(TEXT, 'some\ntext'), (TEXT, 'more\n')])
71 [(TEXT, 'some'), (TEXT, 'text'),
79 [(TEXT, 'some'), (TEXT, 'text'),
72 (TEXT, 'more'), (TEXT, 'text')]
80 (TEXT, 'more'), (TEXT, 'text')]
73 """
81 """
74
82
75 buffer = []
83 buffer = []
76 for token_class, token_text in tokens:
84 for token_class, token_text in tokens:
77 parts = token_text.split(split_string)
85 parts = token_text.split('\n')
78 for part in parts[:-1]:
86 for part in parts[:-1]:
79 buffer.append((token_class, part))
87 buffer.append((token_class, part))
80 yield buffer
88 yield buffer
81 buffer = []
89 buffer = []
82
90
83 buffer.append((token_class, parts[-1]))
91 buffer.append((token_class, parts[-1]))
84
92
85 if buffer:
93 if buffer:
86 yield buffer
94 yield buffer
87
95
88
96
89 def filenode_as_annotated_lines_tokens(filenode):
97 def filenode_as_annotated_lines_tokens(filenode):
90 """
98 """
91 Take a file node and return a list of annotations => lines, if no annotation
99 Take a file node and return a list of annotations => lines, if no annotation
92 is found, it will be None.
100 is found, it will be None.
93
101
94 eg:
102 eg:
95
103
96 [
104 [
97 (annotation1, [
105 (annotation1, [
98 (1, line1_tokens_list),
106 (1, line1_tokens_list),
99 (2, line2_tokens_list),
107 (2, line2_tokens_list),
100 ]),
108 ]),
101 (annotation2, [
109 (annotation2, [
102 (3, line1_tokens_list),
110 (3, line1_tokens_list),
103 ]),
111 ]),
104 (None, [
112 (None, [
105 (4, line1_tokens_list),
113 (4, line1_tokens_list),
106 ]),
114 ]),
107 (annotation1, [
115 (annotation1, [
108 (5, line1_tokens_list),
116 (5, line1_tokens_list),
109 (6, line2_tokens_list),
117 (6, line2_tokens_list),
110 ])
118 ])
111 ]
119 ]
112 """
120 """
113
121
114 commit_cache = {} # cache commit_getter lookups
122 commit_cache = {} # cache commit_getter lookups
115
123
116 def _get_annotation(commit_id, commit_getter):
124 def _get_annotation(commit_id, commit_getter):
117 if commit_id not in commit_cache:
125 if commit_id not in commit_cache:
118 commit_cache[commit_id] = commit_getter()
126 commit_cache[commit_id] = commit_getter()
119 return commit_cache[commit_id]
127 return commit_cache[commit_id]
120
128
121 annotation_lookup = {
129 annotation_lookup = {
122 line_no: _get_annotation(commit_id, commit_getter)
130 line_no: _get_annotation(commit_id, commit_getter)
123 for line_no, commit_id, commit_getter, line_content
131 for line_no, commit_id, commit_getter, line_content
124 in filenode.annotate
132 in filenode.annotate
125 }
133 }
126
134
127 annotations_lines = ((annotation_lookup.get(line_no), line_no, tokens)
135 annotations_lines = ((annotation_lookup.get(line_no), line_no, tokens)
128 for line_no, tokens
136 for line_no, tokens
129 in enumerate(filenode_as_lines_tokens(filenode), 1))
137 in enumerate(filenode_as_lines_tokens(filenode), 1))
130
138
131 grouped_annotations_lines = groupby(annotations_lines, lambda x: x[0])
139 grouped_annotations_lines = groupby(annotations_lines, lambda x: x[0])
132
140
133 for annotation, group in grouped_annotations_lines:
141 for annotation, group in grouped_annotations_lines:
134 yield (
142 yield (
135 annotation, [(line_no, tokens)
143 annotation, [(line_no, tokens)
136 for (_, line_no, tokens) in group]
144 for (_, line_no, tokens) in group]
137 )
145 )
138
146
139
147
140 def render_tokenstream(tokenstream):
148 def render_tokenstream(tokenstream):
141 result = []
149 result = []
142 for token_class, token_ops_texts in rollup_tokenstream(tokenstream):
150 for token_class, token_ops_texts in rollup_tokenstream(tokenstream):
143
151
144 if token_class:
152 if token_class:
145 result.append(u'<span class="%s">' % token_class)
153 result.append(u'<span class="%s">' % token_class)
146 else:
154 else:
147 result.append(u'<span>')
155 result.append(u'<span>')
148
156
149 for op_tag, token_text in token_ops_texts:
157 for op_tag, token_text in token_ops_texts:
150
158
151 if op_tag:
159 if op_tag:
152 result.append(u'<%s>' % op_tag)
160 result.append(u'<%s>' % op_tag)
153
161
154 escaped_text = html_escape(token_text)
162 escaped_text = html_escape(token_text)
155
163
156 # TODO: dan: investigate showing hidden characters like space/nl/tab
164 # TODO: dan: investigate showing hidden characters like space/nl/tab
157 # escaped_text = escaped_text.replace(' ', '<sp> </sp>')
165 # escaped_text = escaped_text.replace(' ', '<sp> </sp>')
158 # escaped_text = escaped_text.replace('\n', '<nl>\n</nl>')
166 # escaped_text = escaped_text.replace('\n', '<nl>\n</nl>')
159 # escaped_text = escaped_text.replace('\t', '<tab>\t</tab>')
167 # escaped_text = escaped_text.replace('\t', '<tab>\t</tab>')
160
168
161 result.append(escaped_text)
169 result.append(escaped_text)
162
170
163 if op_tag:
171 if op_tag:
164 result.append(u'</%s>' % op_tag)
172 result.append(u'</%s>' % op_tag)
165
173
166 result.append(u'</span>')
174 result.append(u'</span>')
167
175
168 html = ''.join(result)
176 html = ''.join(result)
169 return html
177 return html
170
178
171
179
172 def rollup_tokenstream(tokenstream):
180 def rollup_tokenstream(tokenstream):
173 """
181 """
174 Group a token stream of the format:
182 Group a token stream of the format:
175
183
176 ('class', 'op', 'text')
184 ('class', 'op', 'text')
177 or
185 or
178 ('class', 'text')
186 ('class', 'text')
179
187
180 into
188 into
181
189
182 [('class1',
190 [('class1',
183 [('op1', 'text'),
191 [('op1', 'text'),
184 ('op2', 'text')]),
192 ('op2', 'text')]),
185 ('class2',
193 ('class2',
186 [('op3', 'text')])]
194 [('op3', 'text')])]
187
195
188 This is used to get the minimal tags necessary when
196 This is used to get the minimal tags necessary when
189 rendering to html eg for a token stream ie.
197 rendering to html eg for a token stream ie.
190
198
191 <span class="A"><ins>he</ins>llo</span>
199 <span class="A"><ins>he</ins>llo</span>
192 vs
200 vs
193 <span class="A"><ins>he</ins></span><span class="A">llo</span>
201 <span class="A"><ins>he</ins></span><span class="A">llo</span>
194
202
195 If a 2 tuple is passed in, the output op will be an empty string.
203 If a 2 tuple is passed in, the output op will be an empty string.
196
204
197 eg:
205 eg:
198
206
199 >>> rollup_tokenstream([('classA', '', 'h'),
207 >>> rollup_tokenstream([('classA', '', 'h'),
200 ('classA', 'del', 'ell'),
208 ('classA', 'del', 'ell'),
201 ('classA', '', 'o'),
209 ('classA', '', 'o'),
202 ('classB', '', ' '),
210 ('classB', '', ' '),
203 ('classA', '', 'the'),
211 ('classA', '', 'the'),
204 ('classA', '', 're'),
212 ('classA', '', 're'),
205 ])
213 ])
206
214
207 [('classA', [('', 'h'), ('del', 'ell'), ('', 'o')],
215 [('classA', [('', 'h'), ('del', 'ell'), ('', 'o')],
208 ('classB', [('', ' ')],
216 ('classB', [('', ' ')],
209 ('classA', [('', 'there')]]
217 ('classA', [('', 'there')]]
210
218
211 """
219 """
212 if tokenstream and len(tokenstream[0]) == 2:
220 if tokenstream and len(tokenstream[0]) == 2:
213 tokenstream = ((t[0], '', t[1]) for t in tokenstream)
221 tokenstream = ((t[0], '', t[1]) for t in tokenstream)
214
222
215 result = []
223 result = []
216 for token_class, op_list in groupby(tokenstream, lambda t: t[0]):
224 for token_class, op_list in groupby(tokenstream, lambda t: t[0]):
217 ops = []
225 ops = []
218 for token_op, token_text_list in groupby(op_list, lambda o: o[1]):
226 for token_op, token_text_list in groupby(op_list, lambda o: o[1]):
219 text_buffer = []
227 text_buffer = []
220 for t_class, t_op, t_text in token_text_list:
228 for t_class, t_op, t_text in token_text_list:
221 text_buffer.append(t_text)
229 text_buffer.append(t_text)
222 ops.append((token_op, ''.join(text_buffer)))
230 ops.append((token_op, ''.join(text_buffer)))
223 result.append((token_class, ops))
231 result.append((token_class, ops))
224 return result
232 return result
225
233
226
234
227 def tokens_diff(old_tokens, new_tokens, use_diff_match_patch=True):
235 def tokens_diff(old_tokens, new_tokens, use_diff_match_patch=True):
228 """
236 """
229 Converts a list of (token_class, token_text) tuples to a list of
237 Converts a list of (token_class, token_text) tuples to a list of
230 (token_class, token_op, token_text) tuples where token_op is one of
238 (token_class, token_op, token_text) tuples where token_op is one of
231 ('ins', 'del', '')
239 ('ins', 'del', '')
232
240
233 :param old_tokens: list of (token_class, token_text) tuples of old line
241 :param old_tokens: list of (token_class, token_text) tuples of old line
234 :param new_tokens: list of (token_class, token_text) tuples of new line
242 :param new_tokens: list of (token_class, token_text) tuples of new line
235 :param use_diff_match_patch: boolean, will use google's diff match patch
243 :param use_diff_match_patch: boolean, will use google's diff match patch
236 library which has options to 'smooth' out the character by character
244 library which has options to 'smooth' out the character by character
237 differences making nicer ins/del blocks
245 differences making nicer ins/del blocks
238 """
246 """
239
247
240 old_tokens_result = []
248 old_tokens_result = []
241 new_tokens_result = []
249 new_tokens_result = []
242
250
243 similarity = difflib.SequenceMatcher(None,
251 similarity = difflib.SequenceMatcher(None,
244 ''.join(token_text for token_class, token_text in old_tokens),
252 ''.join(token_text for token_class, token_text in old_tokens),
245 ''.join(token_text for token_class, token_text in new_tokens)
253 ''.join(token_text for token_class, token_text in new_tokens)
246 ).ratio()
254 ).ratio()
247
255
248 if similarity < 0.6: # return, the blocks are too different
256 if similarity < 0.6: # return, the blocks are too different
249 for token_class, token_text in old_tokens:
257 for token_class, token_text in old_tokens:
250 old_tokens_result.append((token_class, '', token_text))
258 old_tokens_result.append((token_class, '', token_text))
251 for token_class, token_text in new_tokens:
259 for token_class, token_text in new_tokens:
252 new_tokens_result.append((token_class, '', token_text))
260 new_tokens_result.append((token_class, '', token_text))
253 return old_tokens_result, new_tokens_result, similarity
261 return old_tokens_result, new_tokens_result, similarity
254
262
255 token_sequence_matcher = difflib.SequenceMatcher(None,
263 token_sequence_matcher = difflib.SequenceMatcher(None,
256 [x[1] for x in old_tokens],
264 [x[1] for x in old_tokens],
257 [x[1] for x in new_tokens])
265 [x[1] for x in new_tokens])
258
266
259 for tag, o1, o2, n1, n2 in token_sequence_matcher.get_opcodes():
267 for tag, o1, o2, n1, n2 in token_sequence_matcher.get_opcodes():
260 # check the differences by token block types first to give a more
268 # check the differences by token block types first to give a more
261 # nicer "block" level replacement vs character diffs
269 # nicer "block" level replacement vs character diffs
262
270
263 if tag == 'equal':
271 if tag == 'equal':
264 for token_class, token_text in old_tokens[o1:o2]:
272 for token_class, token_text in old_tokens[o1:o2]:
265 old_tokens_result.append((token_class, '', token_text))
273 old_tokens_result.append((token_class, '', token_text))
266 for token_class, token_text in new_tokens[n1:n2]:
274 for token_class, token_text in new_tokens[n1:n2]:
267 new_tokens_result.append((token_class, '', token_text))
275 new_tokens_result.append((token_class, '', token_text))
268 elif tag == 'delete':
276 elif tag == 'delete':
269 for token_class, token_text in old_tokens[o1:o2]:
277 for token_class, token_text in old_tokens[o1:o2]:
270 old_tokens_result.append((token_class, 'del', token_text))
278 old_tokens_result.append((token_class, 'del', token_text))
271 elif tag == 'insert':
279 elif tag == 'insert':
272 for token_class, token_text in new_tokens[n1:n2]:
280 for token_class, token_text in new_tokens[n1:n2]:
273 new_tokens_result.append((token_class, 'ins', token_text))
281 new_tokens_result.append((token_class, 'ins', token_text))
274 elif tag == 'replace':
282 elif tag == 'replace':
275 # if same type token blocks must be replaced, do a diff on the
283 # if same type token blocks must be replaced, do a diff on the
276 # characters in the token blocks to show individual changes
284 # characters in the token blocks to show individual changes
277
285
278 old_char_tokens = []
286 old_char_tokens = []
279 new_char_tokens = []
287 new_char_tokens = []
280 for token_class, token_text in old_tokens[o1:o2]:
288 for token_class, token_text in old_tokens[o1:o2]:
281 for char in token_text:
289 for char in token_text:
282 old_char_tokens.append((token_class, char))
290 old_char_tokens.append((token_class, char))
283
291
284 for token_class, token_text in new_tokens[n1:n2]:
292 for token_class, token_text in new_tokens[n1:n2]:
285 for char in token_text:
293 for char in token_text:
286 new_char_tokens.append((token_class, char))
294 new_char_tokens.append((token_class, char))
287
295
288 old_string = ''.join([token_text for
296 old_string = ''.join([token_text for
289 token_class, token_text in old_char_tokens])
297 token_class, token_text in old_char_tokens])
290 new_string = ''.join([token_text for
298 new_string = ''.join([token_text for
291 token_class, token_text in new_char_tokens])
299 token_class, token_text in new_char_tokens])
292
300
293 char_sequence = difflib.SequenceMatcher(
301 char_sequence = difflib.SequenceMatcher(
294 None, old_string, new_string)
302 None, old_string, new_string)
295 copcodes = char_sequence.get_opcodes()
303 copcodes = char_sequence.get_opcodes()
296 obuffer, nbuffer = [], []
304 obuffer, nbuffer = [], []
297
305
298 if use_diff_match_patch:
306 if use_diff_match_patch:
299 dmp = diff_match_patch()
307 dmp = diff_match_patch()
300 dmp.Diff_EditCost = 11 # TODO: dan: extract this to a setting
308 dmp.Diff_EditCost = 11 # TODO: dan: extract this to a setting
301 reps = dmp.diff_main(old_string, new_string)
309 reps = dmp.diff_main(old_string, new_string)
302 dmp.diff_cleanupEfficiency(reps)
310 dmp.diff_cleanupEfficiency(reps)
303
311
304 a, b = 0, 0
312 a, b = 0, 0
305 for op, rep in reps:
313 for op, rep in reps:
306 l = len(rep)
314 l = len(rep)
307 if op == 0:
315 if op == 0:
308 for i, c in enumerate(rep):
316 for i, c in enumerate(rep):
309 obuffer.append((old_char_tokens[a+i][0], '', c))
317 obuffer.append((old_char_tokens[a+i][0], '', c))
310 nbuffer.append((new_char_tokens[b+i][0], '', c))
318 nbuffer.append((new_char_tokens[b+i][0], '', c))
311 a += l
319 a += l
312 b += l
320 b += l
313 elif op == -1:
321 elif op == -1:
314 for i, c in enumerate(rep):
322 for i, c in enumerate(rep):
315 obuffer.append((old_char_tokens[a+i][0], 'del', c))
323 obuffer.append((old_char_tokens[a+i][0], 'del', c))
316 a += l
324 a += l
317 elif op == 1:
325 elif op == 1:
318 for i, c in enumerate(rep):
326 for i, c in enumerate(rep):
319 nbuffer.append((new_char_tokens[b+i][0], 'ins', c))
327 nbuffer.append((new_char_tokens[b+i][0], 'ins', c))
320 b += l
328 b += l
321 else:
329 else:
322 for ctag, co1, co2, cn1, cn2 in copcodes:
330 for ctag, co1, co2, cn1, cn2 in copcodes:
323 if ctag == 'equal':
331 if ctag == 'equal':
324 for token_class, token_text in old_char_tokens[co1:co2]:
332 for token_class, token_text in old_char_tokens[co1:co2]:
325 obuffer.append((token_class, '', token_text))
333 obuffer.append((token_class, '', token_text))
326 for token_class, token_text in new_char_tokens[cn1:cn2]:
334 for token_class, token_text in new_char_tokens[cn1:cn2]:
327 nbuffer.append((token_class, '', token_text))
335 nbuffer.append((token_class, '', token_text))
328 elif ctag == 'delete':
336 elif ctag == 'delete':
329 for token_class, token_text in old_char_tokens[co1:co2]:
337 for token_class, token_text in old_char_tokens[co1:co2]:
330 obuffer.append((token_class, 'del', token_text))
338 obuffer.append((token_class, 'del', token_text))
331 elif ctag == 'insert':
339 elif ctag == 'insert':
332 for token_class, token_text in new_char_tokens[cn1:cn2]:
340 for token_class, token_text in new_char_tokens[cn1:cn2]:
333 nbuffer.append((token_class, 'ins', token_text))
341 nbuffer.append((token_class, 'ins', token_text))
334 elif ctag == 'replace':
342 elif ctag == 'replace':
335 for token_class, token_text in old_char_tokens[co1:co2]:
343 for token_class, token_text in old_char_tokens[co1:co2]:
336 obuffer.append((token_class, 'del', token_text))
344 obuffer.append((token_class, 'del', token_text))
337 for token_class, token_text in new_char_tokens[cn1:cn2]:
345 for token_class, token_text in new_char_tokens[cn1:cn2]:
338 nbuffer.append((token_class, 'ins', token_text))
346 nbuffer.append((token_class, 'ins', token_text))
339
347
340 old_tokens_result.extend(obuffer)
348 old_tokens_result.extend(obuffer)
341 new_tokens_result.extend(nbuffer)
349 new_tokens_result.extend(nbuffer)
342
350
343 return old_tokens_result, new_tokens_result, similarity
351 return old_tokens_result, new_tokens_result, similarity
344
352
345
353
346 class DiffSet(object):
354 class DiffSet(object):
347 """
355 """
348 An object for parsing the diff result from diffs.DiffProcessor and
356 An object for parsing the diff result from diffs.DiffProcessor and
349 adding highlighting, side by side/unified renderings and line diffs
357 adding highlighting, side by side/unified renderings and line diffs
350 """
358 """
351
359
352 HL_REAL = 'REAL' # highlights using original file, slow
360 HL_REAL = 'REAL' # highlights using original file, slow
353 HL_FAST = 'FAST' # highlights using just the line, fast but not correct
361 HL_FAST = 'FAST' # highlights using just the line, fast but not correct
354 # in the case of multiline code
362 # in the case of multiline code
355 HL_NONE = 'NONE' # no highlighting, fastest
363 HL_NONE = 'NONE' # no highlighting, fastest
356
364
357 def __init__(self, highlight_mode=HL_REAL, repo_name=None,
365 def __init__(self, highlight_mode=HL_REAL, repo_name=None,
358 source_repo_name=None,
366 source_repo_name=None,
359 source_node_getter=lambda filename: None,
367 source_node_getter=lambda filename: None,
360 target_node_getter=lambda filename: None,
368 target_node_getter=lambda filename: None,
361 source_nodes=None, target_nodes=None,
369 source_nodes=None, target_nodes=None,
362 max_file_size_limit=150 * 1024, # files over this size will
370 max_file_size_limit=150 * 1024, # files over this size will
363 # use fast highlighting
371 # use fast highlighting
364 comments=None,
372 comments=None,
365 ):
373 ):
366
374
367 self.highlight_mode = highlight_mode
375 self.highlight_mode = highlight_mode
368 self.highlighted_filenodes = {}
376 self.highlighted_filenodes = {}
369 self.source_node_getter = source_node_getter
377 self.source_node_getter = source_node_getter
370 self.target_node_getter = target_node_getter
378 self.target_node_getter = target_node_getter
371 self.source_nodes = source_nodes or {}
379 self.source_nodes = source_nodes or {}
372 self.target_nodes = target_nodes or {}
380 self.target_nodes = target_nodes or {}
373 self.repo_name = repo_name
381 self.repo_name = repo_name
374 self.source_repo_name = source_repo_name or repo_name
382 self.source_repo_name = source_repo_name or repo_name
375 self.comments = comments or {}
383 self.comments = comments or {}
376 self.comments_store = self.comments.copy()
384 self.comments_store = self.comments.copy()
377 self.max_file_size_limit = max_file_size_limit
385 self.max_file_size_limit = max_file_size_limit
378
386
379 def render_patchset(self, patchset, source_ref=None, target_ref=None):
387 def render_patchset(self, patchset, source_ref=None, target_ref=None):
380 diffset = AttributeDict(dict(
388 diffset = AttributeDict(dict(
381 lines_added=0,
389 lines_added=0,
382 lines_deleted=0,
390 lines_deleted=0,
383 changed_files=0,
391 changed_files=0,
384 files=[],
392 files=[],
385 file_stats={},
393 file_stats={},
386 limited_diff=isinstance(patchset, LimitedDiffContainer),
394 limited_diff=isinstance(patchset, LimitedDiffContainer),
387 repo_name=self.repo_name,
395 repo_name=self.repo_name,
388 source_repo_name=self.source_repo_name,
396 source_repo_name=self.source_repo_name,
389 source_ref=source_ref,
397 source_ref=source_ref,
390 target_ref=target_ref,
398 target_ref=target_ref,
391 ))
399 ))
392 for patch in patchset:
400 for patch in patchset:
393 diffset.file_stats[patch['filename']] = patch['stats']
401 diffset.file_stats[patch['filename']] = patch['stats']
394 filediff = self.render_patch(patch)
402 filediff = self.render_patch(patch)
395 filediff.diffset = diffset
403 filediff.diffset = diffset
396 diffset.files.append(filediff)
404 diffset.files.append(filediff)
397 diffset.changed_files += 1
405 diffset.changed_files += 1
398 if not patch['stats']['binary']:
406 if not patch['stats']['binary']:
399 diffset.lines_added += patch['stats']['added']
407 diffset.lines_added += patch['stats']['added']
400 diffset.lines_deleted += patch['stats']['deleted']
408 diffset.lines_deleted += patch['stats']['deleted']
401
409
402 return diffset
410 return diffset
403
411
404 _lexer_cache = {}
412 _lexer_cache = {}
405
413
406 def _get_lexer_for_filename(self, filename, filenode=None):
414 def _get_lexer_for_filename(self, filename, filenode=None):
407 # cached because we might need to call it twice for source/target
415 # cached because we might need to call it twice for source/target
408 if filename not in self._lexer_cache:
416 if filename not in self._lexer_cache:
409 if filenode:
417 if filenode:
410 lexer = filenode.lexer
418 lexer = filenode.lexer
411 extension = filenode.extension
419 extension = filenode.extension
412 else:
420 else:
413 lexer = FileNode.get_lexer(filename=filename)
421 lexer = FileNode.get_lexer(filename=filename)
414 extension = filename.split('.')[-1]
422 extension = filename.split('.')[-1]
415
423
416 lexer = get_custom_lexer(extension) or lexer
424 lexer = get_custom_lexer(extension) or lexer
417 self._lexer_cache[filename] = lexer
425 self._lexer_cache[filename] = lexer
418 return self._lexer_cache[filename]
426 return self._lexer_cache[filename]
419
427
420 def render_patch(self, patch):
428 def render_patch(self, patch):
421 log.debug('rendering diff for %r' % patch['filename'])
429 log.debug('rendering diff for %r' % patch['filename'])
422
430
423 source_filename = patch['original_filename']
431 source_filename = patch['original_filename']
424 target_filename = patch['filename']
432 target_filename = patch['filename']
425
433
426 source_lexer = plain_text_lexer
434 source_lexer = plain_text_lexer
427 target_lexer = plain_text_lexer
435 target_lexer = plain_text_lexer
428
436
429 if not patch['stats']['binary']:
437 if not patch['stats']['binary']:
430 if self.highlight_mode == self.HL_REAL:
438 if self.highlight_mode == self.HL_REAL:
431 if (source_filename and patch['operation'] in ('D', 'M')
439 if (source_filename and patch['operation'] in ('D', 'M')
432 and source_filename not in self.source_nodes):
440 and source_filename not in self.source_nodes):
433 self.source_nodes[source_filename] = (
441 self.source_nodes[source_filename] = (
434 self.source_node_getter(source_filename))
442 self.source_node_getter(source_filename))
435
443
436 if (target_filename and patch['operation'] in ('A', 'M')
444 if (target_filename and patch['operation'] in ('A', 'M')
437 and target_filename not in self.target_nodes):
445 and target_filename not in self.target_nodes):
438 self.target_nodes[target_filename] = (
446 self.target_nodes[target_filename] = (
439 self.target_node_getter(target_filename))
447 self.target_node_getter(target_filename))
440
448
441 elif self.highlight_mode == self.HL_FAST:
449 elif self.highlight_mode == self.HL_FAST:
442 source_lexer = self._get_lexer_for_filename(source_filename)
450 source_lexer = self._get_lexer_for_filename(source_filename)
443 target_lexer = self._get_lexer_for_filename(target_filename)
451 target_lexer = self._get_lexer_for_filename(target_filename)
444
452
445 source_file = self.source_nodes.get(source_filename, source_filename)
453 source_file = self.source_nodes.get(source_filename, source_filename)
446 target_file = self.target_nodes.get(target_filename, target_filename)
454 target_file = self.target_nodes.get(target_filename, target_filename)
447
455
448 source_filenode, target_filenode = None, None
456 source_filenode, target_filenode = None, None
449
457
450 # TODO: dan: FileNode.lexer works on the content of the file - which
458 # TODO: dan: FileNode.lexer works on the content of the file - which
451 # can be slow - issue #4289 explains a lexer clean up - which once
459 # can be slow - issue #4289 explains a lexer clean up - which once
452 # done can allow caching a lexer for a filenode to avoid the file lookup
460 # done can allow caching a lexer for a filenode to avoid the file lookup
453 if isinstance(source_file, FileNode):
461 if isinstance(source_file, FileNode):
454 source_filenode = source_file
462 source_filenode = source_file
455 #source_lexer = source_file.lexer
463 #source_lexer = source_file.lexer
456 source_lexer = self._get_lexer_for_filename(source_filename)
464 source_lexer = self._get_lexer_for_filename(source_filename)
457 source_file.lexer = source_lexer
465 source_file.lexer = source_lexer
458
466
459 if isinstance(target_file, FileNode):
467 if isinstance(target_file, FileNode):
460 target_filenode = target_file
468 target_filenode = target_file
461 #target_lexer = target_file.lexer
469 #target_lexer = target_file.lexer
462 target_lexer = self._get_lexer_for_filename(target_filename)
470 target_lexer = self._get_lexer_for_filename(target_filename)
463 target_file.lexer = target_lexer
471 target_file.lexer = target_lexer
464
472
465 source_file_path, target_file_path = None, None
473 source_file_path, target_file_path = None, None
466
474
467 if source_filename != '/dev/null':
475 if source_filename != '/dev/null':
468 source_file_path = source_filename
476 source_file_path = source_filename
469 if target_filename != '/dev/null':
477 if target_filename != '/dev/null':
470 target_file_path = target_filename
478 target_file_path = target_filename
471
479
472 source_file_type = source_lexer.name
480 source_file_type = source_lexer.name
473 target_file_type = target_lexer.name
481 target_file_type = target_lexer.name
474
482
475 filediff = AttributeDict({
483 filediff = AttributeDict({
476 'source_file_path': source_file_path,
484 'source_file_path': source_file_path,
477 'target_file_path': target_file_path,
485 'target_file_path': target_file_path,
478 'source_filenode': source_filenode,
486 'source_filenode': source_filenode,
479 'target_filenode': target_filenode,
487 'target_filenode': target_filenode,
480 'source_file_type': target_file_type,
488 'source_file_type': target_file_type,
481 'target_file_type': source_file_type,
489 'target_file_type': source_file_type,
482 'patch': {'filename': patch['filename'], 'stats': patch['stats']},
490 'patch': {'filename': patch['filename'], 'stats': patch['stats']},
483 'operation': patch['operation'],
491 'operation': patch['operation'],
484 'source_mode': patch['stats']['old_mode'],
492 'source_mode': patch['stats']['old_mode'],
485 'target_mode': patch['stats']['new_mode'],
493 'target_mode': patch['stats']['new_mode'],
486 'limited_diff': isinstance(patch, LimitedDiffContainer),
494 'limited_diff': isinstance(patch, LimitedDiffContainer),
487 'hunks': [],
495 'hunks': [],
488 'diffset': self,
496 'diffset': self,
489 })
497 })
490
498
491 for hunk in patch['chunks'][1:]:
499 for hunk in patch['chunks'][1:]:
492 hunkbit = self.parse_hunk(hunk, source_file, target_file)
500 hunkbit = self.parse_hunk(hunk, source_file, target_file)
493 hunkbit.source_file_path = source_file_path
501 hunkbit.source_file_path = source_file_path
494 hunkbit.target_file_path = target_file_path
502 hunkbit.target_file_path = target_file_path
495 filediff.hunks.append(hunkbit)
503 filediff.hunks.append(hunkbit)
496
504
497 left_comments = {}
505 left_comments = {}
498 if source_file_path in self.comments_store:
506 if source_file_path in self.comments_store:
499 for lineno, comments in self.comments_store[source_file_path].items():
507 for lineno, comments in self.comments_store[source_file_path].items():
500 left_comments[lineno] = comments
508 left_comments[lineno] = comments
501
509
502 if target_file_path in self.comments_store:
510 if target_file_path in self.comments_store:
503 for lineno, comments in self.comments_store[target_file_path].items():
511 for lineno, comments in self.comments_store[target_file_path].items():
504 left_comments[lineno] = comments
512 left_comments[lineno] = comments
505 # left comments are one that we couldn't place in diff lines.
513 # left comments are one that we couldn't place in diff lines.
506 # could be outdated, or the diff changed and this line is no
514 # could be outdated, or the diff changed and this line is no
507 # longer available
515 # longer available
508 filediff.left_comments = left_comments
516 filediff.left_comments = left_comments
509
517
510 return filediff
518 return filediff
511
519
512 def parse_hunk(self, hunk, source_file, target_file):
520 def parse_hunk(self, hunk, source_file, target_file):
513 result = AttributeDict(dict(
521 result = AttributeDict(dict(
514 source_start=hunk['source_start'],
522 source_start=hunk['source_start'],
515 source_length=hunk['source_length'],
523 source_length=hunk['source_length'],
516 target_start=hunk['target_start'],
524 target_start=hunk['target_start'],
517 target_length=hunk['target_length'],
525 target_length=hunk['target_length'],
518 section_header=hunk['section_header'],
526 section_header=hunk['section_header'],
519 lines=[],
527 lines=[],
520 ))
528 ))
521 before, after = [], []
529 before, after = [], []
522
530
523 for line in hunk['lines']:
531 for line in hunk['lines']:
524
532
525 if line['action'] == 'unmod':
533 if line['action'] == 'unmod':
526 result.lines.extend(
534 result.lines.extend(
527 self.parse_lines(before, after, source_file, target_file))
535 self.parse_lines(before, after, source_file, target_file))
528 after.append(line)
536 after.append(line)
529 before.append(line)
537 before.append(line)
530 elif line['action'] == 'add':
538 elif line['action'] == 'add':
531 after.append(line)
539 after.append(line)
532 elif line['action'] == 'del':
540 elif line['action'] == 'del':
533 before.append(line)
541 before.append(line)
534 elif line['action'] == 'old-no-nl':
542 elif line['action'] == 'old-no-nl':
535 before.append(line)
543 before.append(line)
536 elif line['action'] == 'new-no-nl':
544 elif line['action'] == 'new-no-nl':
537 after.append(line)
545 after.append(line)
538
546
539 result.lines.extend(
547 result.lines.extend(
540 self.parse_lines(before, after, source_file, target_file))
548 self.parse_lines(before, after, source_file, target_file))
541 result.unified = self.as_unified(result.lines)
549 result.unified = self.as_unified(result.lines)
542 result.sideside = result.lines
550 result.sideside = result.lines
543
551
544 return result
552 return result
545
553
546 def parse_lines(self, before_lines, after_lines, source_file, target_file):
554 def parse_lines(self, before_lines, after_lines, source_file, target_file):
547 # TODO: dan: investigate doing the diff comparison and fast highlighting
555 # TODO: dan: investigate doing the diff comparison and fast highlighting
548 # on the entire before and after buffered block lines rather than by
556 # on the entire before and after buffered block lines rather than by
549 # line, this means we can get better 'fast' highlighting if the context
557 # line, this means we can get better 'fast' highlighting if the context
550 # allows it - eg.
558 # allows it - eg.
551 # line 4: """
559 # line 4: """
552 # line 5: this gets highlighted as a string
560 # line 5: this gets highlighted as a string
553 # line 6: """
561 # line 6: """
554
562
555 lines = []
563 lines = []
556
564
557 before_newline = AttributeDict()
565 before_newline = AttributeDict()
558 after_newline = AttributeDict()
566 after_newline = AttributeDict()
559 if before_lines and before_lines[-1]['action'] == 'old-no-nl':
567 if before_lines and before_lines[-1]['action'] == 'old-no-nl':
560 before_newline_line = before_lines.pop(-1)
568 before_newline_line = before_lines.pop(-1)
561 before_newline.content = '\n {}'.format(
569 before_newline.content = '\n {}'.format(
562 render_tokenstream(
570 render_tokenstream(
563 [(x[0], '', x[1])
571 [(x[0], '', x[1])
564 for x in [('nonl', before_newline_line['line'])]]))
572 for x in [('nonl', before_newline_line['line'])]]))
565
573
566 if after_lines and after_lines[-1]['action'] == 'new-no-nl':
574 if after_lines and after_lines[-1]['action'] == 'new-no-nl':
567 after_newline_line = after_lines.pop(-1)
575 after_newline_line = after_lines.pop(-1)
568 after_newline.content = '\n {}'.format(
576 after_newline.content = '\n {}'.format(
569 render_tokenstream(
577 render_tokenstream(
570 [(x[0], '', x[1])
578 [(x[0], '', x[1])
571 for x in [('nonl', after_newline_line['line'])]]))
579 for x in [('nonl', after_newline_line['line'])]]))
572
580
573 while before_lines or after_lines:
581 while before_lines or after_lines:
574 before, after = None, None
582 before, after = None, None
575 before_tokens, after_tokens = None, None
583 before_tokens, after_tokens = None, None
576
584
577 if before_lines:
585 if before_lines:
578 before = before_lines.pop(0)
586 before = before_lines.pop(0)
579 if after_lines:
587 if after_lines:
580 after = after_lines.pop(0)
588 after = after_lines.pop(0)
581
589
582 original = AttributeDict()
590 original = AttributeDict()
583 modified = AttributeDict()
591 modified = AttributeDict()
584
592
585 if before:
593 if before:
586 if before['action'] == 'old-no-nl':
594 if before['action'] == 'old-no-nl':
587 before_tokens = [('nonl', before['line'])]
595 before_tokens = [('nonl', before['line'])]
588 else:
596 else:
589 before_tokens = self.get_line_tokens(
597 before_tokens = self.get_line_tokens(
590 line_text=before['line'],
598 line_text=before['line'],
591 line_number=before['old_lineno'],
599 line_number=before['old_lineno'],
592 file=source_file)
600 file=source_file)
593 original.lineno = before['old_lineno']
601 original.lineno = before['old_lineno']
594 original.content = before['line']
602 original.content = before['line']
595 original.action = self.action_to_op(before['action'])
603 original.action = self.action_to_op(before['action'])
596 original.comments = self.get_comments_for('old',
604 original.comments = self.get_comments_for('old',
597 source_file, before['old_lineno'])
605 source_file, before['old_lineno'])
598
606
599 if after:
607 if after:
600 if after['action'] == 'new-no-nl':
608 if after['action'] == 'new-no-nl':
601 after_tokens = [('nonl', after['line'])]
609 after_tokens = [('nonl', after['line'])]
602 else:
610 else:
603 after_tokens = self.get_line_tokens(
611 after_tokens = self.get_line_tokens(
604 line_text=after['line'], line_number=after['new_lineno'],
612 line_text=after['line'], line_number=after['new_lineno'],
605 file=target_file)
613 file=target_file)
606 modified.lineno = after['new_lineno']
614 modified.lineno = after['new_lineno']
607 modified.content = after['line']
615 modified.content = after['line']
608 modified.action = self.action_to_op(after['action'])
616 modified.action = self.action_to_op(after['action'])
609 modified.comments = self.get_comments_for('new',
617 modified.comments = self.get_comments_for('new',
610 target_file, after['new_lineno'])
618 target_file, after['new_lineno'])
611
619
612 # diff the lines
620 # diff the lines
613 if before_tokens and after_tokens:
621 if before_tokens and after_tokens:
614 o_tokens, m_tokens, similarity = tokens_diff(
622 o_tokens, m_tokens, similarity = tokens_diff(
615 before_tokens, after_tokens)
623 before_tokens, after_tokens)
616 original.content = render_tokenstream(o_tokens)
624 original.content = render_tokenstream(o_tokens)
617 modified.content = render_tokenstream(m_tokens)
625 modified.content = render_tokenstream(m_tokens)
618 elif before_tokens:
626 elif before_tokens:
619 original.content = render_tokenstream(
627 original.content = render_tokenstream(
620 [(x[0], '', x[1]) for x in before_tokens])
628 [(x[0], '', x[1]) for x in before_tokens])
621 elif after_tokens:
629 elif after_tokens:
622 modified.content = render_tokenstream(
630 modified.content = render_tokenstream(
623 [(x[0], '', x[1]) for x in after_tokens])
631 [(x[0], '', x[1]) for x in after_tokens])
624
632
625 if not before_lines and before_newline:
633 if not before_lines and before_newline:
626 original.content += before_newline.content
634 original.content += before_newline.content
627 before_newline = None
635 before_newline = None
628 if not after_lines and after_newline:
636 if not after_lines and after_newline:
629 modified.content += after_newline.content
637 modified.content += after_newline.content
630 after_newline = None
638 after_newline = None
631
639
632 lines.append(AttributeDict({
640 lines.append(AttributeDict({
633 'original': original,
641 'original': original,
634 'modified': modified,
642 'modified': modified,
635 }))
643 }))
636
644
637 return lines
645 return lines
638
646
639 def get_comments_for(self, version, filename, line_number):
647 def get_comments_for(self, version, filename, line_number):
640 if hasattr(filename, 'unicode_path'):
648 if hasattr(filename, 'unicode_path'):
641 filename = filename.unicode_path
649 filename = filename.unicode_path
642
650
643 if not isinstance(filename, basestring):
651 if not isinstance(filename, basestring):
644 return None
652 return None
645
653
646 line_key = {
654 line_key = {
647 'old': 'o',
655 'old': 'o',
648 'new': 'n',
656 'new': 'n',
649 }[version] + str(line_number)
657 }[version] + str(line_number)
650
658
651 if filename in self.comments_store:
659 if filename in self.comments_store:
652 file_comments = self.comments_store[filename]
660 file_comments = self.comments_store[filename]
653 if line_key in file_comments:
661 if line_key in file_comments:
654 return file_comments.pop(line_key)
662 return file_comments.pop(line_key)
655
663
656 def get_line_tokens(self, line_text, line_number, file=None):
664 def get_line_tokens(self, line_text, line_number, file=None):
657 filenode = None
665 filenode = None
658 filename = None
666 filename = None
659
667
660 if isinstance(file, basestring):
668 if isinstance(file, basestring):
661 filename = file
669 filename = file
662 elif isinstance(file, FileNode):
670 elif isinstance(file, FileNode):
663 filenode = file
671 filenode = file
664 filename = file.unicode_path
672 filename = file.unicode_path
665
673
666 if self.highlight_mode == self.HL_REAL and filenode:
674 if self.highlight_mode == self.HL_REAL and filenode:
667 lexer = self._get_lexer_for_filename(filename)
675 lexer = self._get_lexer_for_filename(filename)
668 file_size_allowed = file.size < self.max_file_size_limit
676 file_size_allowed = file.size < self.max_file_size_limit
669 if line_number and file_size_allowed:
677 if line_number and file_size_allowed:
670 return self.get_tokenized_filenode_line(
678 return self.get_tokenized_filenode_line(
671 file, line_number, lexer)
679 file, line_number, lexer)
672
680
673 if self.highlight_mode in (self.HL_REAL, self.HL_FAST) and filename:
681 if self.highlight_mode in (self.HL_REAL, self.HL_FAST) and filename:
674 lexer = self._get_lexer_for_filename(filename)
682 lexer = self._get_lexer_for_filename(filename)
675 return list(tokenize_string(line_text, lexer))
683 return list(tokenize_string(line_text, lexer))
676
684
677 return list(tokenize_string(line_text, plain_text_lexer))
685 return list(tokenize_string(line_text, plain_text_lexer))
678
686
679 def get_tokenized_filenode_line(self, filenode, line_number, lexer=None):
687 def get_tokenized_filenode_line(self, filenode, line_number, lexer=None):
680
688
681 if filenode not in self.highlighted_filenodes:
689 if filenode not in self.highlighted_filenodes:
682 tokenized_lines = filenode_as_lines_tokens(filenode, lexer)
690 tokenized_lines = filenode_as_lines_tokens(filenode, lexer)
683 self.highlighted_filenodes[filenode] = tokenized_lines
691 self.highlighted_filenodes[filenode] = tokenized_lines
684 return self.highlighted_filenodes[filenode][line_number - 1]
692 return self.highlighted_filenodes[filenode][line_number - 1]
685
693
686 def action_to_op(self, action):
694 def action_to_op(self, action):
687 return {
695 return {
688 'add': '+',
696 'add': '+',
689 'del': '-',
697 'del': '-',
690 'unmod': ' ',
698 'unmod': ' ',
691 'old-no-nl': ' ',
699 'old-no-nl': ' ',
692 'new-no-nl': ' ',
700 'new-no-nl': ' ',
693 }.get(action, action)
701 }.get(action, action)
694
702
695 def as_unified(self, lines):
703 def as_unified(self, lines):
696 """
704 """
697 Return a generator that yields the lines of a diff in unified order
705 Return a generator that yields the lines of a diff in unified order
698 """
706 """
699 def generator():
707 def generator():
700 buf = []
708 buf = []
701 for line in lines:
709 for line in lines:
702
710
703 if buf and not line.original or line.original.action == ' ':
711 if buf and not line.original or line.original.action == ' ':
704 for b in buf:
712 for b in buf:
705 yield b
713 yield b
706 buf = []
714 buf = []
707
715
708 if line.original:
716 if line.original:
709 if line.original.action == ' ':
717 if line.original.action == ' ':
710 yield (line.original.lineno, line.modified.lineno,
718 yield (line.original.lineno, line.modified.lineno,
711 line.original.action, line.original.content,
719 line.original.action, line.original.content,
712 line.original.comments)
720 line.original.comments)
713 continue
721 continue
714
722
715 if line.original.action == '-':
723 if line.original.action == '-':
716 yield (line.original.lineno, None,
724 yield (line.original.lineno, None,
717 line.original.action, line.original.content,
725 line.original.action, line.original.content,
718 line.original.comments)
726 line.original.comments)
719
727
720 if line.modified.action == '+':
728 if line.modified.action == '+':
721 buf.append((
729 buf.append((
722 None, line.modified.lineno,
730 None, line.modified.lineno,
723 line.modified.action, line.modified.content,
731 line.modified.action, line.modified.content,
724 line.modified.comments))
732 line.modified.comments))
725 continue
733 continue
726
734
727 if line.modified:
735 if line.modified:
728 yield (None, line.modified.lineno,
736 yield (None, line.modified.lineno,
729 line.modified.action, line.modified.content,
737 line.modified.action, line.modified.content,
730 line.modified.comments)
738 line.modified.comments)
731
739
732 for b in buf:
740 for b in buf:
733 yield b
741 yield b
734
742
735 return generator()
743 return generator()
@@ -1,1107 +1,1131
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2011-2018 RhodeCode GmbH
3 # Copyright (C) 2011-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21
21
22 """
22 """
23 Set of diffing helpers, previously part of vcs
23 Set of diffing helpers, previously part of vcs
24 """
24 """
25
25
26 import re
26 import re
27 import collections
27 import collections
28 import difflib
28 import difflib
29 import logging
29 import logging
30
30
31 from itertools import tee, imap
31 from itertools import tee, imap
32
32
33 from rhodecode.lib.vcs.exceptions import VCSError
33 from rhodecode.lib.vcs.exceptions import VCSError
34 from rhodecode.lib.vcs.nodes import FileNode, SubModuleNode
34 from rhodecode.lib.vcs.nodes import FileNode, SubModuleNode
35 from rhodecode.lib.utils2 import safe_unicode
35 from rhodecode.lib.utils2 import safe_unicode
36
36
37 log = logging.getLogger(__name__)
37 log = logging.getLogger(__name__)
38
38
39 # define max context, a file with more than this numbers of lines is unusable
39 # define max context, a file with more than this numbers of lines is unusable
40 # in browser anyway
40 # in browser anyway
41 MAX_CONTEXT = 1024 * 1014
41 MAX_CONTEXT = 1024 * 1014
42
42
43
43
44 class OPS(object):
44 class OPS(object):
45 ADD = 'A'
45 ADD = 'A'
46 MOD = 'M'
46 MOD = 'M'
47 DEL = 'D'
47 DEL = 'D'
48
48
49
49
50 def get_gitdiff(filenode_old, filenode_new, ignore_whitespace=True, context=3):
50 def get_gitdiff(filenode_old, filenode_new, ignore_whitespace=True, context=3):
51 """
51 """
52 Returns git style diff between given ``filenode_old`` and ``filenode_new``.
52 Returns git style diff between given ``filenode_old`` and ``filenode_new``.
53
53
54 :param ignore_whitespace: ignore whitespaces in diff
54 :param ignore_whitespace: ignore whitespaces in diff
55 """
55 """
56 # make sure we pass in default context
56 # make sure we pass in default context
57 context = context or 3
57 context = context or 3
58 # protect against IntOverflow when passing HUGE context
58 # protect against IntOverflow when passing HUGE context
59 if context > MAX_CONTEXT:
59 if context > MAX_CONTEXT:
60 context = MAX_CONTEXT
60 context = MAX_CONTEXT
61
61
62 submodules = filter(lambda o: isinstance(o, SubModuleNode),
62 submodules = filter(lambda o: isinstance(o, SubModuleNode),
63 [filenode_new, filenode_old])
63 [filenode_new, filenode_old])
64 if submodules:
64 if submodules:
65 return ''
65 return ''
66
66
67 for filenode in (filenode_old, filenode_new):
67 for filenode in (filenode_old, filenode_new):
68 if not isinstance(filenode, FileNode):
68 if not isinstance(filenode, FileNode):
69 raise VCSError(
69 raise VCSError(
70 "Given object should be FileNode object, not %s"
70 "Given object should be FileNode object, not %s"
71 % filenode.__class__)
71 % filenode.__class__)
72
72
73 repo = filenode_new.commit.repository
73 repo = filenode_new.commit.repository
74 old_commit = filenode_old.commit or repo.EMPTY_COMMIT
74 old_commit = filenode_old.commit or repo.EMPTY_COMMIT
75 new_commit = filenode_new.commit
75 new_commit = filenode_new.commit
76
76
77 vcs_gitdiff = repo.get_diff(
77 vcs_gitdiff = repo.get_diff(
78 old_commit, new_commit, filenode_new.path,
78 old_commit, new_commit, filenode_new.path,
79 ignore_whitespace, context, path1=filenode_old.path)
79 ignore_whitespace, context, path1=filenode_old.path)
80 return vcs_gitdiff
80 return vcs_gitdiff
81
81
82 NEW_FILENODE = 1
82 NEW_FILENODE = 1
83 DEL_FILENODE = 2
83 DEL_FILENODE = 2
84 MOD_FILENODE = 3
84 MOD_FILENODE = 3
85 RENAMED_FILENODE = 4
85 RENAMED_FILENODE = 4
86 COPIED_FILENODE = 5
86 COPIED_FILENODE = 5
87 CHMOD_FILENODE = 6
87 CHMOD_FILENODE = 6
88 BIN_FILENODE = 7
88 BIN_FILENODE = 7
89
89
90
90
91 class LimitedDiffContainer(object):
91 class LimitedDiffContainer(object):
92
92
93 def __init__(self, diff_limit, cur_diff_size, diff):
93 def __init__(self, diff_limit, cur_diff_size, diff):
94 self.diff = diff
94 self.diff = diff
95 self.diff_limit = diff_limit
95 self.diff_limit = diff_limit
96 self.cur_diff_size = cur_diff_size
96 self.cur_diff_size = cur_diff_size
97
97
98 def __getitem__(self, key):
98 def __getitem__(self, key):
99 return self.diff.__getitem__(key)
99 return self.diff.__getitem__(key)
100
100
101 def __iter__(self):
101 def __iter__(self):
102 for l in self.diff:
102 for l in self.diff:
103 yield l
103 yield l
104
104
105
105
106 class Action(object):
106 class Action(object):
107 """
107 """
108 Contains constants for the action value of the lines in a parsed diff.
108 Contains constants for the action value of the lines in a parsed diff.
109 """
109 """
110
110
111 ADD = 'add'
111 ADD = 'add'
112 DELETE = 'del'
112 DELETE = 'del'
113 UNMODIFIED = 'unmod'
113 UNMODIFIED = 'unmod'
114
114
115 CONTEXT = 'context'
115 CONTEXT = 'context'
116 OLD_NO_NL = 'old-no-nl'
116 OLD_NO_NL = 'old-no-nl'
117 NEW_NO_NL = 'new-no-nl'
117 NEW_NO_NL = 'new-no-nl'
118
118
119
119
120 class DiffProcessor(object):
120 class DiffProcessor(object):
121 """
121 """
122 Give it a unified or git diff and it returns a list of the files that were
122 Give it a unified or git diff and it returns a list of the files that were
123 mentioned in the diff together with a dict of meta information that
123 mentioned in the diff together with a dict of meta information that
124 can be used to render it in a HTML template.
124 can be used to render it in a HTML template.
125
125
126 .. note:: Unicode handling
126 .. note:: Unicode handling
127
127
128 The original diffs are a byte sequence and can contain filenames
128 The original diffs are a byte sequence and can contain filenames
129 in mixed encodings. This class generally returns `unicode` objects
129 in mixed encodings. This class generally returns `unicode` objects
130 since the result is intended for presentation to the user.
130 since the result is intended for presentation to the user.
131
131
132 """
132 """
133 _chunk_re = re.compile(r'^@@ -(\d+)(?:,(\d+))? \+(\d+)(?:,(\d+))? @@(.*)')
133 _chunk_re = re.compile(r'^@@ -(\d+)(?:,(\d+))? \+(\d+)(?:,(\d+))? @@(.*)')
134 _newline_marker = re.compile(r'^\\ No newline at end of file')
134 _newline_marker = re.compile(r'^\\ No newline at end of file')
135
135
136 # used for inline highlighter word split
136 # used for inline highlighter word split
137 _token_re = re.compile(r'()(&gt;|&lt;|&amp;|\W+?)')
137 _token_re = re.compile(r'()(&gt;|&lt;|&amp;|\W+?)')
138
138
139 # collapse ranges of commits over given number
139 # collapse ranges of commits over given number
140 _collapse_commits_over = 5
140 _collapse_commits_over = 5
141
141
142 def __init__(self, diff, format='gitdiff', diff_limit=None,
142 def __init__(self, diff, format='gitdiff', diff_limit=None,
143 file_limit=None, show_full_diff=True):
143 file_limit=None, show_full_diff=True):
144 """
144 """
145 :param diff: A `Diff` object representing a diff from a vcs backend
145 :param diff: A `Diff` object representing a diff from a vcs backend
146 :param format: format of diff passed, `udiff` or `gitdiff`
146 :param format: format of diff passed, `udiff` or `gitdiff`
147 :param diff_limit: define the size of diff that is considered "big"
147 :param diff_limit: define the size of diff that is considered "big"
148 based on that parameter cut off will be triggered, set to None
148 based on that parameter cut off will be triggered, set to None
149 to show full diff
149 to show full diff
150 """
150 """
151 self._diff = diff
151 self._diff = diff
152 self._format = format
152 self._format = format
153 self.adds = 0
153 self.adds = 0
154 self.removes = 0
154 self.removes = 0
155 # calculate diff size
155 # calculate diff size
156 self.diff_limit = diff_limit
156 self.diff_limit = diff_limit
157 self.file_limit = file_limit
157 self.file_limit = file_limit
158 self.show_full_diff = show_full_diff
158 self.show_full_diff = show_full_diff
159 self.cur_diff_size = 0
159 self.cur_diff_size = 0
160 self.parsed = False
160 self.parsed = False
161 self.parsed_diff = []
161 self.parsed_diff = []
162
162
163 log.debug('Initialized DiffProcessor with %s mode', format)
163 log.debug('Initialized DiffProcessor with %s mode', format)
164 if format == 'gitdiff':
164 if format == 'gitdiff':
165 self.differ = self._highlight_line_difflib
165 self.differ = self._highlight_line_difflib
166 self._parser = self._parse_gitdiff
166 self._parser = self._parse_gitdiff
167 else:
167 else:
168 self.differ = self._highlight_line_udiff
168 self.differ = self._highlight_line_udiff
169 self._parser = self._new_parse_gitdiff
169 self._parser = self._new_parse_gitdiff
170
170
171 def _copy_iterator(self):
171 def _copy_iterator(self):
172 """
172 """
173 make a fresh copy of generator, we should not iterate thru
173 make a fresh copy of generator, we should not iterate thru
174 an original as it's needed for repeating operations on
174 an original as it's needed for repeating operations on
175 this instance of DiffProcessor
175 this instance of DiffProcessor
176 """
176 """
177 self.__udiff, iterator_copy = tee(self.__udiff)
177 self.__udiff, iterator_copy = tee(self.__udiff)
178 return iterator_copy
178 return iterator_copy
179
179
180 def _escaper(self, string):
180 def _escaper(self, string):
181 """
181 """
182 Escaper for diff escapes special chars and checks the diff limit
182 Escaper for diff escapes special chars and checks the diff limit
183
183
184 :param string:
184 :param string:
185 """
185 """
186
187 self.cur_diff_size += len(string)
186 self.cur_diff_size += len(string)
188
187
189 if not self.show_full_diff and (self.cur_diff_size > self.diff_limit):
188 if not self.show_full_diff and (self.cur_diff_size > self.diff_limit):
190 raise DiffLimitExceeded('Diff Limit Exceeded')
189 raise DiffLimitExceeded('Diff Limit Exceeded')
191
190
192 return safe_unicode(string)\
191 return string \
193 .replace('&', '&amp;')\
192 .replace('&', '&amp;')\
194 .replace('<', '&lt;')\
193 .replace('<', '&lt;')\
195 .replace('>', '&gt;')
194 .replace('>', '&gt;')
196
195
197 def _line_counter(self, l):
196 def _line_counter(self, l):
198 """
197 """
199 Checks each line and bumps total adds/removes for this diff
198 Checks each line and bumps total adds/removes for this diff
200
199
201 :param l:
200 :param l:
202 """
201 """
203 if l.startswith('+') and not l.startswith('+++'):
202 if l.startswith('+') and not l.startswith('+++'):
204 self.adds += 1
203 self.adds += 1
205 elif l.startswith('-') and not l.startswith('---'):
204 elif l.startswith('-') and not l.startswith('---'):
206 self.removes += 1
205 self.removes += 1
207 return safe_unicode(l)
206 return safe_unicode(l)
208
207
209 def _highlight_line_difflib(self, line, next_):
208 def _highlight_line_difflib(self, line, next_):
210 """
209 """
211 Highlight inline changes in both lines.
210 Highlight inline changes in both lines.
212 """
211 """
213
212
214 if line['action'] == Action.DELETE:
213 if line['action'] == Action.DELETE:
215 old, new = line, next_
214 old, new = line, next_
216 else:
215 else:
217 old, new = next_, line
216 old, new = next_, line
218
217
219 oldwords = self._token_re.split(old['line'])
218 oldwords = self._token_re.split(old['line'])
220 newwords = self._token_re.split(new['line'])
219 newwords = self._token_re.split(new['line'])
221 sequence = difflib.SequenceMatcher(None, oldwords, newwords)
220 sequence = difflib.SequenceMatcher(None, oldwords, newwords)
222
221
223 oldfragments, newfragments = [], []
222 oldfragments, newfragments = [], []
224 for tag, i1, i2, j1, j2 in sequence.get_opcodes():
223 for tag, i1, i2, j1, j2 in sequence.get_opcodes():
225 oldfrag = ''.join(oldwords[i1:i2])
224 oldfrag = ''.join(oldwords[i1:i2])
226 newfrag = ''.join(newwords[j1:j2])
225 newfrag = ''.join(newwords[j1:j2])
227 if tag != 'equal':
226 if tag != 'equal':
228 if oldfrag:
227 if oldfrag:
229 oldfrag = '<del>%s</del>' % oldfrag
228 oldfrag = '<del>%s</del>' % oldfrag
230 if newfrag:
229 if newfrag:
231 newfrag = '<ins>%s</ins>' % newfrag
230 newfrag = '<ins>%s</ins>' % newfrag
232 oldfragments.append(oldfrag)
231 oldfragments.append(oldfrag)
233 newfragments.append(newfrag)
232 newfragments.append(newfrag)
234
233
235 old['line'] = "".join(oldfragments)
234 old['line'] = "".join(oldfragments)
236 new['line'] = "".join(newfragments)
235 new['line'] = "".join(newfragments)
237
236
238 def _highlight_line_udiff(self, line, next_):
237 def _highlight_line_udiff(self, line, next_):
239 """
238 """
240 Highlight inline changes in both lines.
239 Highlight inline changes in both lines.
241 """
240 """
242 start = 0
241 start = 0
243 limit = min(len(line['line']), len(next_['line']))
242 limit = min(len(line['line']), len(next_['line']))
244 while start < limit and line['line'][start] == next_['line'][start]:
243 while start < limit and line['line'][start] == next_['line'][start]:
245 start += 1
244 start += 1
246 end = -1
245 end = -1
247 limit -= start
246 limit -= start
248 while -end <= limit and line['line'][end] == next_['line'][end]:
247 while -end <= limit and line['line'][end] == next_['line'][end]:
249 end -= 1
248 end -= 1
250 end += 1
249 end += 1
251 if start or end:
250 if start or end:
252 def do(l):
251 def do(l):
253 last = end + len(l['line'])
252 last = end + len(l['line'])
254 if l['action'] == Action.ADD:
253 if l['action'] == Action.ADD:
255 tag = 'ins'
254 tag = 'ins'
256 else:
255 else:
257 tag = 'del'
256 tag = 'del'
258 l['line'] = '%s<%s>%s</%s>%s' % (
257 l['line'] = '%s<%s>%s</%s>%s' % (
259 l['line'][:start],
258 l['line'][:start],
260 tag,
259 tag,
261 l['line'][start:last],
260 l['line'][start:last],
262 tag,
261 tag,
263 l['line'][last:]
262 l['line'][last:]
264 )
263 )
265 do(line)
264 do(line)
266 do(next_)
265 do(next_)
267
266
268 def _clean_line(self, line, command):
267 def _clean_line(self, line, command):
269 if command in ['+', '-', ' ']:
268 if command in ['+', '-', ' ']:
270 # only modify the line if it's actually a diff thing
269 # only modify the line if it's actually a diff thing
271 line = line[1:]
270 line = line[1:]
272 return line
271 return line
273
272
274 def _parse_gitdiff(self, inline_diff=True):
273 def _parse_gitdiff(self, inline_diff=True):
275 _files = []
274 _files = []
276 diff_container = lambda arg: arg
275 diff_container = lambda arg: arg
277
276
278 for chunk in self._diff.chunks():
277 for chunk in self._diff.chunks():
279 head = chunk.header
278 head = chunk.header
280
279
281 diff = imap(self._escaper, chunk.diff.splitlines(1))
280 diff = imap(self._escaper, self.diff_splitter(chunk.diff))
282 raw_diff = chunk.raw
281 raw_diff = chunk.raw
283 limited_diff = False
282 limited_diff = False
284 exceeds_limit = False
283 exceeds_limit = False
285
284
286 op = None
285 op = None
287 stats = {
286 stats = {
288 'added': 0,
287 'added': 0,
289 'deleted': 0,
288 'deleted': 0,
290 'binary': False,
289 'binary': False,
291 'ops': {},
290 'ops': {},
292 }
291 }
293
292
294 if head['deleted_file_mode']:
293 if head['deleted_file_mode']:
295 op = OPS.DEL
294 op = OPS.DEL
296 stats['binary'] = True
295 stats['binary'] = True
297 stats['ops'][DEL_FILENODE] = 'deleted file'
296 stats['ops'][DEL_FILENODE] = 'deleted file'
298
297
299 elif head['new_file_mode']:
298 elif head['new_file_mode']:
300 op = OPS.ADD
299 op = OPS.ADD
301 stats['binary'] = True
300 stats['binary'] = True
302 stats['ops'][NEW_FILENODE] = 'new file %s' % head['new_file_mode']
301 stats['ops'][NEW_FILENODE] = 'new file %s' % head['new_file_mode']
303 else: # modify operation, can be copy, rename or chmod
302 else: # modify operation, can be copy, rename or chmod
304
303
305 # CHMOD
304 # CHMOD
306 if head['new_mode'] and head['old_mode']:
305 if head['new_mode'] and head['old_mode']:
307 op = OPS.MOD
306 op = OPS.MOD
308 stats['binary'] = True
307 stats['binary'] = True
309 stats['ops'][CHMOD_FILENODE] = (
308 stats['ops'][CHMOD_FILENODE] = (
310 'modified file chmod %s => %s' % (
309 'modified file chmod %s => %s' % (
311 head['old_mode'], head['new_mode']))
310 head['old_mode'], head['new_mode']))
312 # RENAME
311 # RENAME
313 if head['rename_from'] != head['rename_to']:
312 if head['rename_from'] != head['rename_to']:
314 op = OPS.MOD
313 op = OPS.MOD
315 stats['binary'] = True
314 stats['binary'] = True
316 stats['ops'][RENAMED_FILENODE] = (
315 stats['ops'][RENAMED_FILENODE] = (
317 'file renamed from %s to %s' % (
316 'file renamed from %s to %s' % (
318 head['rename_from'], head['rename_to']))
317 head['rename_from'], head['rename_to']))
319 # COPY
318 # COPY
320 if head.get('copy_from') and head.get('copy_to'):
319 if head.get('copy_from') and head.get('copy_to'):
321 op = OPS.MOD
320 op = OPS.MOD
322 stats['binary'] = True
321 stats['binary'] = True
323 stats['ops'][COPIED_FILENODE] = (
322 stats['ops'][COPIED_FILENODE] = (
324 'file copied from %s to %s' % (
323 'file copied from %s to %s' % (
325 head['copy_from'], head['copy_to']))
324 head['copy_from'], head['copy_to']))
326
325
327 # If our new parsed headers didn't match anything fallback to
326 # If our new parsed headers didn't match anything fallback to
328 # old style detection
327 # old style detection
329 if op is None:
328 if op is None:
330 if not head['a_file'] and head['b_file']:
329 if not head['a_file'] and head['b_file']:
331 op = OPS.ADD
330 op = OPS.ADD
332 stats['binary'] = True
331 stats['binary'] = True
333 stats['ops'][NEW_FILENODE] = 'new file'
332 stats['ops'][NEW_FILENODE] = 'new file'
334
333
335 elif head['a_file'] and not head['b_file']:
334 elif head['a_file'] and not head['b_file']:
336 op = OPS.DEL
335 op = OPS.DEL
337 stats['binary'] = True
336 stats['binary'] = True
338 stats['ops'][DEL_FILENODE] = 'deleted file'
337 stats['ops'][DEL_FILENODE] = 'deleted file'
339
338
340 # it's not ADD not DELETE
339 # it's not ADD not DELETE
341 if op is None:
340 if op is None:
342 op = OPS.MOD
341 op = OPS.MOD
343 stats['binary'] = True
342 stats['binary'] = True
344 stats['ops'][MOD_FILENODE] = 'modified file'
343 stats['ops'][MOD_FILENODE] = 'modified file'
345
344
346 # a real non-binary diff
345 # a real non-binary diff
347 if head['a_file'] or head['b_file']:
346 if head['a_file'] or head['b_file']:
348 try:
347 try:
349 raw_diff, chunks, _stats = self._parse_lines(diff)
348 raw_diff, chunks, _stats = self._parse_lines(diff)
350 stats['binary'] = False
349 stats['binary'] = False
351 stats['added'] = _stats[0]
350 stats['added'] = _stats[0]
352 stats['deleted'] = _stats[1]
351 stats['deleted'] = _stats[1]
353 # explicit mark that it's a modified file
352 # explicit mark that it's a modified file
354 if op == OPS.MOD:
353 if op == OPS.MOD:
355 stats['ops'][MOD_FILENODE] = 'modified file'
354 stats['ops'][MOD_FILENODE] = 'modified file'
356 exceeds_limit = len(raw_diff) > self.file_limit
355 exceeds_limit = len(raw_diff) > self.file_limit
357
356
358 # changed from _escaper function so we validate size of
357 # changed from _escaper function so we validate size of
359 # each file instead of the whole diff
358 # each file instead of the whole diff
360 # diff will hide big files but still show small ones
359 # diff will hide big files but still show small ones
361 # from my tests, big files are fairly safe to be parsed
360 # from my tests, big files are fairly safe to be parsed
362 # but the browser is the bottleneck
361 # but the browser is the bottleneck
363 if not self.show_full_diff and exceeds_limit:
362 if not self.show_full_diff and exceeds_limit:
364 raise DiffLimitExceeded('File Limit Exceeded')
363 raise DiffLimitExceeded('File Limit Exceeded')
365
364
366 except DiffLimitExceeded:
365 except DiffLimitExceeded:
367 diff_container = lambda _diff: \
366 diff_container = lambda _diff: \
368 LimitedDiffContainer(
367 LimitedDiffContainer(
369 self.diff_limit, self.cur_diff_size, _diff)
368 self.diff_limit, self.cur_diff_size, _diff)
370
369
371 exceeds_limit = len(raw_diff) > self.file_limit
370 exceeds_limit = len(raw_diff) > self.file_limit
372 limited_diff = True
371 limited_diff = True
373 chunks = []
372 chunks = []
374
373
375 else: # GIT format binary patch, or possibly empty diff
374 else: # GIT format binary patch, or possibly empty diff
376 if head['bin_patch']:
375 if head['bin_patch']:
377 # we have operation already extracted, but we mark simply
376 # we have operation already extracted, but we mark simply
378 # it's a diff we wont show for binary files
377 # it's a diff we wont show for binary files
379 stats['ops'][BIN_FILENODE] = 'binary diff hidden'
378 stats['ops'][BIN_FILENODE] = 'binary diff hidden'
380 chunks = []
379 chunks = []
381
380
382 if chunks and not self.show_full_diff and op == OPS.DEL:
381 if chunks and not self.show_full_diff and op == OPS.DEL:
383 # if not full diff mode show deleted file contents
382 # if not full diff mode show deleted file contents
384 # TODO: anderson: if the view is not too big, there is no way
383 # TODO: anderson: if the view is not too big, there is no way
385 # to see the content of the file
384 # to see the content of the file
386 chunks = []
385 chunks = []
387
386
388 chunks.insert(0, [{
387 chunks.insert(0, [{
389 'old_lineno': '',
388 'old_lineno': '',
390 'new_lineno': '',
389 'new_lineno': '',
391 'action': Action.CONTEXT,
390 'action': Action.CONTEXT,
392 'line': msg,
391 'line': msg,
393 } for _op, msg in stats['ops'].iteritems()
392 } for _op, msg in stats['ops'].iteritems()
394 if _op not in [MOD_FILENODE]])
393 if _op not in [MOD_FILENODE]])
395
394
396 _files.append({
395 _files.append({
397 'filename': safe_unicode(head['b_path']),
396 'filename': safe_unicode(head['b_path']),
398 'old_revision': head['a_blob_id'],
397 'old_revision': head['a_blob_id'],
399 'new_revision': head['b_blob_id'],
398 'new_revision': head['b_blob_id'],
400 'chunks': chunks,
399 'chunks': chunks,
401 'raw_diff': safe_unicode(raw_diff),
400 'raw_diff': safe_unicode(raw_diff),
402 'operation': op,
401 'operation': op,
403 'stats': stats,
402 'stats': stats,
404 'exceeds_limit': exceeds_limit,
403 'exceeds_limit': exceeds_limit,
405 'is_limited_diff': limited_diff,
404 'is_limited_diff': limited_diff,
406 })
405 })
407
406
408 sorter = lambda info: {OPS.ADD: 0, OPS.MOD: 1,
407 sorter = lambda info: {OPS.ADD: 0, OPS.MOD: 1,
409 OPS.DEL: 2}.get(info['operation'])
408 OPS.DEL: 2}.get(info['operation'])
410
409
411 if not inline_diff:
410 if not inline_diff:
412 return diff_container(sorted(_files, key=sorter))
411 return diff_container(sorted(_files, key=sorter))
413
412
414 # highlight inline changes
413 # highlight inline changes
415 for diff_data in _files:
414 for diff_data in _files:
416 for chunk in diff_data['chunks']:
415 for chunk in diff_data['chunks']:
417 lineiter = iter(chunk)
416 lineiter = iter(chunk)
418 try:
417 try:
419 while 1:
418 while 1:
420 line = lineiter.next()
419 line = lineiter.next()
421 if line['action'] not in (
420 if line['action'] not in (
422 Action.UNMODIFIED, Action.CONTEXT):
421 Action.UNMODIFIED, Action.CONTEXT):
423 nextline = lineiter.next()
422 nextline = lineiter.next()
424 if nextline['action'] in ['unmod', 'context'] or \
423 if nextline['action'] in ['unmod', 'context'] or \
425 nextline['action'] == line['action']:
424 nextline['action'] == line['action']:
426 continue
425 continue
427 self.differ(line, nextline)
426 self.differ(line, nextline)
428 except StopIteration:
427 except StopIteration:
429 pass
428 pass
430
429
431 return diff_container(sorted(_files, key=sorter))
430 return diff_container(sorted(_files, key=sorter))
432
431
433 def _check_large_diff(self):
432 def _check_large_diff(self):
434 log.debug('Diff exceeds current diff_limit of %s', self.diff_limit)
433 log.debug('Diff exceeds current diff_limit of %s', self.diff_limit)
435 if not self.show_full_diff and (self.cur_diff_size > self.diff_limit):
434 if not self.show_full_diff and (self.cur_diff_size > self.diff_limit):
436 raise DiffLimitExceeded('Diff Limit `%s` Exceeded', self.diff_limit)
435 raise DiffLimitExceeded('Diff Limit `%s` Exceeded', self.diff_limit)
437
436
438 # FIXME: NEWDIFFS: dan: this replaces _parse_gitdiff
437 # FIXME: NEWDIFFS: dan: this replaces _parse_gitdiff
439 def _new_parse_gitdiff(self, inline_diff=True):
438 def _new_parse_gitdiff(self, inline_diff=True):
440 _files = []
439 _files = []
441
440
442 # this can be overriden later to a LimitedDiffContainer type
441 # this can be overriden later to a LimitedDiffContainer type
443 diff_container = lambda arg: arg
442 diff_container = lambda arg: arg
444
443
445 for chunk in self._diff.chunks():
444 for chunk in self._diff.chunks():
446 head = chunk.header
445 head = chunk.header
447 log.debug('parsing diff %r' % head)
446 log.debug('parsing diff %r' % head)
448
447
449 raw_diff = chunk.raw
448 raw_diff = chunk.raw
450 limited_diff = False
449 limited_diff = False
451 exceeds_limit = False
450 exceeds_limit = False
452
451
453 op = None
452 op = None
454 stats = {
453 stats = {
455 'added': 0,
454 'added': 0,
456 'deleted': 0,
455 'deleted': 0,
457 'binary': False,
456 'binary': False,
458 'old_mode': None,
457 'old_mode': None,
459 'new_mode': None,
458 'new_mode': None,
460 'ops': {},
459 'ops': {},
461 }
460 }
462 if head['old_mode']:
461 if head['old_mode']:
463 stats['old_mode'] = head['old_mode']
462 stats['old_mode'] = head['old_mode']
464 if head['new_mode']:
463 if head['new_mode']:
465 stats['new_mode'] = head['new_mode']
464 stats['new_mode'] = head['new_mode']
466 if head['b_mode']:
465 if head['b_mode']:
467 stats['new_mode'] = head['b_mode']
466 stats['new_mode'] = head['b_mode']
468
467
469 # delete file
468 # delete file
470 if head['deleted_file_mode']:
469 if head['deleted_file_mode']:
471 op = OPS.DEL
470 op = OPS.DEL
472 stats['binary'] = True
471 stats['binary'] = True
473 stats['ops'][DEL_FILENODE] = 'deleted file'
472 stats['ops'][DEL_FILENODE] = 'deleted file'
474
473
475 # new file
474 # new file
476 elif head['new_file_mode']:
475 elif head['new_file_mode']:
477 op = OPS.ADD
476 op = OPS.ADD
478 stats['binary'] = True
477 stats['binary'] = True
479 stats['old_mode'] = None
478 stats['old_mode'] = None
480 stats['new_mode'] = head['new_file_mode']
479 stats['new_mode'] = head['new_file_mode']
481 stats['ops'][NEW_FILENODE] = 'new file %s' % head['new_file_mode']
480 stats['ops'][NEW_FILENODE] = 'new file %s' % head['new_file_mode']
482
481
483 # modify operation, can be copy, rename or chmod
482 # modify operation, can be copy, rename or chmod
484 else:
483 else:
485 # CHMOD
484 # CHMOD
486 if head['new_mode'] and head['old_mode']:
485 if head['new_mode'] and head['old_mode']:
487 op = OPS.MOD
486 op = OPS.MOD
488 stats['binary'] = True
487 stats['binary'] = True
489 stats['ops'][CHMOD_FILENODE] = (
488 stats['ops'][CHMOD_FILENODE] = (
490 'modified file chmod %s => %s' % (
489 'modified file chmod %s => %s' % (
491 head['old_mode'], head['new_mode']))
490 head['old_mode'], head['new_mode']))
492
491
493 # RENAME
492 # RENAME
494 if head['rename_from'] != head['rename_to']:
493 if head['rename_from'] != head['rename_to']:
495 op = OPS.MOD
494 op = OPS.MOD
496 stats['binary'] = True
495 stats['binary'] = True
497 stats['renamed'] = (head['rename_from'], head['rename_to'])
496 stats['renamed'] = (head['rename_from'], head['rename_to'])
498 stats['ops'][RENAMED_FILENODE] = (
497 stats['ops'][RENAMED_FILENODE] = (
499 'file renamed from %s to %s' % (
498 'file renamed from %s to %s' % (
500 head['rename_from'], head['rename_to']))
499 head['rename_from'], head['rename_to']))
501 # COPY
500 # COPY
502 if head.get('copy_from') and head.get('copy_to'):
501 if head.get('copy_from') and head.get('copy_to'):
503 op = OPS.MOD
502 op = OPS.MOD
504 stats['binary'] = True
503 stats['binary'] = True
505 stats['copied'] = (head['copy_from'], head['copy_to'])
504 stats['copied'] = (head['copy_from'], head['copy_to'])
506 stats['ops'][COPIED_FILENODE] = (
505 stats['ops'][COPIED_FILENODE] = (
507 'file copied from %s to %s' % (
506 'file copied from %s to %s' % (
508 head['copy_from'], head['copy_to']))
507 head['copy_from'], head['copy_to']))
509
508
510 # If our new parsed headers didn't match anything fallback to
509 # If our new parsed headers didn't match anything fallback to
511 # old style detection
510 # old style detection
512 if op is None:
511 if op is None:
513 if not head['a_file'] and head['b_file']:
512 if not head['a_file'] and head['b_file']:
514 op = OPS.ADD
513 op = OPS.ADD
515 stats['binary'] = True
514 stats['binary'] = True
516 stats['new_file'] = True
515 stats['new_file'] = True
517 stats['ops'][NEW_FILENODE] = 'new file'
516 stats['ops'][NEW_FILENODE] = 'new file'
518
517
519 elif head['a_file'] and not head['b_file']:
518 elif head['a_file'] and not head['b_file']:
520 op = OPS.DEL
519 op = OPS.DEL
521 stats['binary'] = True
520 stats['binary'] = True
522 stats['ops'][DEL_FILENODE] = 'deleted file'
521 stats['ops'][DEL_FILENODE] = 'deleted file'
523
522
524 # it's not ADD not DELETE
523 # it's not ADD not DELETE
525 if op is None:
524 if op is None:
526 op = OPS.MOD
525 op = OPS.MOD
527 stats['binary'] = True
526 stats['binary'] = True
528 stats['ops'][MOD_FILENODE] = 'modified file'
527 stats['ops'][MOD_FILENODE] = 'modified file'
529
528
530 # a real non-binary diff
529 # a real non-binary diff
531 if head['a_file'] or head['b_file']:
530 if head['a_file'] or head['b_file']:
532 diff = iter(chunk.diff.splitlines(1))
531 # simulate splitlines, so we keep the line end part
532 diff = self.diff_splitter(chunk.diff)
533
533
534 # append each file to the diff size
534 # append each file to the diff size
535 raw_chunk_size = len(raw_diff)
535 raw_chunk_size = len(raw_diff)
536
536
537 exceeds_limit = raw_chunk_size > self.file_limit
537 exceeds_limit = raw_chunk_size > self.file_limit
538 self.cur_diff_size += raw_chunk_size
538 self.cur_diff_size += raw_chunk_size
539
539
540 try:
540 try:
541 # Check each file instead of the whole diff.
541 # Check each file instead of the whole diff.
542 # Diff will hide big files but still show small ones.
542 # Diff will hide big files but still show small ones.
543 # From the tests big files are fairly safe to be parsed
543 # From the tests big files are fairly safe to be parsed
544 # but the browser is the bottleneck.
544 # but the browser is the bottleneck.
545 if not self.show_full_diff and exceeds_limit:
545 if not self.show_full_diff and exceeds_limit:
546 log.debug('File `%s` exceeds current file_limit of %s',
546 log.debug('File `%s` exceeds current file_limit of %s',
547 safe_unicode(head['b_path']), self.file_limit)
547 safe_unicode(head['b_path']), self.file_limit)
548 raise DiffLimitExceeded(
548 raise DiffLimitExceeded(
549 'File Limit %s Exceeded', self.file_limit)
549 'File Limit %s Exceeded', self.file_limit)
550
550
551 self._check_large_diff()
551 self._check_large_diff()
552
552
553 raw_diff, chunks, _stats = self._new_parse_lines(diff)
553 raw_diff, chunks, _stats = self._new_parse_lines(diff)
554 stats['binary'] = False
554 stats['binary'] = False
555 stats['added'] = _stats[0]
555 stats['added'] = _stats[0]
556 stats['deleted'] = _stats[1]
556 stats['deleted'] = _stats[1]
557 # explicit mark that it's a modified file
557 # explicit mark that it's a modified file
558 if op == OPS.MOD:
558 if op == OPS.MOD:
559 stats['ops'][MOD_FILENODE] = 'modified file'
559 stats['ops'][MOD_FILENODE] = 'modified file'
560
560
561 except DiffLimitExceeded:
561 except DiffLimitExceeded:
562 diff_container = lambda _diff: \
562 diff_container = lambda _diff: \
563 LimitedDiffContainer(
563 LimitedDiffContainer(
564 self.diff_limit, self.cur_diff_size, _diff)
564 self.diff_limit, self.cur_diff_size, _diff)
565
565
566 limited_diff = True
566 limited_diff = True
567 chunks = []
567 chunks = []
568
568
569 else: # GIT format binary patch, or possibly empty diff
569 else: # GIT format binary patch, or possibly empty diff
570 if head['bin_patch']:
570 if head['bin_patch']:
571 # we have operation already extracted, but we mark simply
571 # we have operation already extracted, but we mark simply
572 # it's a diff we wont show for binary files
572 # it's a diff we wont show for binary files
573 stats['ops'][BIN_FILENODE] = 'binary diff hidden'
573 stats['ops'][BIN_FILENODE] = 'binary diff hidden'
574 chunks = []
574 chunks = []
575
575
576 # Hide content of deleted node by setting empty chunks
576 # Hide content of deleted node by setting empty chunks
577 if chunks and not self.show_full_diff and op == OPS.DEL:
577 if chunks and not self.show_full_diff and op == OPS.DEL:
578 # if not full diff mode show deleted file contents
578 # if not full diff mode show deleted file contents
579 # TODO: anderson: if the view is not too big, there is no way
579 # TODO: anderson: if the view is not too big, there is no way
580 # to see the content of the file
580 # to see the content of the file
581 chunks = []
581 chunks = []
582
582
583 chunks.insert(
583 chunks.insert(
584 0, [{'old_lineno': '',
584 0, [{'old_lineno': '',
585 'new_lineno': '',
585 'new_lineno': '',
586 'action': Action.CONTEXT,
586 'action': Action.CONTEXT,
587 'line': msg,
587 'line': msg,
588 } for _op, msg in stats['ops'].iteritems()
588 } for _op, msg in stats['ops'].iteritems()
589 if _op not in [MOD_FILENODE]])
589 if _op not in [MOD_FILENODE]])
590
590
591 original_filename = safe_unicode(head['a_path'])
591 original_filename = safe_unicode(head['a_path'])
592 _files.append({
592 _files.append({
593 'original_filename': original_filename,
593 'original_filename': original_filename,
594 'filename': safe_unicode(head['b_path']),
594 'filename': safe_unicode(head['b_path']),
595 'old_revision': head['a_blob_id'],
595 'old_revision': head['a_blob_id'],
596 'new_revision': head['b_blob_id'],
596 'new_revision': head['b_blob_id'],
597 'chunks': chunks,
597 'chunks': chunks,
598 'raw_diff': safe_unicode(raw_diff),
598 'raw_diff': safe_unicode(raw_diff),
599 'operation': op,
599 'operation': op,
600 'stats': stats,
600 'stats': stats,
601 'exceeds_limit': exceeds_limit,
601 'exceeds_limit': exceeds_limit,
602 'is_limited_diff': limited_diff,
602 'is_limited_diff': limited_diff,
603 })
603 })
604
604
605 sorter = lambda info: {OPS.ADD: 0, OPS.MOD: 1,
605 sorter = lambda info: {OPS.ADD: 0, OPS.MOD: 1,
606 OPS.DEL: 2}.get(info['operation'])
606 OPS.DEL: 2}.get(info['operation'])
607
607
608 return diff_container(sorted(_files, key=sorter))
608 return diff_container(sorted(_files, key=sorter))
609
609
610 # FIXME: NEWDIFFS: dan: this gets replaced by _new_parse_lines
610 # FIXME: NEWDIFFS: dan: this gets replaced by _new_parse_lines
611 def _parse_lines(self, diff):
611 def _parse_lines(self, diff_iter):
612 """
612 """
613 Parse the diff an return data for the template.
613 Parse the diff an return data for the template.
614 """
614 """
615
615
616 lineiter = iter(diff)
617 stats = [0, 0]
616 stats = [0, 0]
618 chunks = []
617 chunks = []
619 raw_diff = []
618 raw_diff = []
620
619
621 try:
620 try:
622 line = lineiter.next()
621 line = diff_iter.next()
623
622
624 while line:
623 while line:
625 raw_diff.append(line)
624 raw_diff.append(line)
626 lines = []
625 lines = []
627 chunks.append(lines)
626 chunks.append(lines)
628
627
629 match = self._chunk_re.match(line)
628 match = self._chunk_re.match(line)
630
629
631 if not match:
630 if not match:
632 break
631 break
633
632
634 gr = match.groups()
633 gr = match.groups()
635 (old_line, old_end,
634 (old_line, old_end,
636 new_line, new_end) = [int(x or 1) for x in gr[:-1]]
635 new_line, new_end) = [int(x or 1) for x in gr[:-1]]
637 old_line -= 1
636 old_line -= 1
638 new_line -= 1
637 new_line -= 1
639
638
640 context = len(gr) == 5
639 context = len(gr) == 5
641 old_end += old_line
640 old_end += old_line
642 new_end += new_line
641 new_end += new_line
643
642
644 if context:
643 if context:
645 # skip context only if it's first line
644 # skip context only if it's first line
646 if int(gr[0]) > 1:
645 if int(gr[0]) > 1:
647 lines.append({
646 lines.append({
648 'old_lineno': '...',
647 'old_lineno': '...',
649 'new_lineno': '...',
648 'new_lineno': '...',
650 'action': Action.CONTEXT,
649 'action': Action.CONTEXT,
651 'line': line,
650 'line': line,
652 })
651 })
653
652
654 line = lineiter.next()
653 line = diff_iter.next()
655
654
656 while old_line < old_end or new_line < new_end:
655 while old_line < old_end or new_line < new_end:
657 command = ' '
656 command = ' '
658 if line:
657 if line:
659 command = line[0]
658 command = line[0]
660
659
661 affects_old = affects_new = False
660 affects_old = affects_new = False
662
661
663 # ignore those if we don't expect them
662 # ignore those if we don't expect them
664 if command in '#@':
663 if command in '#@':
665 continue
664 continue
666 elif command == '+':
665 elif command == '+':
667 affects_new = True
666 affects_new = True
668 action = Action.ADD
667 action = Action.ADD
669 stats[0] += 1
668 stats[0] += 1
670 elif command == '-':
669 elif command == '-':
671 affects_old = True
670 affects_old = True
672 action = Action.DELETE
671 action = Action.DELETE
673 stats[1] += 1
672 stats[1] += 1
674 else:
673 else:
675 affects_old = affects_new = True
674 affects_old = affects_new = True
676 action = Action.UNMODIFIED
675 action = Action.UNMODIFIED
677
676
678 if not self._newline_marker.match(line):
677 if not self._newline_marker.match(line):
679 old_line += affects_old
678 old_line += affects_old
680 new_line += affects_new
679 new_line += affects_new
681 lines.append({
680 lines.append({
682 'old_lineno': affects_old and old_line or '',
681 'old_lineno': affects_old and old_line or '',
683 'new_lineno': affects_new and new_line or '',
682 'new_lineno': affects_new and new_line or '',
684 'action': action,
683 'action': action,
685 'line': self._clean_line(line, command)
684 'line': self._clean_line(line, command)
686 })
685 })
687 raw_diff.append(line)
686 raw_diff.append(line)
688
687
689 line = lineiter.next()
688 line = diff_iter.next()
690
689
691 if self._newline_marker.match(line):
690 if self._newline_marker.match(line):
692 # we need to append to lines, since this is not
691 # we need to append to lines, since this is not
693 # counted in the line specs of diff
692 # counted in the line specs of diff
694 lines.append({
693 lines.append({
695 'old_lineno': '...',
694 'old_lineno': '...',
696 'new_lineno': '...',
695 'new_lineno': '...',
697 'action': Action.CONTEXT,
696 'action': Action.CONTEXT,
698 'line': self._clean_line(line, command)
697 'line': self._clean_line(line, command)
699 })
698 })
700
699
701 except StopIteration:
700 except StopIteration:
702 pass
701 pass
703 return ''.join(raw_diff), chunks, stats
702 return ''.join(raw_diff), chunks, stats
704
703
705 # FIXME: NEWDIFFS: dan: this replaces _parse_lines
704 # FIXME: NEWDIFFS: dan: this replaces _parse_lines
706 def _new_parse_lines(self, diff_iter):
705 def _new_parse_lines(self, diff_iter):
707 """
706 """
708 Parse the diff an return data for the template.
707 Parse the diff an return data for the template.
709 """
708 """
710
709
711 stats = [0, 0]
710 stats = [0, 0]
712 chunks = []
711 chunks = []
713 raw_diff = []
712 raw_diff = []
714
713
715 diff_iter = imap(lambda s: safe_unicode(s), diff_iter)
716
717 try:
714 try:
718 line = diff_iter.next()
715 line = diff_iter.next()
719
716
720 while line:
717 while line:
721 raw_diff.append(line)
718 raw_diff.append(line)
719 # match header e.g @@ -0,0 +1 @@\n'
722 match = self._chunk_re.match(line)
720 match = self._chunk_re.match(line)
723
721
724 if not match:
722 if not match:
725 break
723 break
726
724
727 gr = match.groups()
725 gr = match.groups()
728 (old_line, old_end,
726 (old_line, old_end,
729 new_line, new_end) = [int(x or 1) for x in gr[:-1]]
727 new_line, new_end) = [int(x or 1) for x in gr[:-1]]
730
728
731 lines = []
729 lines = []
732 hunk = {
730 hunk = {
733 'section_header': gr[-1],
731 'section_header': gr[-1],
734 'source_start': old_line,
732 'source_start': old_line,
735 'source_length': old_end,
733 'source_length': old_end,
736 'target_start': new_line,
734 'target_start': new_line,
737 'target_length': new_end,
735 'target_length': new_end,
738 'lines': lines,
736 'lines': lines,
739 }
737 }
740 chunks.append(hunk)
738 chunks.append(hunk)
741
739
742 old_line -= 1
740 old_line -= 1
743 new_line -= 1
741 new_line -= 1
744
742
745 context = len(gr) == 5
743 context = len(gr) == 5
746 old_end += old_line
744 old_end += old_line
747 new_end += new_line
745 new_end += new_line
748
746
749 line = diff_iter.next()
747 line = diff_iter.next()
750
748
751 while old_line < old_end or new_line < new_end:
749 while old_line < old_end or new_line < new_end:
752 command = ' '
750 command = ' '
753 if line:
751 if line:
754 command = line[0]
752 command = line[0]
755
753
756 affects_old = affects_new = False
754 affects_old = affects_new = False
757
755
758 # ignore those if we don't expect them
756 # ignore those if we don't expect them
759 if command in '#@':
757 if command in '#@':
760 continue
758 continue
761 elif command == '+':
759 elif command == '+':
762 affects_new = True
760 affects_new = True
763 action = Action.ADD
761 action = Action.ADD
764 stats[0] += 1
762 stats[0] += 1
765 elif command == '-':
763 elif command == '-':
766 affects_old = True
764 affects_old = True
767 action = Action.DELETE
765 action = Action.DELETE
768 stats[1] += 1
766 stats[1] += 1
769 else:
767 else:
770 affects_old = affects_new = True
768 affects_old = affects_new = True
771 action = Action.UNMODIFIED
769 action = Action.UNMODIFIED
772
770
773 if not self._newline_marker.match(line):
771 if not self._newline_marker.match(line):
774 old_line += affects_old
772 old_line += affects_old
775 new_line += affects_new
773 new_line += affects_new
776 lines.append({
774 lines.append({
777 'old_lineno': affects_old and old_line or '',
775 'old_lineno': affects_old and old_line or '',
778 'new_lineno': affects_new and new_line or '',
776 'new_lineno': affects_new and new_line or '',
779 'action': action,
777 'action': action,
780 'line': self._clean_line(line, command)
778 'line': self._clean_line(line, command)
781 })
779 })
782 raw_diff.append(line)
780 raw_diff.append(line)
783
781
784 line = diff_iter.next()
782 line = diff_iter.next()
785
783
786 if self._newline_marker.match(line):
784 if self._newline_marker.match(line):
787 # we need to append to lines, since this is not
785 # we need to append to lines, since this is not
788 # counted in the line specs of diff
786 # counted in the line specs of diff
789 if affects_old:
787 if affects_old:
790 action = Action.OLD_NO_NL
788 action = Action.OLD_NO_NL
791 elif affects_new:
789 elif affects_new:
792 action = Action.NEW_NO_NL
790 action = Action.NEW_NO_NL
793 else:
791 else:
794 raise Exception('invalid context for no newline')
792 raise Exception('invalid context for no newline')
795
793
796 lines.append({
794 lines.append({
797 'old_lineno': None,
795 'old_lineno': None,
798 'new_lineno': None,
796 'new_lineno': None,
799 'action': action,
797 'action': action,
800 'line': self._clean_line(line, command)
798 'line': self._clean_line(line, command)
801 })
799 })
802
800
803 except StopIteration:
801 except StopIteration:
804 pass
802 pass
805
803
806 return ''.join(raw_diff), chunks, stats
804 return ''.join(raw_diff), chunks, stats
807
805
808 def _safe_id(self, idstring):
806 def _safe_id(self, idstring):
809 """Make a string safe for including in an id attribute.
807 """Make a string safe for including in an id attribute.
810
808
811 The HTML spec says that id attributes 'must begin with
809 The HTML spec says that id attributes 'must begin with
812 a letter ([A-Za-z]) and may be followed by any number
810 a letter ([A-Za-z]) and may be followed by any number
813 of letters, digits ([0-9]), hyphens ("-"), underscores
811 of letters, digits ([0-9]), hyphens ("-"), underscores
814 ("_"), colons (":"), and periods (".")'. These regexps
812 ("_"), colons (":"), and periods (".")'. These regexps
815 are slightly over-zealous, in that they remove colons
813 are slightly over-zealous, in that they remove colons
816 and periods unnecessarily.
814 and periods unnecessarily.
817
815
818 Whitespace is transformed into underscores, and then
816 Whitespace is transformed into underscores, and then
819 anything which is not a hyphen or a character that
817 anything which is not a hyphen or a character that
820 matches \w (alphanumerics and underscore) is removed.
818 matches \w (alphanumerics and underscore) is removed.
821
819
822 """
820 """
823 # Transform all whitespace to underscore
821 # Transform all whitespace to underscore
824 idstring = re.sub(r'\s', "_", '%s' % idstring)
822 idstring = re.sub(r'\s', "_", '%s' % idstring)
825 # Remove everything that is not a hyphen or a member of \w
823 # Remove everything that is not a hyphen or a member of \w
826 idstring = re.sub(r'(?!-)\W', "", idstring).lower()
824 idstring = re.sub(r'(?!-)\W', "", idstring).lower()
827 return idstring
825 return idstring
828
826
827 @classmethod
828 def diff_splitter(cls, string):
829 """
830 Diff split that emulates .splitlines() but works only on \n
831 """
832 if not string:
833 return
834 elif string == '\n':
835 yield u'\n'
836 else:
837
838 has_newline = string.endswith('\n')
839 elements = string.split('\n')
840 if has_newline:
841 # skip last element as it's empty string from newlines
842 elements = elements[:-1]
843
844 len_elements = len(elements)
845
846 for cnt, line in enumerate(elements, start=1):
847 last_line = cnt == len_elements
848 if last_line and not has_newline:
849 yield safe_unicode(line)
850 else:
851 yield safe_unicode(line) + '\n'
852
829 def prepare(self, inline_diff=True):
853 def prepare(self, inline_diff=True):
830 """
854 """
831 Prepare the passed udiff for HTML rendering.
855 Prepare the passed udiff for HTML rendering.
832
856
833 :return: A list of dicts with diff information.
857 :return: A list of dicts with diff information.
834 """
858 """
835 parsed = self._parser(inline_diff=inline_diff)
859 parsed = self._parser(inline_diff=inline_diff)
836 self.parsed = True
860 self.parsed = True
837 self.parsed_diff = parsed
861 self.parsed_diff = parsed
838 return parsed
862 return parsed
839
863
840 def as_raw(self, diff_lines=None):
864 def as_raw(self, diff_lines=None):
841 """
865 """
842 Returns raw diff as a byte string
866 Returns raw diff as a byte string
843 """
867 """
844 return self._diff.raw
868 return self._diff.raw
845
869
846 def as_html(self, table_class='code-difftable', line_class='line',
870 def as_html(self, table_class='code-difftable', line_class='line',
847 old_lineno_class='lineno old', new_lineno_class='lineno new',
871 old_lineno_class='lineno old', new_lineno_class='lineno new',
848 code_class='code', enable_comments=False, parsed_lines=None):
872 code_class='code', enable_comments=False, parsed_lines=None):
849 """
873 """
850 Return given diff as html table with customized css classes
874 Return given diff as html table with customized css classes
851 """
875 """
852 # TODO(marcink): not sure how to pass in translator
876 # TODO(marcink): not sure how to pass in translator
853 # here in an efficient way, leave the _ for proper gettext extraction
877 # here in an efficient way, leave the _ for proper gettext extraction
854 _ = lambda s: s
878 _ = lambda s: s
855
879
856 def _link_to_if(condition, label, url):
880 def _link_to_if(condition, label, url):
857 """
881 """
858 Generates a link if condition is meet or just the label if not.
882 Generates a link if condition is meet or just the label if not.
859 """
883 """
860
884
861 if condition:
885 if condition:
862 return '''<a href="%(url)s" class="tooltip"
886 return '''<a href="%(url)s" class="tooltip"
863 title="%(title)s">%(label)s</a>''' % {
887 title="%(title)s">%(label)s</a>''' % {
864 'title': _('Click to select line'),
888 'title': _('Click to select line'),
865 'url': url,
889 'url': url,
866 'label': label
890 'label': label
867 }
891 }
868 else:
892 else:
869 return label
893 return label
870 if not self.parsed:
894 if not self.parsed:
871 self.prepare()
895 self.prepare()
872
896
873 diff_lines = self.parsed_diff
897 diff_lines = self.parsed_diff
874 if parsed_lines:
898 if parsed_lines:
875 diff_lines = parsed_lines
899 diff_lines = parsed_lines
876
900
877 _html_empty = True
901 _html_empty = True
878 _html = []
902 _html = []
879 _html.append('''<table class="%(table_class)s">\n''' % {
903 _html.append('''<table class="%(table_class)s">\n''' % {
880 'table_class': table_class
904 'table_class': table_class
881 })
905 })
882
906
883 for diff in diff_lines:
907 for diff in diff_lines:
884 for line in diff['chunks']:
908 for line in diff['chunks']:
885 _html_empty = False
909 _html_empty = False
886 for change in line:
910 for change in line:
887 _html.append('''<tr class="%(lc)s %(action)s">\n''' % {
911 _html.append('''<tr class="%(lc)s %(action)s">\n''' % {
888 'lc': line_class,
912 'lc': line_class,
889 'action': change['action']
913 'action': change['action']
890 })
914 })
891 anchor_old_id = ''
915 anchor_old_id = ''
892 anchor_new_id = ''
916 anchor_new_id = ''
893 anchor_old = "%(filename)s_o%(oldline_no)s" % {
917 anchor_old = "%(filename)s_o%(oldline_no)s" % {
894 'filename': self._safe_id(diff['filename']),
918 'filename': self._safe_id(diff['filename']),
895 'oldline_no': change['old_lineno']
919 'oldline_no': change['old_lineno']
896 }
920 }
897 anchor_new = "%(filename)s_n%(oldline_no)s" % {
921 anchor_new = "%(filename)s_n%(oldline_no)s" % {
898 'filename': self._safe_id(diff['filename']),
922 'filename': self._safe_id(diff['filename']),
899 'oldline_no': change['new_lineno']
923 'oldline_no': change['new_lineno']
900 }
924 }
901 cond_old = (change['old_lineno'] != '...' and
925 cond_old = (change['old_lineno'] != '...' and
902 change['old_lineno'])
926 change['old_lineno'])
903 cond_new = (change['new_lineno'] != '...' and
927 cond_new = (change['new_lineno'] != '...' and
904 change['new_lineno'])
928 change['new_lineno'])
905 if cond_old:
929 if cond_old:
906 anchor_old_id = 'id="%s"' % anchor_old
930 anchor_old_id = 'id="%s"' % anchor_old
907 if cond_new:
931 if cond_new:
908 anchor_new_id = 'id="%s"' % anchor_new
932 anchor_new_id = 'id="%s"' % anchor_new
909
933
910 if change['action'] != Action.CONTEXT:
934 if change['action'] != Action.CONTEXT:
911 anchor_link = True
935 anchor_link = True
912 else:
936 else:
913 anchor_link = False
937 anchor_link = False
914
938
915 ###########################################################
939 ###########################################################
916 # COMMENT ICONS
940 # COMMENT ICONS
917 ###########################################################
941 ###########################################################
918 _html.append('''\t<td class="add-comment-line"><span class="add-comment-content">''')
942 _html.append('''\t<td class="add-comment-line"><span class="add-comment-content">''')
919
943
920 if enable_comments and change['action'] != Action.CONTEXT:
944 if enable_comments and change['action'] != Action.CONTEXT:
921 _html.append('''<a href="#"><span class="icon-comment-add"></span></a>''')
945 _html.append('''<a href="#"><span class="icon-comment-add"></span></a>''')
922
946
923 _html.append('''</span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>\n''')
947 _html.append('''</span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>\n''')
924
948
925 ###########################################################
949 ###########################################################
926 # OLD LINE NUMBER
950 # OLD LINE NUMBER
927 ###########################################################
951 ###########################################################
928 _html.append('''\t<td %(a_id)s class="%(olc)s">''' % {
952 _html.append('''\t<td %(a_id)s class="%(olc)s">''' % {
929 'a_id': anchor_old_id,
953 'a_id': anchor_old_id,
930 'olc': old_lineno_class
954 'olc': old_lineno_class
931 })
955 })
932
956
933 _html.append('''%(link)s''' % {
957 _html.append('''%(link)s''' % {
934 'link': _link_to_if(anchor_link, change['old_lineno'],
958 'link': _link_to_if(anchor_link, change['old_lineno'],
935 '#%s' % anchor_old)
959 '#%s' % anchor_old)
936 })
960 })
937 _html.append('''</td>\n''')
961 _html.append('''</td>\n''')
938 ###########################################################
962 ###########################################################
939 # NEW LINE NUMBER
963 # NEW LINE NUMBER
940 ###########################################################
964 ###########################################################
941
965
942 _html.append('''\t<td %(a_id)s class="%(nlc)s">''' % {
966 _html.append('''\t<td %(a_id)s class="%(nlc)s">''' % {
943 'a_id': anchor_new_id,
967 'a_id': anchor_new_id,
944 'nlc': new_lineno_class
968 'nlc': new_lineno_class
945 })
969 })
946
970
947 _html.append('''%(link)s''' % {
971 _html.append('''%(link)s''' % {
948 'link': _link_to_if(anchor_link, change['new_lineno'],
972 'link': _link_to_if(anchor_link, change['new_lineno'],
949 '#%s' % anchor_new)
973 '#%s' % anchor_new)
950 })
974 })
951 _html.append('''</td>\n''')
975 _html.append('''</td>\n''')
952 ###########################################################
976 ###########################################################
953 # CODE
977 # CODE
954 ###########################################################
978 ###########################################################
955 code_classes = [code_class]
979 code_classes = [code_class]
956 if (not enable_comments or
980 if (not enable_comments or
957 change['action'] == Action.CONTEXT):
981 change['action'] == Action.CONTEXT):
958 code_classes.append('no-comment')
982 code_classes.append('no-comment')
959 _html.append('\t<td class="%s">' % ' '.join(code_classes))
983 _html.append('\t<td class="%s">' % ' '.join(code_classes))
960 _html.append('''\n\t\t<pre>%(code)s</pre>\n''' % {
984 _html.append('''\n\t\t<pre>%(code)s</pre>\n''' % {
961 'code': change['line']
985 'code': change['line']
962 })
986 })
963
987
964 _html.append('''\t</td>''')
988 _html.append('''\t</td>''')
965 _html.append('''\n</tr>\n''')
989 _html.append('''\n</tr>\n''')
966 _html.append('''</table>''')
990 _html.append('''</table>''')
967 if _html_empty:
991 if _html_empty:
968 return None
992 return None
969 return ''.join(_html)
993 return ''.join(_html)
970
994
971 def stat(self):
995 def stat(self):
972 """
996 """
973 Returns tuple of added, and removed lines for this instance
997 Returns tuple of added, and removed lines for this instance
974 """
998 """
975 return self.adds, self.removes
999 return self.adds, self.removes
976
1000
977 def get_context_of_line(
1001 def get_context_of_line(
978 self, path, diff_line=None, context_before=3, context_after=3):
1002 self, path, diff_line=None, context_before=3, context_after=3):
979 """
1003 """
980 Returns the context lines for the specified diff line.
1004 Returns the context lines for the specified diff line.
981
1005
982 :type diff_line: :class:`DiffLineNumber`
1006 :type diff_line: :class:`DiffLineNumber`
983 """
1007 """
984 assert self.parsed, "DiffProcessor is not initialized."
1008 assert self.parsed, "DiffProcessor is not initialized."
985
1009
986 if None not in diff_line:
1010 if None not in diff_line:
987 raise ValueError(
1011 raise ValueError(
988 "Cannot specify both line numbers: {}".format(diff_line))
1012 "Cannot specify both line numbers: {}".format(diff_line))
989
1013
990 file_diff = self._get_file_diff(path)
1014 file_diff = self._get_file_diff(path)
991 chunk, idx = self._find_chunk_line_index(file_diff, diff_line)
1015 chunk, idx = self._find_chunk_line_index(file_diff, diff_line)
992
1016
993 first_line_to_include = max(idx - context_before, 0)
1017 first_line_to_include = max(idx - context_before, 0)
994 first_line_after_context = idx + context_after + 1
1018 first_line_after_context = idx + context_after + 1
995 context_lines = chunk[first_line_to_include:first_line_after_context]
1019 context_lines = chunk[first_line_to_include:first_line_after_context]
996
1020
997 line_contents = [
1021 line_contents = [
998 _context_line(line) for line in context_lines
1022 _context_line(line) for line in context_lines
999 if _is_diff_content(line)]
1023 if _is_diff_content(line)]
1000 # TODO: johbo: Interim fixup, the diff chunks drop the final newline.
1024 # TODO: johbo: Interim fixup, the diff chunks drop the final newline.
1001 # Once they are fixed, we can drop this line here.
1025 # Once they are fixed, we can drop this line here.
1002 if line_contents:
1026 if line_contents:
1003 line_contents[-1] = (
1027 line_contents[-1] = (
1004 line_contents[-1][0], line_contents[-1][1].rstrip('\n') + '\n')
1028 line_contents[-1][0], line_contents[-1][1].rstrip('\n') + '\n')
1005 return line_contents
1029 return line_contents
1006
1030
1007 def find_context(self, path, context, offset=0):
1031 def find_context(self, path, context, offset=0):
1008 """
1032 """
1009 Finds the given `context` inside of the diff.
1033 Finds the given `context` inside of the diff.
1010
1034
1011 Use the parameter `offset` to specify which offset the target line has
1035 Use the parameter `offset` to specify which offset the target line has
1012 inside of the given `context`. This way the correct diff line will be
1036 inside of the given `context`. This way the correct diff line will be
1013 returned.
1037 returned.
1014
1038
1015 :param offset: Shall be used to specify the offset of the main line
1039 :param offset: Shall be used to specify the offset of the main line
1016 within the given `context`.
1040 within the given `context`.
1017 """
1041 """
1018 if offset < 0 or offset >= len(context):
1042 if offset < 0 or offset >= len(context):
1019 raise ValueError(
1043 raise ValueError(
1020 "Only positive values up to the length of the context "
1044 "Only positive values up to the length of the context "
1021 "minus one are allowed.")
1045 "minus one are allowed.")
1022
1046
1023 matches = []
1047 matches = []
1024 file_diff = self._get_file_diff(path)
1048 file_diff = self._get_file_diff(path)
1025
1049
1026 for chunk in file_diff['chunks']:
1050 for chunk in file_diff['chunks']:
1027 context_iter = iter(context)
1051 context_iter = iter(context)
1028 for line_idx, line in enumerate(chunk):
1052 for line_idx, line in enumerate(chunk):
1029 try:
1053 try:
1030 if _context_line(line) == context_iter.next():
1054 if _context_line(line) == context_iter.next():
1031 continue
1055 continue
1032 except StopIteration:
1056 except StopIteration:
1033 matches.append((line_idx, chunk))
1057 matches.append((line_idx, chunk))
1034 context_iter = iter(context)
1058 context_iter = iter(context)
1035
1059
1036 # Increment position and triger StopIteration
1060 # Increment position and triger StopIteration
1037 # if we had a match at the end
1061 # if we had a match at the end
1038 line_idx += 1
1062 line_idx += 1
1039 try:
1063 try:
1040 context_iter.next()
1064 context_iter.next()
1041 except StopIteration:
1065 except StopIteration:
1042 matches.append((line_idx, chunk))
1066 matches.append((line_idx, chunk))
1043
1067
1044 effective_offset = len(context) - offset
1068 effective_offset = len(context) - offset
1045 found_at_diff_lines = [
1069 found_at_diff_lines = [
1046 _line_to_diff_line_number(chunk[idx - effective_offset])
1070 _line_to_diff_line_number(chunk[idx - effective_offset])
1047 for idx, chunk in matches]
1071 for idx, chunk in matches]
1048
1072
1049 return found_at_diff_lines
1073 return found_at_diff_lines
1050
1074
1051 def _get_file_diff(self, path):
1075 def _get_file_diff(self, path):
1052 for file_diff in self.parsed_diff:
1076 for file_diff in self.parsed_diff:
1053 if file_diff['filename'] == path:
1077 if file_diff['filename'] == path:
1054 break
1078 break
1055 else:
1079 else:
1056 raise FileNotInDiffException("File {} not in diff".format(path))
1080 raise FileNotInDiffException("File {} not in diff".format(path))
1057 return file_diff
1081 return file_diff
1058
1082
1059 def _find_chunk_line_index(self, file_diff, diff_line):
1083 def _find_chunk_line_index(self, file_diff, diff_line):
1060 for chunk in file_diff['chunks']:
1084 for chunk in file_diff['chunks']:
1061 for idx, line in enumerate(chunk):
1085 for idx, line in enumerate(chunk):
1062 if line['old_lineno'] == diff_line.old:
1086 if line['old_lineno'] == diff_line.old:
1063 return chunk, idx
1087 return chunk, idx
1064 if line['new_lineno'] == diff_line.new:
1088 if line['new_lineno'] == diff_line.new:
1065 return chunk, idx
1089 return chunk, idx
1066 raise LineNotInDiffException(
1090 raise LineNotInDiffException(
1067 "The line {} is not part of the diff.".format(diff_line))
1091 "The line {} is not part of the diff.".format(diff_line))
1068
1092
1069
1093
1070 def _is_diff_content(line):
1094 def _is_diff_content(line):
1071 return line['action'] in (
1095 return line['action'] in (
1072 Action.UNMODIFIED, Action.ADD, Action.DELETE)
1096 Action.UNMODIFIED, Action.ADD, Action.DELETE)
1073
1097
1074
1098
1075 def _context_line(line):
1099 def _context_line(line):
1076 return (line['action'], line['line'])
1100 return (line['action'], line['line'])
1077
1101
1078
1102
1079 DiffLineNumber = collections.namedtuple('DiffLineNumber', ['old', 'new'])
1103 DiffLineNumber = collections.namedtuple('DiffLineNumber', ['old', 'new'])
1080
1104
1081
1105
1082 def _line_to_diff_line_number(line):
1106 def _line_to_diff_line_number(line):
1083 new_line_no = line['new_lineno'] or None
1107 new_line_no = line['new_lineno'] or None
1084 old_line_no = line['old_lineno'] or None
1108 old_line_no = line['old_lineno'] or None
1085 return DiffLineNumber(old=old_line_no, new=new_line_no)
1109 return DiffLineNumber(old=old_line_no, new=new_line_no)
1086
1110
1087
1111
1088 class FileNotInDiffException(Exception):
1112 class FileNotInDiffException(Exception):
1089 """
1113 """
1090 Raised when the context for a missing file is requested.
1114 Raised when the context for a missing file is requested.
1091
1115
1092 If you request the context for a line in a file which is not part of the
1116 If you request the context for a line in a file which is not part of the
1093 given diff, then this exception is raised.
1117 given diff, then this exception is raised.
1094 """
1118 """
1095
1119
1096
1120
1097 class LineNotInDiffException(Exception):
1121 class LineNotInDiffException(Exception):
1098 """
1122 """
1099 Raised when the context for a missing line is requested.
1123 Raised when the context for a missing line is requested.
1100
1124
1101 If you request the context for a line in a file and this line is not
1125 If you request the context for a line in a file and this line is not
1102 part of the given diff, then this exception is raised.
1126 part of the given diff, then this exception is raised.
1103 """
1127 """
1104
1128
1105
1129
1106 class DiffLimitExceeded(Exception):
1130 class DiffLimitExceeded(Exception):
1107 pass
1131 pass
@@ -1,331 +1,311
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2016-2018 RhodeCode GmbH
3 # Copyright (C) 2016-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21 import pytest
21 import pytest
22 from pygments.lexers import get_lexer_by_name
22 from pygments.lexers import get_lexer_by_name
23
23
24 from rhodecode.tests import no_newline_id_generator
24 from rhodecode.tests import no_newline_id_generator
25 from rhodecode.lib.codeblocks import (
25 from rhodecode.lib.codeblocks import (
26 tokenize_string, split_token_stream, rollup_tokenstream,
26 tokenize_string, split_token_stream, rollup_tokenstream,
27 render_tokenstream)
27 render_tokenstream)
28
28
29
29
30 class TestTokenizeString(object):
30 class TestTokenizeString(object):
31
31
32 python_code = '''
32 python_code = '''
33 import this
33 import this
34
34
35 var = 6
35 var = 6
36 print "this"
36 print "this"
37
37
38 '''
38 '''
39
39
40 def test_tokenize_as_python(self):
40 def test_tokenize_as_python(self):
41 lexer = get_lexer_by_name('python')
41 lexer = get_lexer_by_name('python')
42 tokens = list(tokenize_string(self.python_code, lexer))
42 tokens = list(tokenize_string(self.python_code, lexer))
43
43
44 assert tokens == [
44 assert tokens == [
45 ('', u'\n'),
45 ('', u'\n'),
46 ('', u' '),
46 ('', u' '),
47 ('kn', u'import'),
47 ('kn', u'import'),
48 ('', u' '),
48 ('', u' '),
49 ('nn', u'this'),
49 ('nn', u'this'),
50 ('', u'\n'),
50 ('', u'\n'),
51 ('', u'\n'),
51 ('', u'\n'),
52 ('', u' '),
52 ('', u' '),
53 ('n', u'var'),
53 ('n', u'var'),
54 ('', u' '),
54 ('', u' '),
55 ('o', u'='),
55 ('o', u'='),
56 ('', u' '),
56 ('', u' '),
57 ('mi', u'6'),
57 ('mi', u'6'),
58 ('', u'\n'),
58 ('', u'\n'),
59 ('', u' '),
59 ('', u' '),
60 ('k', u'print'),
60 ('k', u'print'),
61 ('', u' '),
61 ('', u' '),
62 ('s2', u'"'),
62 ('s2', u'"'),
63 ('s2', u'this'),
63 ('s2', u'this'),
64 ('s2', u'"'),
64 ('s2', u'"'),
65 ('', u'\n'),
65 ('', u'\n'),
66 ('', u'\n'),
66 ('', u'\n'),
67 ('', u' ')
67 ('', u' ')
68 ]
68 ]
69
69
70 def test_tokenize_as_text(self):
70 def test_tokenize_as_text(self):
71 lexer = get_lexer_by_name('text')
71 lexer = get_lexer_by_name('text')
72 tokens = list(tokenize_string(self.python_code, lexer))
72 tokens = list(tokenize_string(self.python_code, lexer))
73
73
74 assert tokens == [
74 assert tokens == [
75 ('',
75 ('',
76 u'\n import this\n\n var = 6\n print "this"\n\n ')
76 u'\n import this\n\n var = 6\n print "this"\n\n ')
77 ]
77 ]
78
78
79
79
80 class TestSplitTokenStream(object):
80 class TestSplitTokenStream(object):
81
81
82 def test_split_token_stream(self):
82 def test_split_token_stream(self):
83 lines = list(split_token_stream(
83 lines = list(split_token_stream(
84 [('type1', 'some\ntext'), ('type2', 'more\n')]))
84 [('type1', 'some\ntext'), ('type2', 'more\n')]))
85
85
86 assert lines == [
86 assert lines == [
87 [('type1', u'some')],
87 [('type1', u'some')],
88 [('type1', u'text'), ('type2', u'more')],
88 [('type1', u'text'), ('type2', u'more')],
89 [('type2', u'')],
89 [('type2', u'')],
90 ]
90 ]
91
91
92 def test_split_token_stream_other_char(self):
93 lines = list(split_token_stream(
94 [('type1', 'some\ntext'), ('type2', 'more\n')],
95 split_string='m'))
96
97 assert lines == [
98 [('type1', 'so')],
99 [('type1', 'e\ntext'), ('type2', '')],
100 [('type2', 'ore\n')],
101 ]
102
103 def test_split_token_stream_without_char(self):
104 lines = list(split_token_stream(
105 [('type1', 'some\ntext'), ('type2', 'more\n')],
106 split_string='z'))
107
108 assert lines == [
109 [('type1', 'some\ntext'), ('type2', 'more\n')]
110 ]
111
112 def test_split_token_stream_single(self):
92 def test_split_token_stream_single(self):
113 lines = list(split_token_stream(
93 lines = list(split_token_stream(
114 [('type1', '\n')], split_string='\n'))
94 [('type1', '\n')]))
115
95
116 assert lines == [
96 assert lines == [
117 [('type1', '')],
97 [('type1', '')],
118 [('type1', '')],
98 [('type1', '')],
119 ]
99 ]
120
100
121 def test_split_token_stream_single_repeat(self):
101 def test_split_token_stream_single_repeat(self):
122 lines = list(split_token_stream(
102 lines = list(split_token_stream(
123 [('type1', '\n\n\n')], split_string='\n'))
103 [('type1', '\n\n\n')]))
124
104
125 assert lines == [
105 assert lines == [
126 [('type1', '')],
106 [('type1', '')],
127 [('type1', '')],
107 [('type1', '')],
128 [('type1', '')],
108 [('type1', '')],
129 [('type1', '')],
109 [('type1', '')],
130 ]
110 ]
131
111
132 def test_split_token_stream_multiple_repeat(self):
112 def test_split_token_stream_multiple_repeat(self):
133 lines = list(split_token_stream(
113 lines = list(split_token_stream(
134 [('type1', '\n\n'), ('type2', '\n\n')], split_string='\n'))
114 [('type1', '\n\n'), ('type2', '\n\n')]))
135
115
136 assert lines == [
116 assert lines == [
137 [('type1', '')],
117 [('type1', '')],
138 [('type1', '')],
118 [('type1', '')],
139 [('type1', ''), ('type2', '')],
119 [('type1', ''), ('type2', '')],
140 [('type2', '')],
120 [('type2', '')],
141 [('type2', '')],
121 [('type2', '')],
142 ]
122 ]
143
123
144
124
145 class TestRollupTokens(object):
125 class TestRollupTokens(object):
146
126
147 @pytest.mark.parametrize('tokenstream,output', [
127 @pytest.mark.parametrize('tokenstream,output', [
148 ([],
128 ([],
149 []),
129 []),
150 ([('A', 'hell'), ('A', 'o')], [
130 ([('A', 'hell'), ('A', 'o')], [
151 ('A', [
131 ('A', [
152 ('', 'hello')]),
132 ('', 'hello')]),
153 ]),
133 ]),
154 ([('A', 'hell'), ('B', 'o')], [
134 ([('A', 'hell'), ('B', 'o')], [
155 ('A', [
135 ('A', [
156 ('', 'hell')]),
136 ('', 'hell')]),
157 ('B', [
137 ('B', [
158 ('', 'o')]),
138 ('', 'o')]),
159 ]),
139 ]),
160 ([('A', 'hel'), ('A', 'lo'), ('B', ' '), ('A', 'there')], [
140 ([('A', 'hel'), ('A', 'lo'), ('B', ' '), ('A', 'there')], [
161 ('A', [
141 ('A', [
162 ('', 'hello')]),
142 ('', 'hello')]),
163 ('B', [
143 ('B', [
164 ('', ' ')]),
144 ('', ' ')]),
165 ('A', [
145 ('A', [
166 ('', 'there')]),
146 ('', 'there')]),
167 ]),
147 ]),
168 ])
148 ])
169 def test_rollup_tokenstream_without_ops(self, tokenstream, output):
149 def test_rollup_tokenstream_without_ops(self, tokenstream, output):
170 assert list(rollup_tokenstream(tokenstream)) == output
150 assert list(rollup_tokenstream(tokenstream)) == output
171
151
172 @pytest.mark.parametrize('tokenstream,output', [
152 @pytest.mark.parametrize('tokenstream,output', [
173 ([],
153 ([],
174 []),
154 []),
175 ([('A', '', 'hell'), ('A', '', 'o')], [
155 ([('A', '', 'hell'), ('A', '', 'o')], [
176 ('A', [
156 ('A', [
177 ('', 'hello')]),
157 ('', 'hello')]),
178 ]),
158 ]),
179 ([('A', '', 'hell'), ('B', '', 'o')], [
159 ([('A', '', 'hell'), ('B', '', 'o')], [
180 ('A', [
160 ('A', [
181 ('', 'hell')]),
161 ('', 'hell')]),
182 ('B', [
162 ('B', [
183 ('', 'o')]),
163 ('', 'o')]),
184 ]),
164 ]),
185 ([('A', '', 'h'), ('B', '', 'e'), ('C', '', 'y')], [
165 ([('A', '', 'h'), ('B', '', 'e'), ('C', '', 'y')], [
186 ('A', [
166 ('A', [
187 ('', 'h')]),
167 ('', 'h')]),
188 ('B', [
168 ('B', [
189 ('', 'e')]),
169 ('', 'e')]),
190 ('C', [
170 ('C', [
191 ('', 'y')]),
171 ('', 'y')]),
192 ]),
172 ]),
193 ([('A', '', 'h'), ('A', '', 'e'), ('C', '', 'y')], [
173 ([('A', '', 'h'), ('A', '', 'e'), ('C', '', 'y')], [
194 ('A', [
174 ('A', [
195 ('', 'he')]),
175 ('', 'he')]),
196 ('C', [
176 ('C', [
197 ('', 'y')]),
177 ('', 'y')]),
198 ]),
178 ]),
199 ([('A', 'ins', 'h'), ('A', 'ins', 'e')], [
179 ([('A', 'ins', 'h'), ('A', 'ins', 'e')], [
200 ('A', [
180 ('A', [
201 ('ins', 'he')
181 ('ins', 'he')
202 ]),
182 ]),
203 ]),
183 ]),
204 ([('A', 'ins', 'h'), ('A', 'del', 'e')], [
184 ([('A', 'ins', 'h'), ('A', 'del', 'e')], [
205 ('A', [
185 ('A', [
206 ('ins', 'h'),
186 ('ins', 'h'),
207 ('del', 'e')
187 ('del', 'e')
208 ]),
188 ]),
209 ]),
189 ]),
210 ([('A', 'ins', 'h'), ('B', 'del', 'e'), ('B', 'del', 'y')], [
190 ([('A', 'ins', 'h'), ('B', 'del', 'e'), ('B', 'del', 'y')], [
211 ('A', [
191 ('A', [
212 ('ins', 'h'),
192 ('ins', 'h'),
213 ]),
193 ]),
214 ('B', [
194 ('B', [
215 ('del', 'ey'),
195 ('del', 'ey'),
216 ]),
196 ]),
217 ]),
197 ]),
218 ([('A', 'ins', 'h'), ('A', 'del', 'e'), ('B', 'del', 'y')], [
198 ([('A', 'ins', 'h'), ('A', 'del', 'e'), ('B', 'del', 'y')], [
219 ('A', [
199 ('A', [
220 ('ins', 'h'),
200 ('ins', 'h'),
221 ('del', 'e'),
201 ('del', 'e'),
222 ]),
202 ]),
223 ('B', [
203 ('B', [
224 ('del', 'y'),
204 ('del', 'y'),
225 ]),
205 ]),
226 ]),
206 ]),
227 ([('A', '', 'some'), ('A', 'ins', 'new'), ('A', '', 'name')], [
207 ([('A', '', 'some'), ('A', 'ins', 'new'), ('A', '', 'name')], [
228 ('A', [
208 ('A', [
229 ('', 'some'),
209 ('', 'some'),
230 ('ins', 'new'),
210 ('ins', 'new'),
231 ('', 'name'),
211 ('', 'name'),
232 ]),
212 ]),
233 ]),
213 ]),
234 ])
214 ])
235 def test_rollup_tokenstream_with_ops(self, tokenstream, output):
215 def test_rollup_tokenstream_with_ops(self, tokenstream, output):
236 assert list(rollup_tokenstream(tokenstream)) == output
216 assert list(rollup_tokenstream(tokenstream)) == output
237
217
238
218
239 class TestRenderTokenStream(object):
219 class TestRenderTokenStream(object):
240
220
241 @pytest.mark.parametrize('tokenstream,output', [
221 @pytest.mark.parametrize('tokenstream,output', [
242 (
222 (
243 [],
223 [],
244 '',
224 '',
245 ),
225 ),
246 (
226 (
247 [('', '', u'')],
227 [('', '', u'')],
248 '<span></span>',
228 '<span></span>',
249 ),
229 ),
250 (
230 (
251 [('', '', u'text')],
231 [('', '', u'text')],
252 '<span>text</span>',
232 '<span>text</span>',
253 ),
233 ),
254 (
234 (
255 [('A', '', u'')],
235 [('A', '', u'')],
256 '<span class="A"></span>',
236 '<span class="A"></span>',
257 ),
237 ),
258 (
238 (
259 [('A', '', u'hello')],
239 [('A', '', u'hello')],
260 '<span class="A">hello</span>',
240 '<span class="A">hello</span>',
261 ),
241 ),
262 (
242 (
263 [('A', '', u'hel'), ('A', '', u'lo')],
243 [('A', '', u'hel'), ('A', '', u'lo')],
264 '<span class="A">hello</span>',
244 '<span class="A">hello</span>',
265 ),
245 ),
266 (
246 (
267 [('A', '', u'two\n'), ('A', '', u'lines')],
247 [('A', '', u'two\n'), ('A', '', u'lines')],
268 '<span class="A">two\nlines</span>',
248 '<span class="A">two\nlines</span>',
269 ),
249 ),
270 (
250 (
271 [('A', '', u'\nthree\n'), ('A', '', u'lines')],
251 [('A', '', u'\nthree\n'), ('A', '', u'lines')],
272 '<span class="A">\nthree\nlines</span>',
252 '<span class="A">\nthree\nlines</span>',
273 ),
253 ),
274 (
254 (
275 [('', '', u'\n'), ('A', '', u'line')],
255 [('', '', u'\n'), ('A', '', u'line')],
276 '<span>\n</span><span class="A">line</span>',
256 '<span>\n</span><span class="A">line</span>',
277 ),
257 ),
278 (
258 (
279 [('', 'ins', u'\n'), ('A', '', u'line')],
259 [('', 'ins', u'\n'), ('A', '', u'line')],
280 '<span><ins>\n</ins></span><span class="A">line</span>',
260 '<span><ins>\n</ins></span><span class="A">line</span>',
281 ),
261 ),
282 (
262 (
283 [('A', '', u'hel'), ('A', 'ins', u'lo')],
263 [('A', '', u'hel'), ('A', 'ins', u'lo')],
284 '<span class="A">hel<ins>lo</ins></span>',
264 '<span class="A">hel<ins>lo</ins></span>',
285 ),
265 ),
286 (
266 (
287 [('A', '', u'hel'), ('A', 'ins', u'l'), ('A', 'ins', u'o')],
267 [('A', '', u'hel'), ('A', 'ins', u'l'), ('A', 'ins', u'o')],
288 '<span class="A">hel<ins>lo</ins></span>',
268 '<span class="A">hel<ins>lo</ins></span>',
289 ),
269 ),
290 (
270 (
291 [('A', '', u'hel'), ('A', 'ins', u'l'), ('A', 'del', u'o')],
271 [('A', '', u'hel'), ('A', 'ins', u'l'), ('A', 'del', u'o')],
292 '<span class="A">hel<ins>l</ins><del>o</del></span>',
272 '<span class="A">hel<ins>l</ins><del>o</del></span>',
293 ),
273 ),
294 (
274 (
295 [('A', '', u'hel'), ('B', '', u'lo')],
275 [('A', '', u'hel'), ('B', '', u'lo')],
296 '<span class="A">hel</span><span class="B">lo</span>',
276 '<span class="A">hel</span><span class="B">lo</span>',
297 ),
277 ),
298 (
278 (
299 [('A', '', u'hel'), ('B', 'ins', u'lo')],
279 [('A', '', u'hel'), ('B', 'ins', u'lo')],
300 '<span class="A">hel</span><span class="B"><ins>lo</ins></span>',
280 '<span class="A">hel</span><span class="B"><ins>lo</ins></span>',
301 ),
281 ),
302 ], ids=no_newline_id_generator)
282 ], ids=no_newline_id_generator)
303 def test_render_tokenstream_with_ops(self, tokenstream, output):
283 def test_render_tokenstream_with_ops(self, tokenstream, output):
304 html = render_tokenstream(tokenstream)
284 html = render_tokenstream(tokenstream)
305 assert html == output
285 assert html == output
306
286
307 @pytest.mark.parametrize('tokenstream,output', [
287 @pytest.mark.parametrize('tokenstream,output', [
308 (
288 (
309 [('A', u'hel'), ('A', u'lo')],
289 [('A', u'hel'), ('A', u'lo')],
310 '<span class="A">hello</span>',
290 '<span class="A">hello</span>',
311 ),
291 ),
312 (
292 (
313 [('A', u'hel'), ('A', u'l'), ('A', u'o')],
293 [('A', u'hel'), ('A', u'l'), ('A', u'o')],
314 '<span class="A">hello</span>',
294 '<span class="A">hello</span>',
315 ),
295 ),
316 (
296 (
317 [('A', u'hel'), ('A', u'l'), ('A', u'o')],
297 [('A', u'hel'), ('A', u'l'), ('A', u'o')],
318 '<span class="A">hello</span>',
298 '<span class="A">hello</span>',
319 ),
299 ),
320 (
300 (
321 [('A', u'hel'), ('B', u'lo')],
301 [('A', u'hel'), ('B', u'lo')],
322 '<span class="A">hel</span><span class="B">lo</span>',
302 '<span class="A">hel</span><span class="B">lo</span>',
323 ),
303 ),
324 (
304 (
325 [('A', u'hel'), ('B', u'lo')],
305 [('A', u'hel'), ('B', u'lo')],
326 '<span class="A">hel</span><span class="B">lo</span>',
306 '<span class="A">hel</span><span class="B">lo</span>',
327 ),
307 ),
328 ])
308 ])
329 def test_render_tokenstream_without_ops(self, tokenstream, output):
309 def test_render_tokenstream_without_ops(self, tokenstream, output):
330 html = render_tokenstream(tokenstream)
310 html = render_tokenstream(tokenstream)
331 assert html == output
311 assert html == output
@@ -1,813 +1,831
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2
2
3 # Copyright (C) 2010-2018 RhodeCode GmbH
3 # Copyright (C) 2010-2018 RhodeCode GmbH
4 #
4 #
5 # This program is free software: you can redistribute it and/or modify
5 # This program is free software: you can redistribute it and/or modify
6 # it under the terms of the GNU Affero General Public License, version 3
6 # it under the terms of the GNU Affero General Public License, version 3
7 # (only), as published by the Free Software Foundation.
7 # (only), as published by the Free Software Foundation.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU Affero General Public License
14 # You should have received a copy of the GNU Affero General Public License
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15 # along with this program. If not, see <http://www.gnu.org/licenses/>.
16 #
16 #
17 # This program is dual-licensed. If you wish to learn more about the
17 # This program is dual-licensed. If you wish to learn more about the
18 # RhodeCode Enterprise Edition, including its added features, Support services,
18 # RhodeCode Enterprise Edition, including its added features, Support services,
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
19 # and proprietary license terms, please see https://rhodecode.com/licenses/
20
20
21 import textwrap
21 import textwrap
22
22
23 import pytest
23 import pytest
24
24
25 from rhodecode.lib.diffs import (
25 from rhodecode.lib.diffs import (
26 DiffProcessor,
26 DiffProcessor,
27 NEW_FILENODE, DEL_FILENODE, MOD_FILENODE, RENAMED_FILENODE,
27 NEW_FILENODE, DEL_FILENODE, MOD_FILENODE, RENAMED_FILENODE,
28 CHMOD_FILENODE, BIN_FILENODE, COPIED_FILENODE)
28 CHMOD_FILENODE, BIN_FILENODE, COPIED_FILENODE)
29 from rhodecode.tests.fixture import Fixture
29 from rhodecode.tests.fixture import Fixture, no_newline_id_generator
30 from rhodecode.lib.vcs.backends.git.repository import GitDiff
30 from rhodecode.lib.vcs.backends.git.repository import GitDiff
31 from rhodecode.lib.vcs.backends.hg.repository import MercurialDiff
31 from rhodecode.lib.vcs.backends.hg.repository import MercurialDiff
32 from rhodecode.lib.vcs.backends.svn.repository import SubversionDiff
32 from rhodecode.lib.vcs.backends.svn.repository import SubversionDiff
33
33
34 fixture = Fixture()
34 fixture = Fixture()
35
35
36
36
37 def test_diffprocessor_as_html_with_comments():
37 def test_diffprocessor_as_html_with_comments():
38 raw_diff = textwrap.dedent('''
38 raw_diff = textwrap.dedent('''
39 diff --git a/setup.py b/setup.py
39 diff --git a/setup.py b/setup.py
40 index 5b36422..cfd698e 100755
40 index 5b36422..cfd698e 100755
41 --- a/setup.py
41 --- a/setup.py
42 +++ b/setup.py
42 +++ b/setup.py
43 @@ -2,7 +2,7 @@
43 @@ -2,7 +2,7 @@
44 #!/usr/bin/python
44 #!/usr/bin/python
45 # Setup file for X
45 # Setup file for X
46 # Copyright (C) No one
46 # Copyright (C) No one
47 -
47 -
48 +x
48 +x
49 try:
49 try:
50 from setuptools import setup, Extension
50 from setuptools import setup, Extension
51 except ImportError:
51 except ImportError:
52 ''')
52 ''')
53 diff = GitDiff(raw_diff)
53 diff = GitDiff(raw_diff)
54 processor = DiffProcessor(diff)
54 processor = DiffProcessor(diff)
55 processor.prepare()
55 processor.prepare()
56
56
57 # Note that the cell with the context in line 5 (in the html) has the
57 # Note that the cell with the context in line 5 (in the html) has the
58 # no-comment class, which will prevent the add comment icon to be displayed.
58 # no-comment class, which will prevent the add comment icon to be displayed.
59 expected_html = textwrap.dedent('''
59 expected_html = textwrap.dedent('''
60 <table class="code-difftable">
60 <table class="code-difftable">
61 <tr class="line context">
61 <tr class="line context">
62 <td class="add-comment-line"><span class="add-comment-content"></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
62 <td class="add-comment-line"><span class="add-comment-content"></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
63 <td class="lineno old">...</td>
63 <td class="lineno old">...</td>
64 <td class="lineno new">...</td>
64 <td class="lineno new">...</td>
65 <td class="code no-comment">
65 <td class="code no-comment">
66 <pre>@@ -2,7 +2,7 @@
66 <pre>@@ -2,7 +2,7 @@
67 </pre>
67 </pre>
68 </td>
68 </td>
69 </tr>
69 </tr>
70 <tr class="line unmod">
70 <tr class="line unmod">
71 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
71 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
72 <td id="setuppy_o2" class="lineno old"><a href="#setuppy_o2" class="tooltip"
72 <td id="setuppy_o2" class="lineno old"><a href="#setuppy_o2" class="tooltip"
73 title="Click to select line">2</a></td>
73 title="Click to select line">2</a></td>
74 <td id="setuppy_n2" class="lineno new"><a href="#setuppy_n2" class="tooltip"
74 <td id="setuppy_n2" class="lineno new"><a href="#setuppy_n2" class="tooltip"
75 title="Click to select line">2</a></td>
75 title="Click to select line">2</a></td>
76 <td class="code">
76 <td class="code">
77 <pre>#!/usr/bin/python
77 <pre>#!/usr/bin/python
78 </pre>
78 </pre>
79 </td>
79 </td>
80 </tr>
80 </tr>
81 <tr class="line unmod">
81 <tr class="line unmod">
82 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
82 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
83 <td id="setuppy_o3" class="lineno old"><a href="#setuppy_o3" class="tooltip"
83 <td id="setuppy_o3" class="lineno old"><a href="#setuppy_o3" class="tooltip"
84 title="Click to select line">3</a></td>
84 title="Click to select line">3</a></td>
85 <td id="setuppy_n3" class="lineno new"><a href="#setuppy_n3" class="tooltip"
85 <td id="setuppy_n3" class="lineno new"><a href="#setuppy_n3" class="tooltip"
86 title="Click to select line">3</a></td>
86 title="Click to select line">3</a></td>
87 <td class="code">
87 <td class="code">
88 <pre># Setup file for X
88 <pre># Setup file for X
89 </pre>
89 </pre>
90 </td>
90 </td>
91 </tr>
91 </tr>
92 <tr class="line unmod">
92 <tr class="line unmod">
93 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
93 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
94 <td id="setuppy_o4" class="lineno old"><a href="#setuppy_o4" class="tooltip"
94 <td id="setuppy_o4" class="lineno old"><a href="#setuppy_o4" class="tooltip"
95 title="Click to select line">4</a></td>
95 title="Click to select line">4</a></td>
96 <td id="setuppy_n4" class="lineno new"><a href="#setuppy_n4" class="tooltip"
96 <td id="setuppy_n4" class="lineno new"><a href="#setuppy_n4" class="tooltip"
97 title="Click to select line">4</a></td>
97 title="Click to select line">4</a></td>
98 <td class="code">
98 <td class="code">
99 <pre># Copyright (C) No one
99 <pre># Copyright (C) No one
100 </pre>
100 </pre>
101 </td>
101 </td>
102 </tr>
102 </tr>
103 <tr class="line del">
103 <tr class="line del">
104 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
104 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
105 <td id="setuppy_o5" class="lineno old"><a href="#setuppy_o5" class="tooltip"
105 <td id="setuppy_o5" class="lineno old"><a href="#setuppy_o5" class="tooltip"
106 title="Click to select line">5</a></td>
106 title="Click to select line">5</a></td>
107 <td class="lineno new"><a href="#setuppy_n" class="tooltip"
107 <td class="lineno new"><a href="#setuppy_n" class="tooltip"
108 title="Click to select line"></a></td>
108 title="Click to select line"></a></td>
109 <td class="code">
109 <td class="code">
110 <pre>
110 <pre>
111 </pre>
111 </pre>
112 </td>
112 </td>
113 </tr>
113 </tr>
114 <tr class="line add">
114 <tr class="line add">
115 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
115 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
116 <td class="lineno old"><a href="#setuppy_o" class="tooltip"
116 <td class="lineno old"><a href="#setuppy_o" class="tooltip"
117 title="Click to select line"></a></td>
117 title="Click to select line"></a></td>
118 <td id="setuppy_n5" class="lineno new"><a href="#setuppy_n5" class="tooltip"
118 <td id="setuppy_n5" class="lineno new"><a href="#setuppy_n5" class="tooltip"
119 title="Click to select line">5</a></td>
119 title="Click to select line">5</a></td>
120 <td class="code">
120 <td class="code">
121 <pre><ins>x</ins>
121 <pre><ins>x</ins>
122 </pre>
122 </pre>
123 </td>
123 </td>
124 </tr>
124 </tr>
125 <tr class="line unmod">
125 <tr class="line unmod">
126 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
126 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
127 <td id="setuppy_o6" class="lineno old"><a href="#setuppy_o6" class="tooltip"
127 <td id="setuppy_o6" class="lineno old"><a href="#setuppy_o6" class="tooltip"
128 title="Click to select line">6</a></td>
128 title="Click to select line">6</a></td>
129 <td id="setuppy_n6" class="lineno new"><a href="#setuppy_n6" class="tooltip"
129 <td id="setuppy_n6" class="lineno new"><a href="#setuppy_n6" class="tooltip"
130 title="Click to select line">6</a></td>
130 title="Click to select line">6</a></td>
131 <td class="code">
131 <td class="code">
132 <pre>try:
132 <pre>try:
133 </pre>
133 </pre>
134 </td>
134 </td>
135 </tr>
135 </tr>
136 <tr class="line unmod">
136 <tr class="line unmod">
137 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
137 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
138 <td id="setuppy_o7" class="lineno old"><a href="#setuppy_o7" class="tooltip"
138 <td id="setuppy_o7" class="lineno old"><a href="#setuppy_o7" class="tooltip"
139 title="Click to select line">7</a></td>
139 title="Click to select line">7</a></td>
140 <td id="setuppy_n7" class="lineno new"><a href="#setuppy_n7" class="tooltip"
140 <td id="setuppy_n7" class="lineno new"><a href="#setuppy_n7" class="tooltip"
141 title="Click to select line">7</a></td>
141 title="Click to select line">7</a></td>
142 <td class="code">
142 <td class="code">
143 <pre> from setuptools import setup, Extension
143 <pre> from setuptools import setup, Extension
144 </pre>
144 </pre>
145 </td>
145 </td>
146 </tr>
146 </tr>
147 <tr class="line unmod">
147 <tr class="line unmod">
148 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
148 <td class="add-comment-line"><span class="add-comment-content"><a href="#"><span class="icon-comment-add"></span></a></span></td><td class="comment-toggle tooltip" title="Toggle Comment Thread"><i class="icon-comment"></i></td>
149 <td id="setuppy_o8" class="lineno old"><a href="#setuppy_o8" class="tooltip"
149 <td id="setuppy_o8" class="lineno old"><a href="#setuppy_o8" class="tooltip"
150 title="Click to select line">8</a></td>
150 title="Click to select line">8</a></td>
151 <td id="setuppy_n8" class="lineno new"><a href="#setuppy_n8" class="tooltip"
151 <td id="setuppy_n8" class="lineno new"><a href="#setuppy_n8" class="tooltip"
152 title="Click to select line">8</a></td>
152 title="Click to select line">8</a></td>
153 <td class="code">
153 <td class="code">
154 <pre>except ImportError:
154 <pre>except ImportError:
155 </pre>
155 </pre>
156 </td>
156 </td>
157 </tr>
157 </tr>
158 </table>
158 </table>
159 ''').strip()
159 ''').strip()
160 html = processor.as_html(enable_comments=True).replace('\t', ' ')
160 html = processor.as_html(enable_comments=True).replace('\t', ' ')
161
161
162 assert html == expected_html
162 assert html == expected_html
163
163
164
164
165 class TestMixedFilenameEncodings:
165 class TestMixedFilenameEncodings(object):
166
166
167 @pytest.fixture(scope="class")
167 @pytest.fixture(scope="class")
168 def raw_diff(self):
168 def raw_diff(self):
169 return fixture.load_resource(
169 return fixture.load_resource(
170 'hg_diff_mixed_filename_encodings.diff')
170 'hg_diff_mixed_filename_encodings.diff')
171
171
172 @pytest.fixture
172 @pytest.fixture
173 def processor(self, raw_diff):
173 def processor(self, raw_diff):
174 diff = MercurialDiff(raw_diff)
174 diff = MercurialDiff(raw_diff)
175 processor = DiffProcessor(diff)
175 processor = DiffProcessor(diff)
176 return processor
176 return processor
177
177
178 def test_filenames_are_decoded_to_unicode(self, processor):
178 def test_filenames_are_decoded_to_unicode(self, processor):
179 diff_data = processor.prepare()
179 diff_data = processor.prepare()
180 filenames = [item['filename'] for item in diff_data]
180 filenames = [item['filename'] for item in diff_data]
181 assert filenames == [
181 assert filenames == [
182 u'spΓ€cial-utf8.txt', u'spοΏ½cial-cp1252.txt', u'spοΏ½cial-latin1.txt']
182 u'spΓ€cial-utf8.txt', u'spοΏ½cial-cp1252.txt', u'spοΏ½cial-latin1.txt']
183
183
184 def test_raw_diff_is_decoded_to_unicode(self, processor):
184 def test_raw_diff_is_decoded_to_unicode(self, processor):
185 diff_data = processor.prepare()
185 diff_data = processor.prepare()
186 raw_diffs = [item['raw_diff'] for item in diff_data]
186 raw_diffs = [item['raw_diff'] for item in diff_data]
187 new_file_message = u'\nnew file mode 100644\n'
187 new_file_message = u'\nnew file mode 100644\n'
188 expected_raw_diffs = [
188 expected_raw_diffs = [
189 u' a/spΓ€cial-utf8.txt b/spΓ€cial-utf8.txt' + new_file_message,
189 u' a/spΓ€cial-utf8.txt b/spΓ€cial-utf8.txt' + new_file_message,
190 u' a/spοΏ½cial-cp1252.txt b/spοΏ½cial-cp1252.txt' + new_file_message,
190 u' a/spοΏ½cial-cp1252.txt b/spοΏ½cial-cp1252.txt' + new_file_message,
191 u' a/spοΏ½cial-latin1.txt b/spοΏ½cial-latin1.txt' + new_file_message]
191 u' a/spοΏ½cial-latin1.txt b/spοΏ½cial-latin1.txt' + new_file_message]
192 assert raw_diffs == expected_raw_diffs
192 assert raw_diffs == expected_raw_diffs
193
193
194 def test_as_raw_preserves_the_encoding(self, processor, raw_diff):
194 def test_as_raw_preserves_the_encoding(self, processor, raw_diff):
195 assert processor.as_raw() == raw_diff
195 assert processor.as_raw() == raw_diff
196
196
197
197
198 # TODO: mikhail: format the following data structure properly
198 # TODO: mikhail: format the following data structure properly
199 DIFF_FIXTURES = [
199 DIFF_FIXTURES = [
200 ('hg',
200 ('hg',
201 'hg_diff_add_single_binary_file.diff',
201 'hg_diff_add_single_binary_file.diff',
202 [('US Warszawa.jpg', 'A',
202 [('US Warszawa.jpg', 'A',
203 {'added': 0,
203 {'added': 0,
204 'deleted': 0,
204 'deleted': 0,
205 'binary': True,
205 'binary': True,
206 'ops': {NEW_FILENODE: 'new file 100755',
206 'ops': {NEW_FILENODE: 'new file 100755',
207 BIN_FILENODE: 'binary diff hidden'}}),
207 BIN_FILENODE: 'binary diff hidden'}}),
208 ]),
208 ]),
209 ('hg',
209 ('hg',
210 'hg_diff_mod_single_binary_file.diff',
210 'hg_diff_mod_single_binary_file.diff',
211 [('US Warszawa.jpg', 'M',
211 [('US Warszawa.jpg', 'M',
212 {'added': 0,
212 {'added': 0,
213 'deleted': 0,
213 'deleted': 0,
214 'binary': True,
214 'binary': True,
215 'ops': {MOD_FILENODE: 'modified file',
215 'ops': {MOD_FILENODE: 'modified file',
216 BIN_FILENODE: 'binary diff hidden'}}),
216 BIN_FILENODE: 'binary diff hidden'}}),
217 ]),
217 ]),
218 ('hg',
218 ('hg',
219 'hg_diff_mod_single_file_and_rename_and_chmod.diff',
219 'hg_diff_mod_single_file_and_rename_and_chmod.diff',
220 [('README', 'M',
220 [('README', 'M',
221 {'added': 3,
221 {'added': 3,
222 'deleted': 0,
222 'deleted': 0,
223 'binary': False,
223 'binary': False,
224 'ops': {MOD_FILENODE: 'modified file',
224 'ops': {MOD_FILENODE: 'modified file',
225 RENAMED_FILENODE: 'file renamed from README.rst to README',
225 RENAMED_FILENODE: 'file renamed from README.rst to README',
226 CHMOD_FILENODE: 'modified file chmod 100755 => 100644'}}),
226 CHMOD_FILENODE: 'modified file chmod 100755 => 100644'}}),
227 ]),
227 ]),
228 ('hg',
228 ('hg',
229 'hg_diff_no_newline.diff',
229 'hg_diff_no_newline.diff',
230 [('server.properties', 'M',
230 [('server.properties', 'M',
231 {'added': 2,
231 {'added': 2,
232 'deleted': 1,
232 'deleted': 1,
233 'binary': False,
233 'binary': False,
234 'ops': {MOD_FILENODE: 'modified file'}}),
234 'ops': {MOD_FILENODE: 'modified file'}}),
235 ]),
235 ]),
236 ('hg',
236 ('hg',
237 'hg_diff_mod_file_and_rename.diff',
237 'hg_diff_mod_file_and_rename.diff',
238 [('README.rst', 'M',
238 [('README.rst', 'M',
239 {'added': 3,
239 {'added': 3,
240 'deleted': 0,
240 'deleted': 0,
241 'binary': False,
241 'binary': False,
242 'ops': {MOD_FILENODE: 'modified file',
242 'ops': {MOD_FILENODE: 'modified file',
243 RENAMED_FILENODE: 'file renamed from README to README.rst'}}),
243 RENAMED_FILENODE: 'file renamed from README to README.rst'}}),
244 ]),
244 ]),
245 ('hg',
245 ('hg',
246 'hg_diff_del_single_binary_file.diff',
246 'hg_diff_del_single_binary_file.diff',
247 [('US Warszawa.jpg', 'D',
247 [('US Warszawa.jpg', 'D',
248 {'added': 0,
248 {'added': 0,
249 'deleted': 0,
249 'deleted': 0,
250 'binary': True,
250 'binary': True,
251 'ops': {DEL_FILENODE: 'deleted file',
251 'ops': {DEL_FILENODE: 'deleted file',
252 BIN_FILENODE: 'binary diff hidden'}}),
252 BIN_FILENODE: 'binary diff hidden'}}),
253 ]),
253 ]),
254 ('hg',
254 ('hg',
255 'hg_diff_chmod_and_mod_single_binary_file.diff',
255 'hg_diff_chmod_and_mod_single_binary_file.diff',
256 [('gravatar.png', 'M',
256 [('gravatar.png', 'M',
257 {'added': 0,
257 {'added': 0,
258 'deleted': 0,
258 'deleted': 0,
259 'binary': True,
259 'binary': True,
260 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755',
260 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755',
261 BIN_FILENODE: 'binary diff hidden'}}),
261 BIN_FILENODE: 'binary diff hidden'}}),
262 ]),
262 ]),
263 ('hg',
263 ('hg',
264 'hg_diff_chmod.diff',
264 'hg_diff_chmod.diff',
265 [('file', 'M',
265 [('file', 'M',
266 {'added': 0,
266 {'added': 0,
267 'deleted': 0,
267 'deleted': 0,
268 'binary': True,
268 'binary': True,
269 'ops': {CHMOD_FILENODE: 'modified file chmod 100755 => 100644'}}),
269 'ops': {CHMOD_FILENODE: 'modified file chmod 100755 => 100644'}}),
270 ]),
270 ]),
271 ('hg',
271 ('hg',
272 'hg_diff_rename_file.diff',
272 'hg_diff_rename_file.diff',
273 [('file_renamed', 'M',
273 [('file_renamed', 'M',
274 {'added': 0,
274 {'added': 0,
275 'deleted': 0,
275 'deleted': 0,
276 'binary': True,
276 'binary': True,
277 'ops': {RENAMED_FILENODE: 'file renamed from file to file_renamed'}}),
277 'ops': {RENAMED_FILENODE: 'file renamed from file to file_renamed'}}),
278 ]),
278 ]),
279 ('hg',
279 ('hg',
280 'hg_diff_rename_and_chmod_file.diff',
280 'hg_diff_rename_and_chmod_file.diff',
281 [('README', 'M',
281 [('README', 'M',
282 {'added': 0,
282 {'added': 0,
283 'deleted': 0,
283 'deleted': 0,
284 'binary': True,
284 'binary': True,
285 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755',
285 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755',
286 RENAMED_FILENODE: 'file renamed from README.rst to README'}}),
286 RENAMED_FILENODE: 'file renamed from README.rst to README'}}),
287 ]),
287 ]),
288 ('hg',
288 ('hg',
289 'hg_diff_binary_and_normal.diff',
289 'hg_diff_binary_and_normal.diff',
290 [('img/baseline-10px.png', 'A',
290 [('img/baseline-10px.png', 'A',
291 {'added': 0,
291 {'added': 0,
292 'deleted': 0,
292 'deleted': 0,
293 'binary': True,
293 'binary': True,
294 'ops': {NEW_FILENODE: 'new file 100644',
294 'ops': {NEW_FILENODE: 'new file 100644',
295 BIN_FILENODE: 'binary diff hidden'}}),
295 BIN_FILENODE: 'binary diff hidden'}}),
296 ('js/jquery/hashgrid.js', 'A',
296 ('js/jquery/hashgrid.js', 'A',
297 {'added': 340,
297 {'added': 340,
298 'deleted': 0,
298 'deleted': 0,
299 'binary': False,
299 'binary': False,
300 'ops': {NEW_FILENODE: 'new file 100755'}}),
300 'ops': {NEW_FILENODE: 'new file 100755'}}),
301 ('index.html', 'M',
301 ('index.html', 'M',
302 {'added': 3,
302 {'added': 3,
303 'deleted': 2,
303 'deleted': 2,
304 'binary': False,
304 'binary': False,
305 'ops': {MOD_FILENODE: 'modified file'}}),
305 'ops': {MOD_FILENODE: 'modified file'}}),
306 ('less/docs.less', 'M',
306 ('less/docs.less', 'M',
307 {'added': 34,
307 {'added': 34,
308 'deleted': 0,
308 'deleted': 0,
309 'binary': False,
309 'binary': False,
310 'ops': {MOD_FILENODE: 'modified file'}}),
310 'ops': {MOD_FILENODE: 'modified file'}}),
311 ('less/scaffolding.less', 'M',
311 ('less/scaffolding.less', 'M',
312 {'added': 1,
312 {'added': 1,
313 'deleted': 3,
313 'deleted': 3,
314 'binary': False,
314 'binary': False,
315 'ops': {MOD_FILENODE: 'modified file'}}),
315 'ops': {MOD_FILENODE: 'modified file'}}),
316 ('readme.markdown', 'M',
316 ('readme.markdown', 'M',
317 {'added': 1,
317 {'added': 1,
318 'deleted': 10,
318 'deleted': 10,
319 'binary': False,
319 'binary': False,
320 'ops': {MOD_FILENODE: 'modified file'}}),
320 'ops': {MOD_FILENODE: 'modified file'}}),
321 ('img/baseline-20px.png', 'D',
321 ('img/baseline-20px.png', 'D',
322 {'added': 0,
322 {'added': 0,
323 'deleted': 0,
323 'deleted': 0,
324 'binary': True,
324 'binary': True,
325 'ops': {DEL_FILENODE: 'deleted file',
325 'ops': {DEL_FILENODE: 'deleted file',
326 BIN_FILENODE: 'binary diff hidden'}}),
326 BIN_FILENODE: 'binary diff hidden'}}),
327 ('js/global.js', 'D',
327 ('js/global.js', 'D',
328 {'added': 0,
328 {'added': 0,
329 'deleted': 75,
329 'deleted': 75,
330 'binary': False,
330 'binary': False,
331 'ops': {DEL_FILENODE: 'deleted file'}})
331 'ops': {DEL_FILENODE: 'deleted file'}})
332 ]),
332 ]),
333 ('git',
333 ('git',
334 'git_diff_chmod.diff',
334 'git_diff_chmod.diff',
335 [('work-horus.xls', 'M',
335 [('work-horus.xls', 'M',
336 {'added': 0,
336 {'added': 0,
337 'deleted': 0,
337 'deleted': 0,
338 'binary': True,
338 'binary': True,
339 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755'}})
339 'ops': {CHMOD_FILENODE: 'modified file chmod 100644 => 100755'}})
340 ]),
340 ]),
341 ('git',
341 ('git',
342 'git_diff_rename_file.diff',
342 'git_diff_rename_file.diff',
343 [('file.xls', 'M',
343 [('file.xls', 'M',
344 {'added': 0,
344 {'added': 0,
345 'deleted': 0,
345 'deleted': 0,
346 'binary': True,
346 'binary': True,
347 'ops': {
347 'ops': {
348 RENAMED_FILENODE: 'file renamed from work-horus.xls to file.xls'}})
348 RENAMED_FILENODE: 'file renamed from work-horus.xls to file.xls'}})
349 ]),
349 ]),
350 ('git',
350 ('git',
351 'git_diff_mod_single_binary_file.diff',
351 'git_diff_mod_single_binary_file.diff',
352 [('US Warszawa.jpg', 'M',
352 [('US Warszawa.jpg', 'M',
353 {'added': 0,
353 {'added': 0,
354 'deleted': 0,
354 'deleted': 0,
355 'binary': True,
355 'binary': True,
356 'ops': {MOD_FILENODE: 'modified file',
356 'ops': {MOD_FILENODE: 'modified file',
357 BIN_FILENODE: 'binary diff hidden'}})
357 BIN_FILENODE: 'binary diff hidden'}})
358 ]),
358 ]),
359 ('git',
359 ('git',
360 'git_diff_binary_and_normal.diff',
360 'git_diff_binary_and_normal.diff',
361 [('img/baseline-10px.png', 'A',
361 [('img/baseline-10px.png', 'A',
362 {'added': 0,
362 {'added': 0,
363 'deleted': 0,
363 'deleted': 0,
364 'binary': True,
364 'binary': True,
365 'ops': {NEW_FILENODE: 'new file 100644',
365 'ops': {NEW_FILENODE: 'new file 100644',
366 BIN_FILENODE: 'binary diff hidden'}}),
366 BIN_FILENODE: 'binary diff hidden'}}),
367 ('js/jquery/hashgrid.js', 'A',
367 ('js/jquery/hashgrid.js', 'A',
368 {'added': 340,
368 {'added': 340,
369 'deleted': 0,
369 'deleted': 0,
370 'binary': False,
370 'binary': False,
371 'ops': {NEW_FILENODE: 'new file 100755'}}),
371 'ops': {NEW_FILENODE: 'new file 100755'}}),
372 ('index.html', 'M',
372 ('index.html', 'M',
373 {'added': 3,
373 {'added': 3,
374 'deleted': 2,
374 'deleted': 2,
375 'binary': False,
375 'binary': False,
376 'ops': {MOD_FILENODE: 'modified file'}}),
376 'ops': {MOD_FILENODE: 'modified file'}}),
377 ('less/docs.less', 'M',
377 ('less/docs.less', 'M',
378 {'added': 34,
378 {'added': 34,
379 'deleted': 0,
379 'deleted': 0,
380 'binary': False,
380 'binary': False,
381 'ops': {MOD_FILENODE: 'modified file'}}),
381 'ops': {MOD_FILENODE: 'modified file'}}),
382 ('less/scaffolding.less', 'M',
382 ('less/scaffolding.less', 'M',
383 {'added': 1,
383 {'added': 1,
384 'deleted': 3,
384 'deleted': 3,
385 'binary': False,
385 'binary': False,
386 'ops': {MOD_FILENODE: 'modified file'}}),
386 'ops': {MOD_FILENODE: 'modified file'}}),
387 ('readme.markdown', 'M',
387 ('readme.markdown', 'M',
388 {'added': 1,
388 {'added': 1,
389 'deleted': 10,
389 'deleted': 10,
390 'binary': False,
390 'binary': False,
391 'ops': {MOD_FILENODE: 'modified file'}}),
391 'ops': {MOD_FILENODE: 'modified file'}}),
392 ('img/baseline-20px.png', 'D',
392 ('img/baseline-20px.png', 'D',
393 {'added': 0,
393 {'added': 0,
394 'deleted': 0,
394 'deleted': 0,
395 'binary': True,
395 'binary': True,
396 'ops': {DEL_FILENODE: 'deleted file',
396 'ops': {DEL_FILENODE: 'deleted file',
397 BIN_FILENODE: 'binary diff hidden'}}),
397 BIN_FILENODE: 'binary diff hidden'}}),
398 ('js/global.js', 'D',
398 ('js/global.js', 'D',
399 {'added': 0,
399 {'added': 0,
400 'deleted': 75,
400 'deleted': 75,
401 'binary': False,
401 'binary': False,
402 'ops': {DEL_FILENODE: 'deleted file'}}),
402 'ops': {DEL_FILENODE: 'deleted file'}}),
403 ]),
403 ]),
404 ('hg',
404 ('hg',
405 'diff_with_diff_data.diff',
405 'diff_with_diff_data.diff',
406 [('vcs/backends/base.py', 'M',
406 [('vcs/backends/base.py', 'M',
407 {'added': 18,
407 {'added': 18,
408 'deleted': 2,
408 'deleted': 2,
409 'binary': False,
409 'binary': False,
410 'ops': {MOD_FILENODE: 'modified file'}}),
410 'ops': {MOD_FILENODE: 'modified file'}}),
411 ('vcs/backends/git/repository.py', 'M',
411 ('vcs/backends/git/repository.py', 'M',
412 {'added': 46,
412 {'added': 46,
413 'deleted': 15,
413 'deleted': 15,
414 'binary': False,
414 'binary': False,
415 'ops': {MOD_FILENODE: 'modified file'}}),
415 'ops': {MOD_FILENODE: 'modified file'}}),
416 ('vcs/backends/hg.py', 'M',
416 ('vcs/backends/hg.py', 'M',
417 {'added': 22,
417 {'added': 22,
418 'deleted': 3,
418 'deleted': 3,
419 'binary': False,
419 'binary': False,
420 'ops': {MOD_FILENODE: 'modified file'}}),
420 'ops': {MOD_FILENODE: 'modified file'}}),
421 ('vcs/tests/test_git.py', 'M',
421 ('vcs/tests/test_git.py', 'M',
422 {'added': 5,
422 {'added': 5,
423 'deleted': 5,
423 'deleted': 5,
424 'binary': False,
424 'binary': False,
425 'ops': {MOD_FILENODE: 'modified file'}}),
425 'ops': {MOD_FILENODE: 'modified file'}}),
426 ('vcs/tests/test_repository.py', 'M',
426 ('vcs/tests/test_repository.py', 'M',
427 {'added': 174,
427 {'added': 174,
428 'deleted': 2,
428 'deleted': 2,
429 'binary': False,
429 'binary': False,
430 'ops': {MOD_FILENODE: 'modified file'}}),
430 'ops': {MOD_FILENODE: 'modified file'}}),
431 ]),
431 ]),
432 ('hg',
432 ('hg',
433 'hg_diff_copy_file.diff',
433 'hg_diff_copy_file.diff',
434 [('file2', 'M',
434 [('file2', 'M',
435 {'added': 0,
435 {'added': 0,
436 'deleted': 0,
436 'deleted': 0,
437 'binary': True,
437 'binary': True,
438 'ops': {COPIED_FILENODE: 'file copied from file1 to file2'}}),
438 'ops': {COPIED_FILENODE: 'file copied from file1 to file2'}}),
439 ]),
439 ]),
440 ('hg',
440 ('hg',
441 'hg_diff_copy_and_modify_file.diff',
441 'hg_diff_copy_and_modify_file.diff',
442 [('file3', 'M',
442 [('file3', 'M',
443 {'added': 1,
443 {'added': 1,
444 'deleted': 0,
444 'deleted': 0,
445 'binary': False,
445 'binary': False,
446 'ops': {COPIED_FILENODE: 'file copied from file2 to file3',
446 'ops': {COPIED_FILENODE: 'file copied from file2 to file3',
447 MOD_FILENODE: 'modified file'}}),
447 MOD_FILENODE: 'modified file'}}),
448 ]),
448 ]),
449 ('hg',
449 ('hg',
450 'hg_diff_copy_and_chmod_file.diff',
450 'hg_diff_copy_and_chmod_file.diff',
451 [('file4', 'M',
451 [('file4', 'M',
452 {'added': 0,
452 {'added': 0,
453 'deleted': 0,
453 'deleted': 0,
454 'binary': True,
454 'binary': True,
455 'ops': {COPIED_FILENODE: 'file copied from file3 to file4',
455 'ops': {COPIED_FILENODE: 'file copied from file3 to file4',
456 CHMOD_FILENODE: 'modified file chmod 100644 => 100755'}}),
456 CHMOD_FILENODE: 'modified file chmod 100644 => 100755'}}),
457 ]),
457 ]),
458 ('hg',
458 ('hg',
459 'hg_diff_copy_chmod_and_edit_file.diff',
459 'hg_diff_copy_chmod_and_edit_file.diff',
460 [('file5', 'M',
460 [('file5', 'M',
461 {'added': 2,
461 {'added': 2,
462 'deleted': 1,
462 'deleted': 1,
463 'binary': False,
463 'binary': False,
464 'ops': {COPIED_FILENODE: 'file copied from file4 to file5',
464 'ops': {COPIED_FILENODE: 'file copied from file4 to file5',
465 CHMOD_FILENODE: 'modified file chmod 100755 => 100644',
465 CHMOD_FILENODE: 'modified file chmod 100755 => 100644',
466 MOD_FILENODE: 'modified file'}})]),
466 MOD_FILENODE: 'modified file'}})]),
467
467
468 # Diffs to validate rename and copy file with space in its name
468 # Diffs to validate rename and copy file with space in its name
469 ('git',
469 ('git',
470 'git_diff_rename_file_with_spaces.diff',
470 'git_diff_rename_file_with_spaces.diff',
471 [('file_with_ two spaces.txt', 'M',
471 [('file_with_ two spaces.txt', 'M',
472 {'added': 0,
472 {'added': 0,
473 'deleted': 0,
473 'deleted': 0,
474 'binary': True,
474 'binary': True,
475 'ops': {
475 'ops': {
476 RENAMED_FILENODE: (
476 RENAMED_FILENODE: (
477 'file renamed from file_with_ spaces.txt to file_with_ '
477 'file renamed from file_with_ spaces.txt to file_with_ '
478 ' two spaces.txt')}
478 ' two spaces.txt')}
479 }), ]),
479 }), ]),
480 ('hg',
480 ('hg',
481 'hg_diff_rename_file_with_spaces.diff',
481 'hg_diff_rename_file_with_spaces.diff',
482 [('file_changed _.txt', 'M',
482 [('file_changed _.txt', 'M',
483 {'added': 0,
483 {'added': 0,
484 'deleted': 0,
484 'deleted': 0,
485 'binary': True,
485 'binary': True,
486 'ops': {
486 'ops': {
487 RENAMED_FILENODE: (
487 RENAMED_FILENODE: (
488 'file renamed from file_ with update.txt to file_changed'
488 'file renamed from file_ with update.txt to file_changed'
489 ' _.txt')}
489 ' _.txt')}
490 }), ]),
490 }), ]),
491 ('hg',
491 ('hg',
492 'hg_diff_copy_file_with_spaces.diff',
492 'hg_diff_copy_file_with_spaces.diff',
493 [('file_copied_ with spaces.txt', 'M',
493 [('file_copied_ with spaces.txt', 'M',
494 {'added': 0,
494 {'added': 0,
495 'deleted': 0,
495 'deleted': 0,
496 'binary': True,
496 'binary': True,
497 'ops': {
497 'ops': {
498 COPIED_FILENODE: (
498 COPIED_FILENODE: (
499 'file copied from file_changed_without_spaces.txt to'
499 'file copied from file_changed_without_spaces.txt to'
500 ' file_copied_ with spaces.txt')}
500 ' file_copied_ with spaces.txt')}
501 }),
501 }),
502 ]),
502 ]),
503
503
504 # special signs from git
504 # special signs from git
505 ('git',
505 ('git',
506 'git_diff_binary_special_files.diff',
506 'git_diff_binary_special_files.diff',
507 [('css/_Icon\\r', 'A',
507 [('css/_Icon\\r', 'A',
508 {'added': 0,
508 {'added': 0,
509 'deleted': 0,
509 'deleted': 0,
510 'binary': True,
510 'binary': True,
511 'ops': {NEW_FILENODE: 'new file 100644',
511 'ops': {NEW_FILENODE: 'new file 100644',
512 BIN_FILENODE: 'binary diff hidden'}
512 BIN_FILENODE: 'binary diff hidden'}
513 }),
513 }),
514 ]),
514 ]),
515 ('git',
515 ('git',
516 'git_diff_binary_special_files_2.diff',
516 'git_diff_binary_special_files_2.diff',
517 [('css/Icon\\r', 'A',
517 [('css/Icon\\r', 'A',
518 {'added': 0,
518 {'added': 0,
519 'deleted': 0,
519 'deleted': 0,
520 'binary': True,
520 'binary': True,
521 'ops': {NEW_FILENODE: 'new file 100644', }
521 'ops': {NEW_FILENODE: 'new file 100644', }
522 }),
522 }),
523 ]),
523 ]),
524
524
525 ('svn',
525 ('svn',
526 'svn_diff_binary_add_file.diff',
526 'svn_diff_binary_add_file.diff',
527 [('intl.dll', 'A',
527 [('intl.dll', 'A',
528 {'added': 0,
528 {'added': 0,
529 'deleted': 0,
529 'deleted': 0,
530 'binary': False,
530 'binary': False,
531 'ops': {NEW_FILENODE: 'new file 10644',
531 'ops': {NEW_FILENODE: 'new file 10644',
532 #TODO(Marcink): depends on binary detection on svn patches
532 #TODO(Marcink): depends on binary detection on svn patches
533 # BIN_FILENODE: 'binary diff hidden'
533 # BIN_FILENODE: 'binary diff hidden'
534 }
534 }
535 }),
535 }),
536 ]),
536 ]),
537
537
538 ('svn',
538 ('svn',
539 'svn_diff_multiple_changes.diff',
539 'svn_diff_multiple_changes.diff',
540 [('trunk/doc/images/SettingsOverlay.png', 'M',
540 [('trunk/doc/images/SettingsOverlay.png', 'M',
541 {'added': 0,
541 {'added': 0,
542 'deleted': 0,
542 'deleted': 0,
543 'binary': False,
543 'binary': False,
544 'ops': {MOD_FILENODE: 'modified file',
544 'ops': {MOD_FILENODE: 'modified file',
545 #TODO(Marcink): depends on binary detection on svn patches
545 #TODO(Marcink): depends on binary detection on svn patches
546 # BIN_FILENODE: 'binary diff hidden'
546 # BIN_FILENODE: 'binary diff hidden'
547 }
547 }
548 }),
548 }),
549 ('trunk/doc/source/de/tsvn_ch04.xml', 'M',
549 ('trunk/doc/source/de/tsvn_ch04.xml', 'M',
550 {'added': 89,
550 {'added': 89,
551 'deleted': 34,
551 'deleted': 34,
552 'binary': False,
552 'binary': False,
553 'ops': {MOD_FILENODE: 'modified file'}
553 'ops': {MOD_FILENODE: 'modified file'}
554 }),
554 }),
555 ('trunk/doc/source/en/tsvn_ch04.xml', 'M',
555 ('trunk/doc/source/en/tsvn_ch04.xml', 'M',
556 {'added': 66,
556 {'added': 66,
557 'deleted': 21,
557 'deleted': 21,
558 'binary': False,
558 'binary': False,
559 'ops': {MOD_FILENODE: 'modified file'}
559 'ops': {MOD_FILENODE: 'modified file'}
560 }),
560 }),
561 ('trunk/src/Changelog.txt', 'M',
561 ('trunk/src/Changelog.txt', 'M',
562 {'added': 2,
562 {'added': 2,
563 'deleted': 0,
563 'deleted': 0,
564 'binary': False,
564 'binary': False,
565 'ops': {MOD_FILENODE: 'modified file'}
565 'ops': {MOD_FILENODE: 'modified file'}
566 }),
566 }),
567 ('trunk/src/Resources/TortoiseProcENG.rc', 'M',
567 ('trunk/src/Resources/TortoiseProcENG.rc', 'M',
568 {'added': 19,
568 {'added': 19,
569 'deleted': 13,
569 'deleted': 13,
570 'binary': False,
570 'binary': False,
571 'ops': {MOD_FILENODE: 'modified file'}
571 'ops': {MOD_FILENODE: 'modified file'}
572 }),
572 }),
573 ('trunk/src/TortoiseProc/SetOverlayPage.cpp', 'M',
573 ('trunk/src/TortoiseProc/SetOverlayPage.cpp', 'M',
574 {'added': 16,
574 {'added': 16,
575 'deleted': 1,
575 'deleted': 1,
576 'binary': False,
576 'binary': False,
577 'ops': {MOD_FILENODE: 'modified file'}
577 'ops': {MOD_FILENODE: 'modified file'}
578 }),
578 }),
579 ('trunk/src/TortoiseProc/SetOverlayPage.h', 'M',
579 ('trunk/src/TortoiseProc/SetOverlayPage.h', 'M',
580 {'added': 3,
580 {'added': 3,
581 'deleted': 0,
581 'deleted': 0,
582 'binary': False,
582 'binary': False,
583 'ops': {MOD_FILENODE: 'modified file'}
583 'ops': {MOD_FILENODE: 'modified file'}
584 }),
584 }),
585 ('trunk/src/TortoiseProc/resource.h', 'M',
585 ('trunk/src/TortoiseProc/resource.h', 'M',
586 {'added': 2,
586 {'added': 2,
587 'deleted': 0,
587 'deleted': 0,
588 'binary': False,
588 'binary': False,
589 'ops': {MOD_FILENODE: 'modified file'}
589 'ops': {MOD_FILENODE: 'modified file'}
590 }),
590 }),
591 ('trunk/src/TortoiseShell/ShellCache.h', 'M',
591 ('trunk/src/TortoiseShell/ShellCache.h', 'M',
592 {'added': 50,
592 {'added': 50,
593 'deleted': 1,
593 'deleted': 1,
594 'binary': False,
594 'binary': False,
595 'ops': {MOD_FILENODE: 'modified file'}
595 'ops': {MOD_FILENODE: 'modified file'}
596 }),
596 }),
597 ]),
597 ]),
598
598
599
599
600 # TODO: mikhail: do we still need this?
600 # TODO: mikhail: do we still need this?
601 # (
601 # (
602 # 'hg',
602 # 'hg',
603 # 'large_diff.diff',
603 # 'large_diff.diff',
604 # [
604 # [
605 # ('.hgignore', 'A', {
605 # ('.hgignore', 'A', {
606 # 'deleted': 0, 'binary': False, 'added': 3, 'ops': {
606 # 'deleted': 0, 'binary': False, 'added': 3, 'ops': {
607 # 1: 'new file 100644'}}),
607 # 1: 'new file 100644'}}),
608 # (
608 # (
609 # 'MANIFEST.in', 'A',
609 # 'MANIFEST.in', 'A',
610 # {'deleted': 0, 'binary': False, 'added': 3, 'ops': {
610 # {'deleted': 0, 'binary': False, 'added': 3, 'ops': {
611 # 1: 'new file 100644'}}),
611 # 1: 'new file 100644'}}),
612 # (
612 # (
613 # 'README.txt', 'A',
613 # 'README.txt', 'A',
614 # {'deleted': 0, 'binary': False, 'added': 19, 'ops': {
614 # {'deleted': 0, 'binary': False, 'added': 19, 'ops': {
615 # 1: 'new file 100644'}}),
615 # 1: 'new file 100644'}}),
616 # (
616 # (
617 # 'development.ini', 'A', {
617 # 'development.ini', 'A', {
618 # 'deleted': 0, 'binary': False, 'added': 116, 'ops': {
618 # 'deleted': 0, 'binary': False, 'added': 116, 'ops': {
619 # 1: 'new file 100644'}}),
619 # 1: 'new file 100644'}}),
620 # (
620 # (
621 # 'docs/index.txt', 'A', {
621 # 'docs/index.txt', 'A', {
622 # 'deleted': 0, 'binary': False, 'added': 19, 'ops': {
622 # 'deleted': 0, 'binary': False, 'added': 19, 'ops': {
623 # 1: 'new file 100644'}}),
623 # 1: 'new file 100644'}}),
624 # (
624 # (
625 # 'ez_setup.py', 'A', {
625 # 'ez_setup.py', 'A', {
626 # 'deleted': 0, 'binary': False, 'added': 276, 'ops': {
626 # 'deleted': 0, 'binary': False, 'added': 276, 'ops': {
627 # 1: 'new file 100644'}}),
627 # 1: 'new file 100644'}}),
628 # (
628 # (
629 # 'hgapp.py', 'A', {
629 # 'hgapp.py', 'A', {
630 # 'deleted': 0, 'binary': False, 'added': 26, 'ops': {
630 # 'deleted': 0, 'binary': False, 'added': 26, 'ops': {
631 # 1: 'new file 100644'}}),
631 # 1: 'new file 100644'}}),
632 # (
632 # (
633 # 'hgwebdir.config', 'A', {
633 # 'hgwebdir.config', 'A', {
634 # 'deleted': 0, 'binary': False, 'added': 21, 'ops': {
634 # 'deleted': 0, 'binary': False, 'added': 21, 'ops': {
635 # 1: 'new file 100644'}}),
635 # 1: 'new file 100644'}}),
636 # (
636 # (
637 # 'pylons_app.egg-info/PKG-INFO', 'A', {
637 # 'pylons_app.egg-info/PKG-INFO', 'A', {
638 # 'deleted': 0, 'binary': False, 'added': 10, 'ops': {
638 # 'deleted': 0, 'binary': False, 'added': 10, 'ops': {
639 # 1: 'new file 100644'}}),
639 # 1: 'new file 100644'}}),
640 # (
640 # (
641 # 'pylons_app.egg-info/SOURCES.txt', 'A', {
641 # 'pylons_app.egg-info/SOURCES.txt', 'A', {
642 # 'deleted': 0, 'binary': False, 'added': 33, 'ops': {
642 # 'deleted': 0, 'binary': False, 'added': 33, 'ops': {
643 # 1: 'new file 100644'}}),
643 # 1: 'new file 100644'}}),
644 # (
644 # (
645 # 'pylons_app.egg-info/dependency_links.txt', 'A', {
645 # 'pylons_app.egg-info/dependency_links.txt', 'A', {
646 # 'deleted': 0, 'binary': False, 'added': 1, 'ops': {
646 # 'deleted': 0, 'binary': False, 'added': 1, 'ops': {
647 # 1: 'new file 100644'}}),
647 # 1: 'new file 100644'}}),
648 # ]
648 # ]
649 # ),
649 # ),
650 ]
650 ]
651
651
652 DIFF_FIXTURES_WITH_CONTENT = [
652 DIFF_FIXTURES_WITH_CONTENT = [
653 (
653 (
654 'hg', 'hg_diff_single_file_change_newline.diff',
654 'hg', 'hg_diff_single_file_change_newline.diff',
655 [
655 [
656 (
656 (
657 'file_b', # filename
657 'file_b', # filename
658 'A', # change
658 'A', # change
659 { # stats
659 { # stats
660 'added': 1,
660 'added': 1,
661 'deleted': 0,
661 'deleted': 0,
662 'binary': False,
662 'binary': False,
663 'ops': {NEW_FILENODE: 'new file 100644', }
663 'ops': {NEW_FILENODE: 'new file 100644', }
664 },
664 },
665 '@@ -0,0 +1 @@\n+test_content b\n' # diff
665 '@@ -0,0 +1 @@\n+test_content b\n' # diff
666 ),
666 ),
667 ],
667 ],
668 ),
668 ),
669 (
669 (
670 'hg', 'hg_diff_double_file_change_newline.diff',
670 'hg', 'hg_diff_double_file_change_newline.diff',
671 [
671 [
672 (
672 (
673 'file_b', # filename
673 'file_b', # filename
674 'A', # change
674 'A', # change
675 { # stats
675 { # stats
676 'added': 1,
676 'added': 1,
677 'deleted': 0,
677 'deleted': 0,
678 'binary': False,
678 'binary': False,
679 'ops': {NEW_FILENODE: 'new file 100644', }
679 'ops': {NEW_FILENODE: 'new file 100644', }
680 },
680 },
681 '@@ -0,0 +1 @@\n+test_content b\n' # diff
681 '@@ -0,0 +1 @@\n+test_content b\n' # diff
682 ),
682 ),
683 (
683 (
684 'file_c', # filename
684 'file_c', # filename
685 'A', # change
685 'A', # change
686 { # stats
686 { # stats
687 'added': 1,
687 'added': 1,
688 'deleted': 0,
688 'deleted': 0,
689 'binary': False,
689 'binary': False,
690 'ops': {NEW_FILENODE: 'new file 100644', }
690 'ops': {NEW_FILENODE: 'new file 100644', }
691 },
691 },
692 '@@ -0,0 +1 @@\n+test_content c\n' # diff
692 '@@ -0,0 +1 @@\n+test_content c\n' # diff
693 ),
693 ),
694 ],
694 ],
695 ),
695 ),
696 (
696 (
697 'hg', 'hg_diff_double_file_change_double_newline.diff',
697 'hg', 'hg_diff_double_file_change_double_newline.diff',
698 [
698 [
699 (
699 (
700 'file_b', # filename
700 'file_b', # filename
701 'A', # change
701 'A', # change
702 { # stats
702 { # stats
703 'added': 1,
703 'added': 1,
704 'deleted': 0,
704 'deleted': 0,
705 'binary': False,
705 'binary': False,
706 'ops': {NEW_FILENODE: 'new file 100644', }
706 'ops': {NEW_FILENODE: 'new file 100644', }
707 },
707 },
708 '@@ -0,0 +1 @@\n+test_content b\n\n' # diff
708 '@@ -0,0 +1 @@\n+test_content b\n\n' # diff
709 ),
709 ),
710 (
710 (
711 'file_c', # filename
711 'file_c', # filename
712 'A', # change
712 'A', # change
713 { # stats
713 { # stats
714 'added': 1,
714 'added': 1,
715 'deleted': 0,
715 'deleted': 0,
716 'binary': False,
716 'binary': False,
717 'ops': {NEW_FILENODE: 'new file 100644', }
717 'ops': {NEW_FILENODE: 'new file 100644', }
718 },
718 },
719 '@@ -0,0 +1 @@\n+test_content c\n' # diff
719 '@@ -0,0 +1 @@\n+test_content c\n' # diff
720 ),
720 ),
721 ],
721 ],
722 ),
722 ),
723 (
723 (
724 'hg', 'hg_diff_four_file_change_newline.diff',
724 'hg', 'hg_diff_four_file_change_newline.diff',
725 [
725 [
726 (
726 (
727 'file', # filename
727 'file', # filename
728 'A', # change
728 'A', # change
729 { # stats
729 { # stats
730 'added': 1,
730 'added': 1,
731 'deleted': 0,
731 'deleted': 0,
732 'binary': False,
732 'binary': False,
733 'ops': {NEW_FILENODE: 'new file 100644', }
733 'ops': {NEW_FILENODE: 'new file 100644', }
734 },
734 },
735 '@@ -0,0 +1,1 @@\n+file\n' # diff
735 '@@ -0,0 +1,1 @@\n+file\n' # diff
736 ),
736 ),
737 (
737 (
738 'file2', # filename
738 'file2', # filename
739 'A', # change
739 'A', # change
740 { # stats
740 { # stats
741 'added': 1,
741 'added': 1,
742 'deleted': 0,
742 'deleted': 0,
743 'binary': False,
743 'binary': False,
744 'ops': {NEW_FILENODE: 'new file 100644', }
744 'ops': {NEW_FILENODE: 'new file 100644', }
745 },
745 },
746 '@@ -0,0 +1,1 @@\n+another line\n' # diff
746 '@@ -0,0 +1,1 @@\n+another line\n' # diff
747 ),
747 ),
748 (
748 (
749 'file3', # filename
749 'file3', # filename
750 'A', # change
750 'A', # change
751 { # stats
751 { # stats
752 'added': 1,
752 'added': 1,
753 'deleted': 0,
753 'deleted': 0,
754 'binary': False,
754 'binary': False,
755 'ops': {NEW_FILENODE: 'new file 100644', }
755 'ops': {NEW_FILENODE: 'new file 100644', }
756 },
756 },
757 '@@ -0,0 +1,1 @@\n+newline\n' # diff
757 '@@ -0,0 +1,1 @@\n+newline\n' # diff
758 ),
758 ),
759 (
759 (
760 'file4', # filename
760 'file4', # filename
761 'A', # change
761 'A', # change
762 { # stats
762 { # stats
763 'added': 1,
763 'added': 1,
764 'deleted': 0,
764 'deleted': 0,
765 'binary': False,
765 'binary': False,
766 'ops': {NEW_FILENODE: 'new file 100644', }
766 'ops': {NEW_FILENODE: 'new file 100644', }
767 },
767 },
768 '@@ -0,0 +1,1 @@\n+fil4\n\\ No newline at end of file' # diff
768 '@@ -0,0 +1,1 @@\n+fil4\n\\ No newline at end of file' # diff
769 ),
769 ),
770 ],
770 ],
771 ),
771 ),
772
772
773 ]
773 ]
774
774
775
775
776 diff_class = {
776 diff_class = {
777 'git': GitDiff,
777 'git': GitDiff,
778 'hg': MercurialDiff,
778 'hg': MercurialDiff,
779 'svn': SubversionDiff,
779 'svn': SubversionDiff,
780 }
780 }
781
781
782
782
783 @pytest.fixture(params=DIFF_FIXTURES)
783 @pytest.fixture(params=DIFF_FIXTURES)
784 def diff_fixture(request):
784 def diff_fixture(request):
785 vcs, diff_fixture, expected = request.param
785 vcs, diff_fixture, expected = request.param
786 diff_txt = fixture.load_resource(diff_fixture)
786 diff_txt = fixture.load_resource(diff_fixture)
787 diff = diff_class[vcs](diff_txt)
787 diff = diff_class[vcs](diff_txt)
788 return diff, expected
788 return diff, expected
789
789
790
790
791 def test_diff_lib(diff_fixture):
791 def test_diff_lib(diff_fixture):
792 diff, expected_data = diff_fixture
792 diff, expected_data = diff_fixture
793 diff_proc = DiffProcessor(diff)
793 diff_proc = DiffProcessor(diff)
794 diff_proc_d = diff_proc.prepare()
794 diff_proc_d = diff_proc.prepare()
795 data = [(x['filename'], x['operation'], x['stats']) for x in diff_proc_d]
795 data = [(x['filename'], x['operation'], x['stats']) for x in diff_proc_d]
796 assert expected_data == data
796 assert expected_data == data
797
797
798
798
799 @pytest.fixture(params=DIFF_FIXTURES_WITH_CONTENT)
799 @pytest.fixture(params=DIFF_FIXTURES_WITH_CONTENT)
800 def diff_fixture_w_content(request):
800 def diff_fixture_w_content(request):
801 vcs, diff_fixture, expected = request.param
801 vcs, diff_fixture, expected = request.param
802 diff_txt = fixture.load_resource(diff_fixture)
802 diff_txt = fixture.load_resource(diff_fixture)
803 diff = diff_class[vcs](diff_txt)
803 diff = diff_class[vcs](diff_txt)
804 return diff, expected
804 return diff, expected
805
805
806
806
807 def test_diff_lib_newlines(diff_fixture_w_content):
807 def test_diff_lib_newlines(diff_fixture_w_content):
808 diff, expected_data = diff_fixture_w_content
808 diff, expected_data = diff_fixture_w_content
809 diff_proc = DiffProcessor(diff)
809 diff_proc = DiffProcessor(diff)
810 diff_proc_d = diff_proc.prepare()
810 diff_proc_d = diff_proc.prepare()
811 data = [(x['filename'], x['operation'], x['stats'], x['raw_diff'])
811 data = [(x['filename'], x['operation'], x['stats'], x['raw_diff'])
812 for x in diff_proc_d]
812 for x in diff_proc_d]
813 assert expected_data == data
813 assert expected_data == data
814
815
816 @pytest.mark.parametrize('input_str', [
817 '',
818 '\n',
819 '\n\n',
820 'First\n+second',
821 'First\n+second\n',
822
823 '\n\n\n Multi \n\n\n',
824 '\n\n\n Multi beginning',
825 'Multi end \n\n\n',
826 'Multi end',
827 '@@ -0,0 +1 @@\n+test_content \n\n b\n'
828 ], ids=no_newline_id_generator)
829 def test_splitlines(input_str):
830 result = DiffProcessor.diff_splitter(input_str)
831 assert list(result) == input_str.splitlines(True)
General Comments 0
You need to be logged in to leave comments. Login now