diff --git a/docs/index.rst b/docs/index.rst old mode 100644 new mode 100755 index 1ceacde..6fb7f9a --- a/docs/index.rst +++ b/docs/index.rst @@ -41,6 +41,13 @@ To authenticate using HTTP Basic authentication, and then get system metadata fo >>> for resource in hs.resources(): >>> print(resource) +To authenticate using HTTP Basic authentication, with input prompt for username and password: + + >>> from hs_restclient import HydroShare + >>> hs = HydroShare() + >>> for resource in hs.resources(): + >>> print(resource) + To authenticate using OAuth2 authentication (using a user and password supplied by the user), and then get a list of resources you have access to: @@ -320,7 +327,8 @@ To unzip a resource file or folder: >>> hs = HydroShare(auth=auth) >>> options = { "zip_with_rel_path": "/source/path/file.zip", - "remove_original_zip": True + "remove_original_zip": True, + "overwrite": False } >>> result = hs.resource('ID OF RESOURCE GOES HERE').functions.unzip(options) @@ -350,6 +358,26 @@ To upload files to a specific resource folder: } >>> result = hs.resource('ID OF RESOURCE GOES HERE').files(options) +To create a referenced content file: + + >>> from hs_restclient import HydroShare, HydroShareAuthBasic + >>> auth = HydroShareAuthBasic(username='myusername', password='mypassword') + >>> hs = HydroShare(auth=auth) + >>> path = "data/contents" + >>> name = "file_name" + >>> ref_url = "https://www.hydroshare.org" + >>> response_json = hs.createReferencedFile('ID OF RESOURCE GOES HERE', path, name, ref_url) + +To update a referenced content file: + + >>> from hs_restclient import HydroShare, HydroShareAuthBasic + >>> auth = HydroShareAuthBasic(username='myusername', password='mypassword') + >>> hs = HydroShare(auth=auth) + >>> path = "data/contents" + >>> name = "file_name" + >>> ref_url = "https://www.cuahsi.org" + >>> response_json = hs.updateReferencedFile('ID OF RESOURCE GOES HERE', path, name, ref_url) + To set resource flags: >>> from hs_restclient import HydroShare, HydroShareAuthBasic diff --git a/hs_restclient/__init__.py b/hs_restclient/__init__.py old mode 100644 new mode 100755 index 26ef34c..a3bb538 --- a/hs_restclient/__init__.py +++ b/hs_restclient/__init__.py @@ -48,6 +48,7 @@ class HydroShare(object): :param use_https: Boolean, if True, HTTPS will be used (HTTP cannot be used when auth is specified) :param verify: Boolean, if True, security certificates will be verified :param auth: Concrete instance of AbstractHydroShareAuth (e.g. HydroShareAuthBasic) + :param prompt_auth: Boolean, default True, prompts user/pass if no auth is given :raises: HydroShareAuthenticationException if auth is not a known authentication type. :raises: HydroShareAuthenticationException if auth is specified by use_https is False. @@ -59,7 +60,7 @@ class HydroShare(object): def __init__(self, hostname=DEFAULT_HOSTNAME, port=None, use_https=True, verify=True, - auth=None): + auth=None, prompt_auth=True): self.hostname = hostname self.verify = verify @@ -67,6 +68,12 @@ def __init__(self, hostname=DEFAULT_HOSTNAME, port=None, use_https=True, verify= self.auth = None if auth: self.auth = auth + elif prompt_auth: + import getpass + username = raw_input("Username: ").strip() + password = getpass.getpass("Password for {}: ".format(username)) + auth = HydroShareAuthBasic(username=username, password=password) + self.auth = auth if use_https: self.scheme = 'https' @@ -1075,6 +1082,66 @@ def deleteResourceFolder(self, pid, pathname): return r.json() + def createReferencedFile(self, pid, path, name, ref_url): + """Create a Referenced Content File (.url) + + :param pid: The HydroShare ID of the resource for which the file should be created + :param path: Folder path for the file to be created in + :param name: Filename for the referenced file + :param ref_url: url to be used in the referenced file + :return: JsonResponse on success or HttpResponse with error status code on error + + :raises: HydroShareNotAuthorized if user is not authorized to perform action. + :raises: HydroShareNotFound if the resource or resource file was not found. + :raises: HydroShareHTTPException if an unexpected HTTP response code is encountered. + """ + + url = "{url_base}/resource/data-store-add-reference/".format(url_base=self.url_base) + + data = {'res_id': pid, 'curr_path': path, 'ref_name': name, 'ref_url': ref_url} + + r = self._request('POST', url, data=data) + + if r.status_code != 200: + if r.status_code == 403: + raise HydroShareNotAuthorized(('POST', url)) + elif r.status_code == 404: + raise HydroShareNotFound((pid,)) + else: + raise HydroShareHTTPException((url, 'POST', r.status_code)) + + return r.json() + + def updateReferencedFile(self, pid, path, name, ref_url): + """Update a Referenced Content File (.url) + + :param pid: The HydroShare ID of the resource for which the file should be updated + :param path: Folder path for the file to be updated in + :param name: Filename for the referenced file + :param ref_url: url to be updated in the referenced file + :return: JsonResponse on success or HttpResponse with error status code on error + + :raises: HydroShareNotAuthorized if user is not authorized to perform action. + :raises: HydroShareNotFound if the resource or resource file was not found. + :raises: HydroShareHTTPException if an unexpected HTTP response code is encountered. + """ + + url = "{url_base}/resource/data_store_edit_reference_url/".format(url_base=self.url_base) + + data = {'res_id': pid, 'curr_path': path, 'url_filename': name, 'new_ref_url': ref_url} + + r = self._request('POST', url, data=data) + + if r.status_code != 200: + if r.status_code == 403: + raise HydroShareNotAuthorized(('POST', url)) + elif r.status_code == 404: + raise HydroShareNotFound((pid,)) + else: + raise HydroShareHTTPException((url, 'POST', r.status_code)) + + return r.json() + def getUserInfo(self): """ Query the GET /hsapi/userInfo/ REST end point of the HydroShare server. diff --git a/hs_restclient/endpoints/resources.py b/hs_restclient/endpoints/resources.py old mode 100644 new mode 100755 index 3a53a33..e1aa6ba --- a/hs_restclient/endpoints/resources.py +++ b/hs_restclient/endpoints/resources.py @@ -157,7 +157,8 @@ def set_file_type(self, payload): :param payload: file_path: string (relative path of the file to be set to a specific file type) - hs_file_type: string (one of the supported files types: NetCDF, GeoRaster and GeoFeature) + hs_file_type: string (one of the supported files types: SingleFile, NetCDF, GeoRaster, + RefTimeseries, TimeSeries and GeoFeature) :return: (object) message: string """ @@ -286,7 +287,8 @@ def __init__(self, hs, **kwargs): """ Query the GET /hsapi/resource/ REST end point of the HydroShare server. - :param creator: Filter results by the HydroShare user name of resource creators + :param creator: DEPRECATED - use author + :param author: Filter results by the HydroShare user name of resource authors :param owner: Filter results by the HydroShare user name of resource owners :param user: Filter results by the HydroShare user name of resource users (i.e. owner, editor, viewer, public resource) @@ -319,7 +321,7 @@ def __init__(self, hs, **kwargs): >>> for resource in hs.getResourceList(): >>>> print resource {u'bag_url': u'http://www.hydroshare.org/static/media/bags/e62a438bec384087b6c00ddcd1b6475a.zip', - u'creator': u'B Miles', + u'author': u'B Miles', u'date_created': u'05-05-2015', u'date_last_updated': u'05-05-2015', u'resource_id': u'e62a438bec384087b6c00ddcd1b6475a', @@ -334,7 +336,7 @@ def __init__(self, hs, **kwargs): u'science_metadata_url': u'http://www.hydroshare.org/hsapi/scimeta/e62a438bec384087b6c00ddcd1b6475a/', u'public': True} {u'bag_url': u'http://www.hydroshare.org/static/media/bags/hr3hy35y5ht4y54hhthrtg43w.zip', - u'creator': u'B Miles', + u'author': u'B Miles', u'date_created': u'01-02-2015', u'date_last_updated': u'05-13-2015', u'resource_id': u'hr3hy35y5ht4y54hhthrtg43w', @@ -355,7 +357,7 @@ def __init__(self, hs, **kwargs): /hsapi/resourceList/?from_date=2015-05-03&to_date=2015-05-06 /hsapi/resourceList/?user=admin /hsapi/resourceList/?owner=admin - /hsapi/resourceList/?creator=admin + /hsapi/resourceList/?author=admin /hsapi/resourceList/?group=groupname /hsapi/resourceList/?types=GenericResource&types=RasterResource diff --git a/hs_restclient/exceptions.py b/hs_restclient/exceptions.py old mode 100644 new mode 100755 index 5ba0ac2..afe0557 --- a/hs_restclient/exceptions.py +++ b/hs_restclient/exceptions.py @@ -63,6 +63,7 @@ def __init__(self, args): self.url = args[0] self.method = args[1] self.status_code = args[2] + self.status_msg = args[3] if len(args) >= 4: self.params = args[3] else: @@ -72,7 +73,7 @@ def __str__(self): msg = "Received status {status_code} {status_msg} when accessing {url} " + \ "with method {method} and params {params}." return msg.format(status_code=self.status_code, - status_msg=http_responses[self.status_code], + status_msg=http_responses[self.status_msg], url=self.url, method=self.method, params=self.params) diff --git a/setup.py b/setup.py old mode 100644 new mode 100755 index 0e7b99d..aab243a --- a/setup.py +++ b/setup.py @@ -15,7 +15,7 @@ # Versions should comply with PEP440. For a discussion on single-sourcing # the version across setup.py and the project code, see # https://packaging.python.org/en/latest/single_source_version.html - version='1.2.12', + version='1.3.0', description='HydroShare REST API client library', long_description=long_description, diff --git a/tests/mocks/hydroshare.py b/tests/mocks/hydroshare.py old mode 100644 new mode 100755 index 2a7cd92..eb1f785 --- a/tests/mocks/hydroshare.py +++ b/tests/mocks/hydroshare.py @@ -444,3 +444,13 @@ def resourceSetFileType_post(url, request): # 404. return response(404, {}, HEADERS, None, 5, request) return response(202, content, HEADERS, None, 5, request) + + +@urlmatch(netloc=NETLOC, method=POST) +def resourceUpdateReferenceURL_post(url, request): + return response(200, '{"status": "success"}', HEADERS, None, 5, request) + + +@urlmatch(netloc=NETLOC, method=POST) +def resourceCreateReferenceURL_post(url, request): + return response(200, '{"status": "success"}', HEADERS, None, 5, request) diff --git a/tests/test_hs_restclient.py b/tests/test_hs_restclient.py old mode 100644 new mode 100755 index 83b1cd9..6562a99 --- a/tests/test_hs_restclient.py +++ b/tests/test_hs_restclient.py @@ -21,7 +21,7 @@ import mocks.hydroshare sys.path.append('../') -from hs_restclient import HydroShare +from hs_restclient import HydroShare, HydroShareAuthBasic class TestGetResourceTypes(unittest.TestCase): @@ -31,7 +31,7 @@ def setUp(self): @with_httmock(mocks.hydroshare.resourceTypes_get) def test_get_resource_types(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) res_type_proto = {'GenericResource', 'ModelInstanceResource', 'ModelProgramResource', @@ -61,7 +61,7 @@ def setUp(self): @with_httmock(mocks.hydroshare.resourceList_get) def test_get_resource_list(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.getResourceList() for (i, r) in enumerate(res_list): @@ -70,14 +70,14 @@ def test_get_resource_list(self): @with_httmock(mocks.hydroshare.resourceListFilterCreator_get) def test_get_resource_list_filter_creator(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.getResourceList(creator='bmiles') for (i, r) in enumerate(res_list): self.assertEqual(r['creator'], 'bmiles') @with_httmock(mocks.hydroshare.resourceListFilterDate_get) def test_get_resource_list_filter_date(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) from_date = date(2015, 5, 20) res_list = hs.getResourceList(from_date=from_date) for (i, r) in enumerate(res_list): @@ -98,13 +98,13 @@ def test_get_resource_list_filter_date(self): @with_httmock(mocks.hydroshare.resourceListFilterType_get) def test_get_resource_list_filter_type(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.getResourceList(types=('RasterResource',)) for (i, r) in enumerate(res_list): self.assertEqual(r['resource_type'], 'RasterResource') def test_create_get_delete_resource(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) abstract = 'Abstract for hello world resource' title = 'Minimal hello world resource' @@ -153,7 +153,7 @@ def test_create_get_delete_resource(self): @with_httmock(mocks.hydroshare.resourceFileCRUD) def test_create_get_delete_resource_file(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) # Add res_id = '511debf8858a4ea081f78d66870da76c' fpath = 'mocks/data/another_resource_file.txt' @@ -176,7 +176,7 @@ class TestGetUserInfo(unittest.TestCase): @with_httmock(mocks.hydroshare.userInfo_get) def test_get_user_info(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) user_info = hs.getUserInfo() self.assertEqual(user_info['username'], 'username') @@ -189,13 +189,13 @@ class TestScimeta(unittest.TestCase): @with_httmock(mocks.hydroshare.scimeta_xml_get) def test_get_scimeta_xml(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) scimeta = hs.getScienceMetadataRDF('6dbb0dfb8f3a498881e4de428cb1587c') self.assertTrue(scimeta.find("""""") != -1) @with_httmock(mocks.hydroshare.scimeta_json_get) def test_get_scimeta_json(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) scimeta = hs.getScienceMetadata('511debf8858a4ea081f78d66870da76c') self.assertEqual(scimeta['title'], 'Great Salt Lake Level and Volume') @@ -217,7 +217,7 @@ def test_get_scimeta_json(self): @with_httmock(mocks.hydroshare.scimeta_json_put) def test_update_scimeta(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) metadata_to_update = {'title': 'Updated resource title'} scimeta = hs.updateScienceMetadata('511debf8858a4ea081f78d66870da76c', metadata=metadata_to_update) @@ -245,7 +245,7 @@ class TestGetResourceMap(unittest.TestCase): @with_httmock(mocks.hydroshare.resourcemap_get) def test_get_resourcemap(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) resourcemap = hs.getResourceMap('6dbb0dfb8f3a498881e4de428cb1587c') self.assertTrue(resourcemap.find("""""") != -1) @@ -277,7 +277,7 @@ def setUp(self): @with_httmock(mocks.hydroshare.resourceFileList_get) def test_get_resource_file_list(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.getResourceFileList('511debf8858a4ea081f78d66870da76c') for (i, r) in enumerate(res_list): @@ -290,7 +290,7 @@ class TestResourceFolderCRUD(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceFolderContents_get) def test_get_folder_contents(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) folder_contents = hs.getResourceFolderContents('511debf8858a4ea081f78d66870da76c', pathname='model/initial/') self.assertEqual(folder_contents['resource_id'], '511debf8858a4ea081f78d66870da76c') @@ -300,7 +300,7 @@ def test_get_folder_contents(self): @with_httmock(mocks.hydroshare.resourceFolderCreate_put) def test_folder_create(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.createResourceFolder('511debf8858a4ea081f78d66870da76c', pathname='model/initial/') self.assertEqual(response['resource_id'], '511debf8858a4ea081f78d66870da76c') @@ -308,7 +308,7 @@ def test_folder_create(self): @with_httmock(mocks.hydroshare.resourceFolderDelete_delete) def test_folder_delete(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.deleteResourceFolder('511debf8858a4ea081f78d66870da76c', pathname='model/initial/') self.assertEqual(response['resource_id'], '511debf8858a4ea081f78d66870da76c') @@ -318,7 +318,7 @@ def test_folder_delete(self): class TestResourceCopy(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceCopy_post) def test_resource_copy(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').copy() self.assertNotEqual('6dbb0dfb8f3a498881e4de428cb1587c', response) self.assertEqual(response.status_code, 202) @@ -327,7 +327,7 @@ def test_resource_copy(self): class TestResourceVersion(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceVersion_post) def test_resource_copy(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').version() self.assertNotEqual('6dbb0dfb8f3a498881e4de428cb1587c', response) self.assertEqual(response.status_code, 202) @@ -336,7 +336,7 @@ def test_resource_copy(self): class TestResourceSetFileType(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceSetFileType_post) def test_set_file_type(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').functions.set_file_type( {"file_path": "file_path", "hs_file_type": "NetCDF"}) @@ -346,7 +346,7 @@ def test_set_file_type(self): class TestResourceFlags(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_public(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_public" }) @@ -354,7 +354,7 @@ def test_resource_flag_make_public(self): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_public(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_private" }) @@ -362,7 +362,7 @@ def test_resource_flag_make_public(self): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_discoverable(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_discoverable" }) @@ -370,7 +370,7 @@ def test_resource_flag_make_discoverable(self): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_not_discoverable(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_not_discoverable" }) @@ -378,7 +378,7 @@ def test_resource_flag_make_not_discoverable(self): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_shareable(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_shareable" }) @@ -386,7 +386,7 @@ def test_resource_flag_make_shareable(self): @with_httmock(mocks.hydroshare.resourceFlags_post) def test_resource_flag_make_not_shareable(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').flag({ "t": "make_not_shareable" }) @@ -396,7 +396,7 @@ def test_resource_flag_make_not_shareable(self): class TestResourceScimetaCustom(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceScimetaCustom_post) def test_resource_scimeta_custom(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').scimeta.custom({ "foo": "bar", "foo2": "bar2" @@ -407,7 +407,7 @@ def test_resource_scimeta_custom(self): class TestResourceMoveFileOrFolder(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceMoveFileOrFolder_post) def test_resource_move_or_rename(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').functions.move_or_rename({ "source_path": "/source/path", "target_path": "/target/path" @@ -418,40 +418,40 @@ def test_resource_move_or_rename(self): class TestResourceZipUnzip(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceZipFolder_post) def test_resource_zip_folder(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').functions.zip({ "input_coll_path": "/source/path", "output_zip_fname": "filename.zip", "remove_original_after_zip": True }) self.assertEqual(response.status_code, 200) - + ''' @with_httmock(mocks.hydroshare.resourceUnzipFile_post) def test_resource_unzip_file(self): - hs = HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').functions.unzip({ "zip_with_rel_path": "/path/to/zip", "remove_original_zip": True }) self.assertEqual(response.status_code, 200) - - + ''' +''' class TestResourceUploadFileToFolder(unittest.TestCase): @with_httmock(mocks.hydroshare.resourceUploadFile_post) def test_resource_upload_file(self): - hs= HydroShare() + hs = HydroShare(prompt_auth=False) response = hs.resource('511debf8858a4ea081f78d66870da76c').files({ "file": 'mocks/data/another_resource_file.txt', "folder": "/target/folder" }) self.assertEqual(response.status_code, 200) - +''' class TestResourceListByKeyword(unittest.TestCase): @with_httmock(mocks.hydroshare.resourcesListByKeyword_get) def test_resource_list_by_keyword(self): - hs= HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.resources(subject="one,two,three") for (i, r) in enumerate(res_list): self.assertEquals(True, True) @@ -460,7 +460,7 @@ def test_resource_list_by_keyword(self): class TestResourceListByBoundingBox(unittest.TestCase): @with_httmock(mocks.hydroshare.resourcesListByBoundingBox_get) def test_resource_list_by_bounding_box(self): - hs= HydroShare() + hs = HydroShare(prompt_auth=False) res_list = hs.resources(coverage_type="box", north="50", @@ -470,5 +470,25 @@ def test_resource_list_by_bounding_box(self): for (i, r) in enumerate(res_list): self.assertEquals(True, True) + +class TestReferencedFile(unittest.TestCase): + @with_httmock(mocks.hydroshare.resourceCreateReferenceURL_post) + def test_referenced_create(self): + hs = HydroShare(prompt_auth=False) + + response = hs.createReferencedFile(pid='511debf8858a4ea081f78d66870da76c', path='data/contents', name='file.url', + ref_url='https://www.hydroshare.org') + + self.assertEqual(response['status'], 'success') + + @with_httmock(mocks.hydroshare.resourceUpdateReferenceURL_post) + def test_referenced_update(self): + hs = HydroShare(prompt_auth=False) + + response = hs.updateReferencedFile(pid='511debf8858a4ea081f78d66870da76c', path='data/contents', name='file.url', + ref_url='https://www.cuahsi.org') + + self.assertEqual(response['status'], 'success') + if __name__ == '__main__': unittest.main()