Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Query programs by name and search #77

Closed
wants to merge 44 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
07460d7
query programs by name
kt474 Nov 17, 2021
85be645
refactor cache logic
kt474 Nov 17, 2021
7413496
Merge branch 'main' into query-program-name
rathishcholarajan Nov 23, 2021
3c282e1
Merge branch 'main' into query-program-name
rathishcholarajan Nov 23, 2021
d2cc513
Merge branch 'main' into query-program-name
kt474 Nov 29, 2021
f733ee0
fix lint
kt474 Nov 29, 2021
b26f544
update tests
kt474 Nov 29, 2021
17d1f47
Merge branch 'main' into query-program-name
rathishcholarajan Nov 29, 2021
610e8a0
refactor programs cache
kt474 Nov 30, 2021
ad53714
fix lint
kt474 Nov 30, 2021
7be7a77
Merge branch 'main' into query-program-name
kt474 Nov 30, 2021
bf87cae
unrelated lint fix
kt474 Nov 30, 2021
5a1d7c2
Merge branch 'query-program-name' of https://github.com/Qiskit/qiskit…
kt474 Nov 30, 2021
b9933c9
Merge branch 'main' into query-program-name
rathishcholarajan Dec 1, 2021
830ddfc
Merge branch 'main' into query-program-name
rathishcholarajan Dec 1, 2021
ebacbee
update cache logic
kt474 Dec 2, 2021
d48e892
Merge branch 'main' into query-program-name
rathishcholarajan Dec 2, 2021
0ab06e6
update cache logic
kt474 Dec 5, 2021
c188a14
Merge branch 'main' into query-program-name
kt474 Dec 5, 2021
8d287d9
Update qiskit_ibm_runtime/ibm_runtime_service.py
kt474 Dec 7, 2021
861cf00
update retrieve programs method
kt474 Dec 7, 2021
af309c7
refactor logic
kt474 Dec 8, 2021
ea04aa8
Merge branch 'main' into query-program-name
kt474 Dec 16, 2021
fca5a98
Merge branch 'main' into query-program-name
kt474 Jan 4, 2022
0bc0bca
Merge branch 'main' into query-program-name
kt474 Jan 5, 2022
6671b7d
add tests back in
kt474 Jan 5, 2022
851caa2
Merge branch 'main' into query-program-name
kt474 Jan 6, 2022
d33595e
Merge branch 'main' into query-program-name
kt474 Jan 7, 2022
5e827fe
fix list programs test
kt474 Jan 7, 2022
d0ac07e
Merge branch 'main' into query-program-name
kt474 Jan 7, 2022
2a479ae
Merge branch 'main' into query-program-name
kt474 Jan 12, 2022
079a88e
Merge branch 'main' into query-program-name
rathishcholarajan Jan 13, 2022
7186f5a
Merge branch 'main' into query-program-name
kt474 Jan 13, 2022
4bf740e
Merge branch 'main' into query-program-name
kt474 Jan 17, 2022
21ec9a2
Merge branch 'main' into query-program-name
kt474 Jan 31, 2022
fac7775
Merge branch 'query-program-name' of https://github.com/kt474/qiskit-…
kt474 Jan 31, 2022
7688814
Merge branch 'main' into query-program-name
kt474 Jan 31, 2022
162576b
Merge branch 'query-program-name' of https://github.com/kt474/qiskit-…
kt474 Jan 31, 2022
3f48cea
revert commit
kt474 Jan 31, 2022
b9746c1
Merge branch 'main' into query-program-name
daka1510 Feb 7, 2022
cbb5d63
combine qith search query
kt474 Feb 8, 2022
4b9a3eb
update reno
kt474 Feb 8, 2022
9669bd2
remove whitespace
kt474 Feb 8, 2022
764b8cb
Merge branch 'main' into query-program-name
kt474 Feb 9, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 6 additions & 3 deletions qiskit_ibm_runtime/api/clients/runtime.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,17 +45,20 @@ def __init__(
)
self._api = Runtime(self._session)

def list_programs(self, limit: int = None, skip: int = None) -> Dict[str, Any]:
def list_programs(
self, name: str = "", search: str = "", limit: int = None, skip: int = None
) -> Dict[str, Any]:
"""Return a list of runtime programs.

Args:
name: Name of the program.
search: Returns programs containing search word in name or description.
limit: The number of programs to return.
skip: The number of programs to skip.

Returns:
A list of runtime programs.
"""
return self._api.list_programs(limit, skip)
return self._api.list_programs(name, search, limit, skip)

def program_create(
self,
Expand Down
13 changes: 10 additions & 3 deletions qiskit_ibm_runtime/api/rest/runtime.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,18 +56,25 @@ def program_job(self, job_id: str) -> "ProgramJob":
"""
return ProgramJob(self.session, job_id)

def list_programs(self, limit: int = None, skip: int = None) -> Dict[str, Any]:
def list_programs(
self, name: str = "", search: str = "", limit: int = None, skip: int = None
) -> Dict[str, Any]:
"""Return a list of runtime programs.

Args:
name: Name of the program.
search: Returns programs containing search word in name or description.
limit: The number of programs to return.
skip: The number of programs to skip.

Returns:
A list of runtime programs.
"""
url = self.get_url("programs")
payload: Dict[str, int] = {}
payload: Dict[str, Union[int, str]] = {}
if name:
payload["name"] = name
if search:
payload["search"] = search
if limit:
payload["limit"] = limit
if skip:
Expand Down
100 changes: 79 additions & 21 deletions qiskit_ibm_runtime/ibm_runtime_service.py
Original file line number Diff line number Diff line change
Expand Up @@ -641,6 +641,8 @@ def pprint_programs(
self,
refresh: bool = False,
detailed: bool = False,
name: Optional[str] = "",
search: Optional[str] = "",
limit: int = 20,
skip: int = 0,
) -> None:
Expand All @@ -650,11 +652,13 @@ def pprint_programs(
refresh: If ``True``, re-query the server for the programs. Otherwise
return the cached value.
detailed: If ``True`` print all details about available runtime programs.
name: Only retrieve programs with the exact program name given.
search: Returns programs containing search word in name or description.
limit: The number of programs returned at a time. Default and maximum
value of 20.
skip: The number of programs to skip.
"""
programs = self.programs(refresh, limit, skip)
programs = self.programs(refresh, name, search, limit, skip)
for prog in programs:
print("=" * 50)
if detailed:
Expand All @@ -667,7 +671,12 @@ def pprint_programs(
print(f" Description: {prog.description}")

def programs(
self, refresh: bool = False, limit: int = 20, skip: int = 0
self,
refresh: bool = False,
name: Optional[str] = "",
search: Optional[str] = "",
limit: int = 20,
skip: int = 0,
) -> List[RuntimeProgram]:
"""Return available runtime programs.

Expand All @@ -676,35 +685,52 @@ def programs(
Args:
refresh: If ``True``, re-query the server for the programs. Otherwise
return the cached value.
name: Only retrieve programs with the exact program name given.
search: Returns programs containing search word in name or description.
limit: The number of programs returned at a time. ``None`` means no limit.
skip: The number of programs to skip.

Returns:
A list of runtime programs.
"""
already_retrieved = False
if skip is None:
skip = 0
if not self._programs or refresh:
self._programs = {}
current_page_limit = 20
offset = 0
while True:
response = self._api_client.list_programs(
limit=current_page_limit, skip=offset
)
program_page = response.get("programs", [])
# count is the total number of programs that would be returned if
# there was no limit or skip
count = response.get("count", 0)
for prog_dict in program_page:
program = self._to_program(prog_dict)
self._programs[program.program_id] = program
if len(self._programs) == count:
# Stop if there are no more programs returned by the server.
break
offset += len(program_page)
if not self._programs or (refresh and not name and not search):
self._programs = self._retrieve_programs()
already_retrieved = True
if limit is None:
limit = len(self._programs)
if name and search:
matched_programs = []
if refresh and not already_retrieved:
matched_programs = list(self._retrieve_programs(name, search).values())
else:
for program in list(self._programs.values()):
if program.name == name and (
search in program.name or search in program.description
):
matched_programs.append(program)
return matched_programs[skip : limit + skip]
elif name:
matched_programs = []
if refresh and not already_retrieved:
matched_programs = list(self._retrieve_programs(name).values())
else:
for program in list(self._programs.values()):
if program.name == name:
matched_programs.append(program)
return matched_programs[skip : limit + skip]
elif search:
matched_programs = []
if refresh and not already_retrieved:
matched_programs = list(self._retrieve_programs(search).values())
else:
for program in list(self._programs.values()):
if search in program.name or search in program.description:
matched_programs.append(program)
return matched_programs[skip : limit + skip]

return list(self._programs.values())[skip : limit + skip]

def program(self, program_id: str, refresh: bool = False) -> RuntimeProgram:
Expand Down Expand Up @@ -738,6 +764,38 @@ def program(self, program_id: str, refresh: bool = False) -> RuntimeProgram:

return self._programs[program_id]

def _retrieve_programs(
self, name: str = "", search: str = ""
) -> Dict[str, RuntimeProgram]:
"""Make an API call to fetch programs.

Args:
name: Name of the program.
search: Search query for program name and description.

Returns:
A dict of ``RuntimeProgram`` instances, keyed by program name.
"""
programs = {}
current_page_limit = 20
offset = 0
while True:
response = self._api_client.list_programs(
name=name, search=search, limit=current_page_limit, skip=offset
)
program_page = response.get("programs", [])
# count is the total number of programs that would be returned if
# there was no limit or skip
count = response.get("count", 0)
for prog_dict in program_page:
program = self._to_program(prog_dict)
programs[program.program_id] = program
if len(programs) == count:
# Stop if there are no more programs returned by the server.
break
offset += len(program_page)
return programs

def _to_program(self, response: Dict) -> RuntimeProgram:
"""Convert server response to ``RuntimeProgram`` instances.

Expand Down
8 changes: 8 additions & 0 deletions releasenotes/notes/query-program-name-22b97d6b4c5731d2.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
upgrade:
- |
The ``name`` and ``search`` parameters have been added to
:meth:`qiskit_ibm_runtime.IBMRuntimeService.programs` and
:meth:`qiskit_ibm_runtime.IBMRuntimeService.pprint_programs`.
``name`` can be used to filter by program names and ``search`` can be used to
filter by keywords in the program name and description.
39 changes: 38 additions & 1 deletion test/integration/test_program.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ class TestIntegrationProgram(IBMIntegrationTestCase):
def test_list_programs(self, service):
"""Test listing programs."""
program_id = self._upload_program(service)
programs = service.programs()
programs = service.programs(refresh=True)
self.assertTrue(programs)
found = False
for prog in programs:
Expand All @@ -57,6 +57,43 @@ def test_list_programs_with_limit_skip(self, service):
self.assertIn(all_ids[1], some_ids)
self.assertIn(all_ids[2], some_ids)

@run_integration_test
def test_filter_programs_with_program_name(self, service):
"""Test filter programs with the program name"""
program_id = self._upload_program(service, name="qiskit-test-sample")
programs = service.programs(name="qiskit-test-sample", refresh=True)
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(name="qiskit-test", refresh=True)
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_integration_test
def test_filter_programs_with_search(self, service):
"""Test filtering programs with the search parameter"""
program_id = self._upload_program(service)
programs = service.programs(search="qiskit-test", refresh=True)
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(search="qiskit-test-not", refresh=True)
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_integration_test
def test_filter_programs_with_name_and_search(self, service):
"""Test filtering programs with both search and name parameter"""
program_id = self._upload_program(service, name="qiskit-test-sample")
programs = service.programs(
search="qiskit-test", name="qiskit-test-sample", refresh=True
)
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(
search="qiskit-test", name="qiskit-test", refresh=True
)
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_integration_test
def test_list_program(self, service):
"""Test listing a single program."""
Expand Down
18 changes: 16 additions & 2 deletions test/unit/mock/fake_runtime_client.py
Original file line number Diff line number Diff line change
Expand Up @@ -270,11 +270,25 @@ def set_final_status(self, final_status):
"""Set job status to passed in final status instantly."""
self._final_status = final_status

def list_programs(self, limit, skip):
def list_programs(self, name, search, limit, skip):
"""List all programs."""
programs = []
for prog in self._programs.values():
programs.append(prog.to_dict())
if not name and not search:
programs.append(prog.to_dict())
elif name == prog.to_dict()["name"] and (
search in prog.to_dict()["name"].lower()
or search in prog.to_dict()["description"].lower()
):
programs.append(prog.to_dict())
elif name == prog.to_dict()["name"]:
programs.append(prog.to_dict())
elif (
search in prog.to_dict()["name"].lower()
or search in prog.to_dict()["description"].lower()
):
programs.append(prog.to_dict())

return {"programs": programs[skip : limit + skip], "count": len(self._programs)}

def program_create(
Expand Down
33 changes: 33 additions & 0 deletions test/unit/test_programs.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,39 @@ def test_list_programs_with_limit_skip(self, service):
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_ids[0], all_ids)

@run_legacy_and_cloud_fake
def test_filter_programs_with_program_name(self, service):
"""Test filter programs with the program name"""
program_id = upload_program(service, name="qiskit-test-sample")
programs = service.programs(name="qiskit-test-sample")
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(name="qiskit-test")
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_legacy_and_cloud_fake
def test_filter_programs_with_search(self, service):
"""Test filtering programs with the search parameter"""
program_id = upload_program(service)
programs = service.programs(search="Test program")
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(search="Test sample")
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_legacy_and_cloud_fake
def test_filter_programs_with_name_and_search(self, service):
"""Test filtering programs with both search and name parameter"""
program_id = upload_program(service, name="qiskit-test-sample")
programs = service.programs(search="qiskit-test", name="qiskit-test-sample")
all_ids = [prog.program_id for prog in programs]
self.assertIn(program_id, all_ids)
programs = service.programs(search="qiskit-test", name="qiskit-test")
all_ids = [prog.program_id for prog in programs]
self.assertNotIn(program_id, all_ids)

@run_legacy_and_cloud_fake
def test_list_program(self, service):
"""Test listing a single program."""
Expand Down