diff --git a/.gitignore b/.gitignore --- a/.gitignore +++ b/.gitignore @@ -1,10 +1,12 @@ *.pyc *.sw? *~ +\#* +.\#* .coverage .eggs/ +resources/test/ __pycache__ version.txt swh.web.ui.egg-info docs/build/ -resources/test/webapp.ini \ No newline at end of file diff --git a/swh/web/ui/service.py b/swh/web/ui/service.py --- a/swh/web/ui/service.py +++ b/swh/web/ui/service.py @@ -157,6 +157,9 @@ to which we append paths to (hopefully) find the entry - the relative path to the entry starting from the directory pointed by directory_sha1_git + + Raises: + NotFoundExc if the directory entry is not found """ _, sha1_git_bin = query.parse_hash_with_algorithms_or_throws( directory_sha1_git, @@ -167,7 +170,8 @@ sha1_git_bin, path_string) if not queried_dir: - return None + raise NotFoundExc(('Directory entry with path %s from %s not found') % + (path_string, directory_sha1_git)) return converters.from_directory_entry(queried_dir) diff --git a/swh/web/ui/tests/views/test_browse.py b/swh/web/ui/tests/views/test_browse.py --- a/swh/web/ui/tests/views/test_browse.py +++ b/swh/web/ui/tests/views/test_browse.py @@ -316,6 +316,96 @@ mock_api.api_directory.assert_called_once_with( 'some-sha1') + @patch('swh.web.ui.views.browse.service') + @patch('swh.web.ui.views.browse.api') + @istest + def browse_directory_relative_file(self, mock_api, mock_service): + # given + stub_entry = { + 'sha256': '240', + 'type': 'file' + } + mock_service.lookup_directory_with_path.return_value = stub_entry + stub_file = { + 'sha1_git': '123', + 'sha1': '456', + 'status': 'visible', + 'data_url': '/api/1/content/890', + 'length': 42, + 'ctime': 'Thu, 01 Oct 2015 12:13:53 GMT', + 'target': 'file.txt', + 'sha256': '148' + } + mock_api.api_content_metadata.return_value = stub_file + mock_service.lookup_content_raw.return_value = { + 'data': 'this is my file'} + + # when + rv = self.client.get('/browse/directory/sha1/path/to/file/') + + # then + self.assertEqual(rv.status_code, 200) + self.assert_template_used('content.html') + self.assertIsNotNone(self.get_context_variable('content')) + content = self.get_context_variable('content') + # change caused by call to prepare_data_for_view + self.assertEqual(content['data_url'], '/browse/content/890') + self.assertEqual(content['data'], 'this is my file') + mock_api.api_content_metadata.assert_called_once_with('sha256:240') + mock_service.lookup_content_raw.assert_called_once_with('sha256:240') + + @patch('swh.web.ui.views.browse.service') + @patch('swh.web.ui.views.browse.api') + @istest + def browse_directory_relative_dir(self, mock_api, mock_service): + # given + mock_service.lookup_directory_with_path.return_value = { + 'sha256': '240', + 'target': 'abcd', + 'type': 'dir' + } + + stub_directory_ls = [ + {'type': 'dir', + 'target': '123', + 'name': 'some-dir-name'}, + {'type': 'file', + 'sha1': '654', + 'name': 'some-filename'}, + {'type': 'dir', + 'target': '987', + 'name': 'some-other-dirname'} + ] + mock_api.api_directory.return_value = stub_directory_ls + + # when + rv = self.client.get('/browse/directory/sha1/path/to/dir/') + + # then + self.assertEqual(rv.status_code, 200) + self.assert_template_used('directory.html') + self.assertIsNotNone(self.get_context_variable('files')) + self.assertEqual(len(self.get_context_variable('files')), + len(stub_directory_ls)) + mock_api.api_directory.assert_called_once_with('abcd') + + @patch('swh.web.ui.views.browse.service') + @patch('swh.web.ui.views.browse.api') + @istest + def browse_directory_relative_not_found(self, mock_api, mock_service): + # given + mock_service.lookup_directory_with_path.side_effect = NotFoundExc( + 'Directory entry not found.') + + # when + rv = self.client.get('/browse/directory/some-sha1/some/path/') + + # then + self.assertEqual(rv.status_code, 200) + self.assert_template_used('directory.html') + self.assertEqual(self.get_context_variable('message'), + 'Directory entry not found.') + @patch('swh.web.ui.views.browse.api') @patch('swh.web.ui.views.browse.utils') @istest diff --git a/swh/web/ui/views/browse.py b/swh/web/ui/views/browse.py --- a/swh/web/ui/views/browse.py +++ b/swh/web/ui/views/browse.py @@ -210,22 +210,44 @@ @app.route('/browse/directory/') @app.route('/browse/directory//') +@app.route('/browse/directory///') @set_renderers(HTMLRenderer) -def browse_directory(sha1_git): +def browse_directory(sha1_git, path=None): """Show directory information. Args: - - sha1_git: the directory's sha1 git identifier. + - sha1_git: the directory's sha1 git identifier. If path + is set, the base directory for the relative path to the entry + - path: the path to the requested entry, relative to + the directory pointed by sha1_git Returns: - The content's information at sha1_git + The content's information at sha1_git, or at sha1_git/path if + path is set. """ env = {'sha1_git': sha1_git, 'files': []} - try: - directory_files = api.api_directory(sha1_git) - env['message'] = "Listing for directory %s:" % sha1_git + if path: + env['message'] = ('Listing for directory with path %s from %s:' + % (path, sha1_git)) + dir_or_file = service.lookup_directory_with_path( + sha1_git, path) + if dir_or_file['type'] == 'file': + fsha = 'sha256:%s' % dir_or_file['sha256'] + content = api.api_content_metadata(fsha) + content_raw = service.lookup_content_raw(fsha) + if content_raw: # FIXME: currently assuming utf8 encoding + content['data'] = content_raw['data'] + env['content'] = utils.prepare_data_for_view( + content, encoding='utf-8') + return render_template('content.html', **env) + else: + directory_files = api.api_directory(dir_or_file['target']) + env['files'] = utils.prepare_data_for_view(directory_files) + else: + env['message'] = "Listing for directory %s:" % sha1_git + directory_files = api.api_directory(sha1_git) env['files'] = utils.prepare_data_for_view(directory_files) except (NotFoundExc, BadInputExc) as e: env['message'] = str(e) @@ -529,9 +551,9 @@ encoding = request.args.get('encoding', 'utf8') if encoding not in aliases: - env['message'] = 'Encoding %s not supported.' \ - 'Supported Encodings: %s' % ( - encoding, list(aliases.keys())) + env['message'] = (('Encoding %s not supported.' + 'Supported Encodings: %s') % ( + encoding, list(aliases.keys()))) return render_template('revision-directory.html', **env) try: