views.py
476 lines
| 19.4 KiB
| text/x-python
|
PythonLexer
r5053 | ||||
r1900 | ||||
r4306 | # Copyright (C) 2016-2020 RhodeCode GmbH | |||
r1900 | # | |||
# This program is free software: you can redistribute it and/or modify | ||||
# it under the terms of the GNU Affero General Public License, version 3 | ||||
# (only), as published by the Free Software Foundation. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU Affero General Public License | ||||
# along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
# | ||||
# This program is dual-licensed. If you wish to learn more about the | ||||
# RhodeCode Enterprise Edition, including its added features, Support services, | ||||
# and proprietary license terms, please see https://rhodecode.com/licenses/ | ||||
import os | ||||
import logging | ||||
r4038 | import datetime | |||
r1900 | ||||
from pyramid.renderers import render_to_response | ||||
from rhodecode.apps._base import BaseAppView | ||||
r4038 | from rhodecode.lib.celerylib import run_task, tasks | |||
from rhodecode.lib.utils2 import AttributeDict | ||||
from rhodecode.model.db import User | ||||
from rhodecode.model.notification import EmailNotificationModel | ||||
r1900 | ||||
log = logging.getLogger(__name__) | ||||
class DebugStyleView(BaseAppView): | ||||
r4500 | ||||
r1900 | def load_default_context(self): | |||
c = self._get_local_tmpl_context() | ||||
return c | ||||
def index(self): | ||||
c = self.load_default_context() | ||||
c.active = 'index' | ||||
return render_to_response( | ||||
'debug_style/index.html', self._get_template_context(c), | ||||
request=self.request) | ||||
r4038 | def render_email(self): | |||
c = self.load_default_context() | ||||
email_id = self.request.matchdict['email_id'] | ||||
c.active = 'emails' | ||||
pr = AttributeDict( | ||||
pull_request_id=123, | ||||
title='digital_ocean: fix redis, elastic search start on boot, ' | ||||
'fix fd limits on supervisor, set postgres 11 version', | ||||
description=''' | ||||
Check if we should use full-topic or mini-topic. | ||||
- full topic produces some problems with merge states etc | ||||
- server-mini-topic needs probably tweeks. | ||||
''', | ||||
repo_name='foobar', | ||||
source_ref_parts=AttributeDict(type='branch', name='fix-ticket-2000'), | ||||
target_ref_parts=AttributeDict(type='branch', name='master'), | ||||
) | ||||
r4500 | ||||
r4038 | target_repo = AttributeDict(repo_name='repo_group/target_repo') | |||
source_repo = AttributeDict(repo_name='repo_group/source_repo') | ||||
user = User.get_by_username(self.request.GET.get('user')) or self._rhodecode_db_user | ||||
r4120 | # file/commit changes for PR update | |||
commit_changes = AttributeDict({ | ||||
'added': ['aaaaaaabbbbb', 'cccccccddddddd'], | ||||
'removed': ['eeeeeeeeeee'], | ||||
}) | ||||
r4500 | ||||
r4120 | file_changes = AttributeDict({ | |||
'added': ['a/file1.md', 'file2.py'], | ||||
'modified': ['b/modified_file.rst'], | ||||
'removed': ['.idea'], | ||||
}) | ||||
r4276 | ||||
exc_traceback = { | ||||
'exc_utc_date': '2020-03-26T12:54:50.683281', | ||||
'exc_id': 139638856342656, | ||||
'exc_timestamp': '1585227290.683288', | ||||
'version': 'v1', | ||||
'exc_message': 'Traceback (most recent call last):\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/tweens.py", line 41, in excview_tween\n response = handler(request)\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/router.py", line 148, in handle_request\n registry, request, context, context_iface, view_name\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/view.py", line 667, in _call_view\n response = view_callable(context, request)\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/config/views.py", line 188, in attr_view\n return view(context, request)\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/config/views.py", line 214, in predicate_wrapper\n return view(context, request)\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/viewderivers.py", line 401, in viewresult_to_response\n result = view(context, request)\n File "/nix/store/s43k2r9rysfbzmsjdqnxgzvvb7zjhkxb-python2.7-pyramid-1.10.4/lib/python2.7/site-packages/pyramid/viewderivers.py", line 132, in _class_view\n response = getattr(inst, attr)()\n File "/mnt/hgfs/marcink/workspace/rhodecode-enterprise-ce/rhodecode/apps/debug_style/views.py", line 355, in render_email\n template_type, **email_kwargs.get(email_id, {}))\n File "/mnt/hgfs/marcink/workspace/rhodecode-enterprise-ce/rhodecode/model/notification.py", line 402, in render_email\n body = email_template.render(None, **_kwargs)\n File "/mnt/hgfs/marcink/workspace/rhodecode-enterprise-ce/rhodecode/lib/partial_renderer.py", line 95, in render\n return self._render_with_exc(tmpl, args, kwargs)\n File "/mnt/hgfs/marcink/workspace/rhodecode-enterprise-ce/rhodecode/lib/partial_renderer.py", line 79, in _render_with_exc\n return render_func.render(*args, **kwargs)\n File "/nix/store/dakh34sxz4yfr435c0cwjz0sd6hnd5g3-python2.7-mako-1.1.0/lib/python2.7/site-packages/mako/template.py", line 476, in render\n return runtime._render(self, self.callable_, args, data)\n File "/nix/store/dakh34sxz4yfr435c0cwjz0sd6hnd5g3-python2.7-mako-1.1.0/lib/python2.7/site-packages/mako/runtime.py", line 883, in _render\n **_kwargs_for_callable(callable_, data)\n File "/nix/store/dakh34sxz4yfr435c0cwjz0sd6hnd5g3-python2.7-mako-1.1.0/lib/python2.7/site-packages/mako/runtime.py", line 920, in _render_context\n _exec_template(inherit, lclcontext, args=args, kwargs=kwargs)\n File "/nix/store/dakh34sxz4yfr435c0cwjz0sd6hnd5g3-python2.7-mako-1.1.0/lib/python2.7/site-packages/mako/runtime.py", line 947, in _exec_template\n callable_(context, *args, **kwargs)\n File "rhodecode_templates_email_templates_base_mako", line 63, in render_body\n File "rhodecode_templates_email_templates_exception_tracker_mako", line 43, in render_body\nAttributeError: \'str\' object has no attribute \'get\'\n', | ||||
'exc_type': 'AttributeError' | ||||
} | ||||
r4500 | ||||
r4038 | email_kwargs = { | |||
'test': {}, | ||||
r4500 | ||||
r4038 | 'message': { | |||
'body': 'message body !' | ||||
}, | ||||
r4500 | ||||
r4038 | 'email_test': { | |||
'user': user, | ||||
'date': datetime.datetime.now(), | ||||
}, | ||||
r4500 | ||||
r4634 | 'update_available': { | |||
'current_ver': '4.23.0', | ||||
'latest_ver': '4.24.0', | ||||
}, | ||||
r4276 | 'exception': { | |||
'email_prefix': '[RHODECODE ERROR]', | ||||
'exc_id': exc_traceback['exc_id'], | ||||
'exc_url': 'http://server-url/{}'.format(exc_traceback['exc_id']), | ||||
'exc_type_name': 'NameError', | ||||
'exc_traceback': exc_traceback, | ||||
}, | ||||
r4500 | ||||
r4038 | 'password_reset': { | |||
'password_reset_url': 'http://example.com/reset-rhodecode-password/token', | ||||
'user': user, | ||||
'date': datetime.datetime.now(), | ||||
'email': 'test@rhodecode.com', | ||||
'first_admin_email': User.get_first_super_admin().email | ||||
}, | ||||
r4500 | ||||
r4038 | 'password_reset_confirmation': { | |||
'new_password': 'new-password-example', | ||||
'user': user, | ||||
'date': datetime.datetime.now(), | ||||
'email': 'test@rhodecode.com', | ||||
'first_admin_email': User.get_first_super_admin().email | ||||
}, | ||||
r4500 | ||||
r4038 | 'registration': { | |||
'user': user, | ||||
'date': datetime.datetime.now(), | ||||
}, | ||||
'pull_request_comment': { | ||||
'user': user, | ||||
'status_change': None, | ||||
'status_change_type': None, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://localhost/pr1', | ||||
'pr_comment_url': 'http://comment-url', | ||||
r4050 | 'pr_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | ||||
'comment_file': None, | ||||
'comment_line': None, | ||||
'comment_type': 'note', | ||||
'comment_body': 'This is my comment body. *I like !*', | ||||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'markdown', | |||
'mention': True, | ||||
}, | ||||
r4500 | ||||
r4038 | 'pull_request_comment+status': { | |||
'user': user, | ||||
'status_change': 'approved', | ||||
'status_change_type': 'approved', | ||||
'pull_request': pr, | ||||
'pull_request_commits': [], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://localhost/pr1', | ||||
'pr_comment_url': 'http://comment-url', | ||||
r4050 | 'pr_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | ||||
'comment_type': 'todo', | ||||
'comment_file': None, | ||||
'comment_line': None, | ||||
'comment_body': ''' | ||||
I think something like this would be better | ||||
```py | ||||
r4377 | // markdown renderer | |||
r4038 | ||||
def db(): | ||||
global connection | ||||
return connection | ||||
``` | ||||
''', | ||||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'markdown', | |||
'mention': True, | ||||
}, | ||||
r4500 | ||||
r4038 | 'pull_request_comment+file': { | |||
'user': user, | ||||
'status_change': None, | ||||
'status_change_type': None, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://localhost/pr1', | ||||
'pr_comment_url': 'http://comment-url', | ||||
r4050 | 'pr_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | ||||
r4327 | 'comment_file': 'rhodecode/model/get_flow_commits', | |||
r4038 | 'comment_line': 'o1210', | |||
'comment_type': 'todo', | ||||
'comment_body': ''' | ||||
I like this ! | ||||
r4377 | But please check this code | |||
.. code-block:: javascript | ||||
// THIS IS RST CODE | ||||
this.createResolutionComment = function(commentId) { | ||||
// hide the trigger text | ||||
$('#resolve-comment-{0}'.format(commentId)).hide(); | ||||
var comment = $('#comment-'+commentId); | ||||
var commentData = comment.data(); | ||||
if (commentData.commentInline) { | ||||
r4543 | this.createComment(comment, f_path, line_no, commentId) | |||
r4377 | } else { | |||
Rhodecode.comments.createGeneralComment('general', "$placeholder", commentId) | ||||
} | ||||
return false; | ||||
}; | ||||
r4038 | ||||
This should work better ! | ||||
''', | ||||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'rst', | |||
'mention': True, | ||||
}, | ||||
r4120 | 'pull_request_update': { | |||
'updating_user': user, | ||||
'status_change': None, | ||||
'status_change_type': None, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://localhost/pr1', | ||||
# update comment links | ||||
'pr_comment_url': 'http://comment-url', | ||||
'pr_comment_reply_url': 'http://comment-url#reply', | ||||
'ancestor_commit_id': 'f39bd443', | ||||
'added_commits': commit_changes.added, | ||||
'removed_commits': commit_changes.removed, | ||||
'changed_files': (file_changes.added + file_changes.modified + file_changes.removed), | ||||
'added_files': file_changes.added, | ||||
'modified_files': file_changes.modified, | ||||
'removed_files': file_changes.removed, | ||||
}, | ||||
r4038 | 'cs_comment': { | |||
'user': user, | ||||
'commit': AttributeDict(idx=123, raw_id='a'*40, message='Commit message'), | ||||
'status_change': None, | ||||
'status_change_type': None, | ||||
'commit_target_repo_url': 'http://foo.example.com/#comment1', | ||||
'repo_name': 'test-repo', | ||||
'comment_type': 'note', | ||||
'comment_file': None, | ||||
'comment_line': None, | ||||
'commit_comment_url': 'http://comment-url', | ||||
r4050 | 'commit_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | 'comment_body': 'This is my comment body. *I like !*', | |||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'markdown', | |||
'mention': True, | ||||
}, | ||||
r4500 | ||||
r4038 | 'cs_comment+status': { | |||
'user': user, | ||||
'commit': AttributeDict(idx=123, raw_id='a' * 40, message='Commit message'), | ||||
'status_change': 'approved', | ||||
'status_change_type': 'approved', | ||||
'commit_target_repo_url': 'http://foo.example.com/#comment1', | ||||
'repo_name': 'test-repo', | ||||
'comment_type': 'note', | ||||
'comment_file': None, | ||||
'comment_line': None, | ||||
'commit_comment_url': 'http://comment-url', | ||||
r4050 | 'commit_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | 'comment_body': ''' | |||
Hello **world** | ||||
This is a multiline comment :) | ||||
- list | ||||
- list2 | ||||
''', | ||||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'markdown', | |||
'mention': True, | ||||
}, | ||||
r4500 | ||||
r4038 | 'cs_comment+file': { | |||
'user': user, | ||||
'commit': AttributeDict(idx=123, raw_id='a' * 40, message='Commit message'), | ||||
'status_change': None, | ||||
'status_change_type': None, | ||||
'commit_target_repo_url': 'http://foo.example.com/#comment1', | ||||
'repo_name': 'test-repo', | ||||
'comment_type': 'note', | ||||
'comment_file': 'test-file.py', | ||||
'comment_line': 'n100', | ||||
'commit_comment_url': 'http://comment-url', | ||||
r4050 | 'commit_comment_reply_url': 'http://comment-url#reply', | |||
r4038 | 'comment_body': 'This is my comment body. *I like !*', | |||
r4050 | 'comment_id': 2048, | |||
r4038 | 'renderer_type': 'markdown', | |||
'mention': True, | ||||
}, | ||||
r4500 | ||||
r4038 | 'pull_request': { | |||
'user': user, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [ | ||||
('472d1df03bf7206e278fcedc6ac92b46b01c4e21', '''\ | ||||
r4500 | my-account: moved email closer to profile as it's similar data just moved outside. | |||
'''), | ||||
('cbfa3061b6de2696c7161ed15ba5c6a0045f90a7', '''\ | ||||
users: description edit fixes | ||||
- tests | ||||
- added metatags info | ||||
'''), | ||||
], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://code.rhodecode.com/_pull-request/123', | ||||
'user_role': 'reviewer', | ||||
}, | ||||
'pull_request+reviewer_role': { | ||||
'user': user, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [ | ||||
('472d1df03bf7206e278fcedc6ac92b46b01c4e21', '''\ | ||||
r4038 | my-account: moved email closer to profile as it's similar data just moved outside. | |||
'''), | ||||
('cbfa3061b6de2696c7161ed15ba5c6a0045f90a7', '''\ | ||||
users: description edit fixes | ||||
- tests | ||||
- added metatags info | ||||
'''), | ||||
], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://code.rhodecode.com/_pull-request/123', | ||||
r4500 | 'user_role': 'reviewer', | |||
}, | ||||
'pull_request+observer_role': { | ||||
'user': user, | ||||
'pull_request': pr, | ||||
'pull_request_commits': [ | ||||
('472d1df03bf7206e278fcedc6ac92b46b01c4e21', '''\ | ||||
my-account: moved email closer to profile as it's similar data just moved outside. | ||||
'''), | ||||
('cbfa3061b6de2696c7161ed15ba5c6a0045f90a7', '''\ | ||||
users: description edit fixes | ||||
- tests | ||||
- added metatags info | ||||
'''), | ||||
], | ||||
'pull_request_target_repo': target_repo, | ||||
'pull_request_target_repo_url': 'http://target-repo/url', | ||||
'pull_request_source_repo': source_repo, | ||||
'pull_request_source_repo_url': 'http://source-repo/url', | ||||
'pull_request_url': 'http://code.rhodecode.com/_pull-request/123', | ||||
'user_role': 'observer' | ||||
r4038 | } | |||
} | ||||
template_type = email_id.split('+')[0] | ||||
r4447 | (c.subject, c.email_body, c.email_body_plaintext) = EmailNotificationModel().render_email( | |||
r4038 | template_type, **email_kwargs.get(email_id, {})) | |||
test_email = self.request.GET.get('email') | ||||
if test_email: | ||||
recipients = [test_email] | ||||
run_task(tasks.send_email, recipients, c.subject, | ||||
c.email_body_plaintext, c.email_body) | ||||
if self.request.matched_route.name == 'debug_style_email_plain_rendered': | ||||
template = 'debug_style/email_plain_rendered.mako' | ||||
else: | ||||
template = 'debug_style/email.mako' | ||||
return render_to_response( | ||||
template, self._get_template_context(c), | ||||
request=self.request) | ||||
r1900 | def template(self): | |||
t_path = self.request.matchdict['t_path'] | ||||
c = self.load_default_context() | ||||
c.active = os.path.splitext(t_path)[0] | ||||
r2103 | c.came_from = '' | |||
r4500 | # NOTE(marcink): extend the email types with variations based on data sets | |||
r4038 | c.email_types = { | |||
'cs_comment+file': {}, | ||||
'cs_comment+status': {}, | ||||
'pull_request_comment+file': {}, | ||||
'pull_request_comment+status': {}, | ||||
r4120 | ||||
'pull_request_update': {}, | ||||
r4500 | ||||
'pull_request+reviewer_role': {}, | ||||
'pull_request+observer_role': {}, | ||||
r4038 | } | |||
c.email_types.update(EmailNotificationModel.email_types) | ||||
r1900 | ||||
return render_to_response( | ||||
'debug_style/' + t_path, self._get_template_context(c), | ||||
r4038 | request=self.request) | |||