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

Fixed #107 #108 #109

Merged
merged 4 commits into from
Dec 18, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
2 changes: 1 addition & 1 deletion pepdbagent/_version.py
Original file line number Diff line number Diff line change
@@ -1 +1 @@
__version__ = "0.7.0a2"
__version__ = "0.7.0a3"
2 changes: 1 addition & 1 deletion pepdbagent/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ class AnnotationList(BaseModel):
count: int
limit: int
offset: int
results: List[AnnotationModel]
results: List[Union[AnnotationModel, None]]


class Namespace(BaseModel):
Expand Down
100 changes: 100 additions & 0 deletions pepdbagent/modules/annotation.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@ def get(
filter_by: Optional[Literal["submission_date", "last_update_date"]] = None,
filter_start_date: Optional[str] = None,
filter_end_date: Optional[str] = None,
pep_type: Optional[Literal["pep", "pop"]] = None,
) -> AnnotationList:
"""
Get project annotations.
Expand Down Expand Up @@ -77,6 +78,7 @@ def get(
[Default: filter won't be used]
:param filter_start_date: Filter start date. Format: "YYYY/MM/DD"
:param filter_end_date: Filter end date. Format: "YYYY/MM/DD". if None: present date will be used
:param pep_type: Get pep with specified type. Options: ["pep", "pop"]. Default: None, get all peps
khoroshevskyi marked this conversation as resolved.
Show resolved Hide resolved
:return: pydantic model: AnnotationList
"""
if all([namespace, name, tag]):
Expand All @@ -94,6 +96,10 @@ def get(
offset=0,
results=found_annotation,
)

if pep_type not in [None, "pep", "pop"]:
raise ValueError(f"pep_type should be one of ['pep', 'pop'], got {pep_type}")
khoroshevskyi marked this conversation as resolved.
Show resolved Hide resolved

return AnnotationList(
limit=limit,
offset=offset,
Expand All @@ -104,6 +110,7 @@ def get(
filter_by=filter_by,
filter_end_date=filter_end_date,
filter_start_date=filter_start_date,
pep_type=pep_type,
),
results=self._get_projects(
namespace=namespace,
Expand All @@ -116,6 +123,7 @@ def get(
filter_by=filter_by,
filter_end_date=filter_end_date,
filter_start_date=filter_start_date,
pep_type=pep_type,
),
)

Expand Down Expand Up @@ -222,9 +230,11 @@ def _count_projects(
filter_by: Optional[Literal["submission_date", "last_update_date"]] = None,
filter_start_date: Optional[str] = None,
filter_end_date: Optional[str] = None,
pep_type: Optional[Literal["pep", "pop"]] = None,
) -> int:
"""
Count projects. [This function is related to _find_projects]

:param namespace: namespace where to search for a project
:param search_str: search string. will be searched in name, tag and description information
:param admin: string or list of admins [e.g. "Khoroshevskyi", or ["doc_adin","Khoroshevskyi"]]
Expand All @@ -233,6 +243,8 @@ def _count_projects(
[Default: filter won't be used]
:param filter_start_date: Filter start date. Format: "YYYY:MM:DD"
:param filter_end_date: Filter end date. Format: "YYYY:MM:DD". if None: present date will be used
:param pep_type: Get pep with specified type. Options: ["pep", "pop"]. Default: None, get all peps

:return: number of found project in specified namespace
"""
if admin is None:
Expand All @@ -247,6 +259,8 @@ def _count_projects(
statement = self._add_date_filter_if_provided(
statement, filter_by, filter_start_date, filter_end_date
)
if pep_type:
statement = statement.where(Projects.pop.is_(pep_type == "pop"))
result = self._pep_db_engine.session_execute(statement).first()

try:
Expand All @@ -266,6 +280,7 @@ def _get_projects(
filter_by: Optional[Literal["submission_date", "last_update_date"]] = None,
filter_start_date: Optional[str] = None,
filter_end_date: Optional[str] = None,
pep_type: Optional[Literal["pep", "pop"]] = None,
) -> List[AnnotationModel]:
"""
Get projects by providing search string.
Expand All @@ -284,6 +299,7 @@ def _get_projects(
[Default: filter won't be used]
:param filter_start_date: Filter start date. Format: "YYYY:MM:DD"
:param filter_end_date: Filter end date. Format: "YYYY:MM:DD". if None: present date will be used
:param pep_type: Get pep with specified type. Options: ["pep", "pop"]. Default: None, get all peps
:return: list of found projects with their annotations.
"""
_LOGGER.info(f"Running annotation search: (namespace: {namespace}, query: {search_str}.")
Expand All @@ -303,6 +319,8 @@ def _get_projects(
)
statement = self._add_order_by_keyword(statement, by=order_by, desc=order_desc)
statement = statement.limit(limit).offset(offset)
if pep_type:
statement = statement.where(Projects.pop.is_(pep_type == "pop"))

id_results = self._pep_db_engine.session_execute(statement).all()

Expand Down Expand Up @@ -463,3 +481,85 @@ def get_project_number_in_namespace(
return result[0]
except IndexError:
return 0

def get_by_rp_list(
self,
registry_paths: List[str],
admin: Union[str, List[str]] = None,
) -> AnnotationList:
"""
Get project annotations by providing list of registry paths.

:param registry_paths: registry path string or list of registry paths
:param admin: list of namespaces where user is admin
:return: pydantic model: AnnotationReturnModel(
limit:
offset:
count:
result: List [AnnotationModel])
"""
admin_tuple = tuple_converter(admin)

if isinstance(registry_paths, list):
or_statement_list = []
for path in registry_paths:
try:
namespace, name, tag = registry_path_converter(path)
or_statement_list.append(
and_(
Projects.name == name,
Projects.namespace == namespace,
Projects.tag == tag,
or_(
Projects.namespace.in_(admin_tuple),
Projects.private.is_(False),
),
)
)
except RegistryPathError as err:
_LOGGER.error(str(err), registry_paths)
continue
khoroshevskyi marked this conversation as resolved.
Show resolved Hide resolved
if not or_statement_list:
_LOGGER.error("No valid registry paths were provided!")
return AnnotationList(
count=0,
limit=len(registry_paths),
offset=0,
results=[],
)

statement = select(Projects).where(or_(*or_statement_list))
anno_results = []
with Session(self._sa_engine) as session:
query_result = session.execute(statement).all()
for result in query_result:
project_obj = result[0]
annot = AnnotationModel(
namespace=project_obj.namespace,
name=project_obj.name,
tag=project_obj.tag,
is_private=project_obj.private,
description=project_obj.description,
number_of_samples=project_obj.number_of_samples,
submission_date=str(project_obj.submission_date),
last_update_date=str(project_obj.last_update_date),
digest=project_obj.digest,
pep_schema=project_obj.pep_schema,
pop=project_obj.pop,
stars_number=len(project_obj.stars_mapping),
)
anno_results.append(annot)

found_dict = {f"{r.namespace}/{r.name}:{r.tag}": r for r in anno_results}
end_results = [found_dict.get(project) for project in registry_paths]

return_len = len(anno_results)
return AnnotationList(
count=return_len,
limit=len(registry_paths),
offset=0,
results=end_results,
)

else:
return self.get_by_rp(registry_paths, admin)
6 changes: 3 additions & 3 deletions requirements/requirements-all.txt
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
sqlalchemy>=2.0.0
logmuse
peppy>=0.40.0a4
logmuse>=0.2.7
peppy>=0.40.0
ubiquerg>=0.6.2
coloredlogs>=15.0.1
pytest-mock
pydantic>=2.0
psycopg
psycopg>=3.1.15
45 changes: 45 additions & 0 deletions tests/test_pepagent.py
Original file line number Diff line number Diff line change
Expand Up @@ -508,6 +508,51 @@ def test_search_incorrect_filter_by_string(
filter_end_date=date_now.strftime("%Y/%m/%d"),
)

@pytest.mark.parametrize(
"rp_list, admin, found_number",
[
[
[
"namespace1/amendments1:default",
"namespace1/amendments2:default",
"namespace2/derive:default",
"private_test/amendments1:default",
],
"namespace1",
4,
],
[
[
"namespace1/amendments1:default",
"namespace1/amendments2:default",
"namespace2/derive:default",
"private_test/amendments1:default",
],
"private_test",
4,
],
],
)
def test_get_annotation_by_rp_list(self, initiate_pepdb_con, rp_list, admin, found_number):
result = initiate_pepdb_con.annotation.get_by_rp_list(rp_list)
assert len(result.results) == found_number

def test_get_annotation_by_rp_enpty_list(self, initiate_pepdb_con):
result = initiate_pepdb_con.annotation.get_by_rp_list([])
assert len(result.results) == 0

@pytest.mark.parametrize(
"namespace, query, found_number",
[
["namespace1", "ame", 2],
],
)
def test_search_incorrect_incorrect_pep_type(
self, initiate_pepdb_con, namespace, query, found_number
):
with pytest.raises(ValueError):
initiate_pepdb_con.annotation.get(namespace=namespace, pep_type="incorrect")


@pytest.mark.skipif(
not db_setup(),
Expand Down
Loading