diff --git a/poetry.lock b/poetry.lock index 15ddd37..aa7f8ac 100644 --- a/poetry.lock +++ b/poetry.lock @@ -405,16 +405,16 @@ requests = "^2.31.0" type = "git" url = "https://github.com/wtsi-npg/npg_porch_cli.git" reference = "0.1.0" -resolved_reference = "31e1371051d5b98a50a633d6b0d05d4fdd6a60d5" +resolved_reference = "5f463f74fb915b8a9fd87138c138fb1e1b5162aa" [[package]] name = "npg-python-lib" -version = "0.3.2" +version = "0.3.4" description = "A library of Python functions and classes common to NPG applications." optional = false python-versions = ">=3.10" files = [ - {file = "npg_python_lib-0.3.2.tar.gz", hash = "sha256:c30a9d28fd54e45eab64c4561036cf4a770b483d89b9949ecf282090b136af5a"}, + {file = "npg_python_lib-0.3.4.tar.gz", hash = "sha256:02cdd0659f91680a9ed160d538548722e070d6f0bd51548b0ea7ffd90458b6cb"}, ] [package.dependencies] @@ -426,7 +426,7 @@ test = ["black (>=24.3.0,<25)", "pytest (>=8.0,<9)", "pytest-it (>=0.1.5)"] [package.source] type = "url" -url = "https://github.com/wtsi-npg/npg-python-lib/releases/download/0.3.2/npg_python_lib-0.3.2.tar.gz" +url = "https://github.com/wtsi-npg/npg-python-lib/releases/download/0.3.4/npg_python_lib-0.3.4.tar.gz" [[package]] name = "packaging" @@ -856,4 +856,4 @@ zstd = ["zstandard (>=0.18.0)"] [metadata] lock-version = "2.0" python-versions = "^3.11" -content-hash = "69f5d8c5817f08543331022fe3b73a4072778628bf252e432a1e156923d6e559" +content-hash = "5ca0265e080baeb760d922e18416a0c9636bfe5ac37e3310d3e76e17dd4490d1" diff --git a/pyproject.toml b/pyproject.toml index c3d77c9..f811557 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -18,7 +18,7 @@ cryptography = "^43.0.3" PyYAML = "^6.0.0" npg_porch_cli = { git="https://github.com/wtsi-npg/npg_porch_cli.git", tag="0.1.0" } partisan = { url = "https://github.com/wtsi-npg/partisan/releases/download/2.13.0/partisan-2.13.0.tar.gz" } -npg-python-lib = { url = "https://github.com/wtsi-npg/npg-python-lib/releases/download/0.3.2/npg_python_lib-0.3.2.tar.gz" } +npg-python-lib = { url = "https://github.com/wtsi-npg/npg-python-lib/releases/download/0.3.4/npg_python_lib-0.3.4.tar.gz" } requests = "^2.32.0" structlog = "^24.4.0" diff --git a/src/npg_notify/data/resources/ont_event_email_template.txt b/src/npg_notify/data/resources/ont_event_email_template.txt index b9f463a..46fd2e0 100644 --- a/src/npg_notify/data/resources/ont_event_email_template.txt +++ b/src/npg_notify/data/resources/ont_event_email_template.txt @@ -1,9 +1,11 @@ -The ONT run for experiment $experiment_name, flowcell $flowcell_id has been $event. -The data are available in iRODS at the following path: +The ONT run for experiment $experiment_name, flowcell $flowcell_id has been $event. The data are available in iRODS at the following path: $path -This is an automated email from NPG. You are receiving it because you are registered -as a contact for one or more of the Studies listed below: +This is an automated email from NPG. You are receiving it because you are registered as a contact for one or more of the Studies listed below: $studies + +If you have any questions or need further assistance, please feel free to contact a Scientific Service Representative at dnap-ssr@$domain. + +NPG on behalf of DNA Pipelines. diff --git a/src/npg_notify/ont/event.py b/src/npg_notify/ont/event.py index 01b235b..c70fa78 100644 --- a/src/npg_notify/ont/event.py +++ b/src/npg_notify/ont/event.py @@ -21,6 +21,7 @@ from enum import Enum from importlib import resources from string import Template +from typing import Type from npg.cli import add_io_arguments, add_logging_arguments from npg.conf import IniData @@ -79,6 +80,9 @@ user = password = db = + + [MAIL] + domain = """ @@ -167,15 +171,23 @@ def subject(self) -> str: f"has been {self.event}" ) - def body(self, *study_ids: list[str]) -> str: + def body(self, studies: list[Study], domain: str = None) -> str: """Return the body of the email. Args: - *study_ids: The study IDs associated with the run. + studies: The studies associated with the run. + domain: A network domain name to use when sending email. The main will + be sent from mail. with @ in the From: header. """ + if domain is None: + raise ValueError("domain is required") + source = resources.files("npg_notify.data.resources").joinpath( "ont_event_email_template.txt" ) + + study_descs = [f"{s.id_study_lims} ({s.name})" for s in studies] + with resources.as_file(source) as template: with open(template) as f: t = Template(f.read()) @@ -185,7 +197,8 @@ def body(self, *study_ids: list[str]) -> str: "flowcell_id": self.flowcell_id, "path": self.path, "event": self.event, - "studies": "\n".join([*study_ids]), + "studies": "\n".join(study_descs), + "domain": domain, } ) @@ -202,6 +215,14 @@ def to_serializable(self) -> dict: def from_serializable(cls, serializable: dict): return cls(**serializable) + def __str__(self): + return ( + f"" + ) + def add_email_tasks( pipeline: Pipeline, event: EventType, reader, writer @@ -285,22 +306,30 @@ def run_email_tasks( for task in pipeline.claim(batch_size): try: np += 1 - study_ids = find_studies_for_run( + studies = find_studies_for_run( session, task.experiment_name, task.instrument_slot, task.flowcell_id ) # We are sending a single email to all contacts of all studies in the run contacts = set() - for study_id in study_ids: - c = get_study_contacts(session=session, study_id=study_id) + for study in studies: + c = get_study_contacts(session=session, study_id=study.id_study_lims) contacts.update(c) + log.info( + "Preparing email", + pipeline=pipeline, + task=task, + studies=[s.id_study_lims for s in studies], + contacts=contacts, + ) + if len(contacts) == 0: log.info( "No contacts found", pipeline=pipeline, task=task, - study_ids=study_ids, + studies=studies, ) pipeline.done(task) @@ -312,7 +341,7 @@ def run_email_tasks( domain=domain, contacts=sorted(contacts), subject=task.subject(), - content=task.body(study_ids), + content=task.body(studies), ) pipeline.done(task) @@ -341,8 +370,9 @@ def run_email_tasks( def find_studies_for_run( session: Session, experiment_name: str, instrument_slot: int, flowcell_id: str -) -> list[str]: - """Return the study IDs associated with an ONT run. +) -> list[Type[Study]]: + """Return the studies associated with an ONT run, ordered by ascending + id_study_lims. Args: session: An open MLWH DB session. @@ -351,11 +381,10 @@ def find_studies_for_run( flowcell_id: The flowcell ID. Returns: - Study IDs associated with the run. + Studies associated with the run. """ - return [ - elt - for elt, in session.query(distinct(Study.id_study_lims)) + return ( + session.query(Study) .join(OseqFlowcell) .filter( OseqFlowcell.experiment_name == experiment_name, @@ -364,7 +393,13 @@ def find_studies_for_run( ) .order_by(asc(Study.id_study_lims)) .all() - ] + ) + + +@dataclass +class EmailConfig: + domain: str + """The domain name to use when sending email. The main will be sent from mail.""" def main(): @@ -378,9 +413,15 @@ def main(): help="The 'add' action acts as a producer by sending new notification " "tasks to the Porch server. The 'run' action acts as a consumer " "by retrieving any notification tasks that have not been done and " - "running them to send notifications.", - choices=["add", "run"], - # nargs="?", + "running them to send notifications." + "" + "The remaining actions are for administrative purposes and require an admin " + "token to be set in the configuration file." + "The 'register' action registers the pipeline with the Porch server. It must " + "be run once, before any tasks can be added or run. The 'token' action " + "generates a new token for the pipeline. This token is used to authenticate " + "the pipeline with the Porch server.", + choices=["add", "run", "register", "token"], ) parser.add_argument( "--event", @@ -416,14 +457,15 @@ def main(): ) config_file = args.conf_file_path - config = IniData(Pipeline.ServerConfig).from_file(config_file, "PORCH") + pipeline_config = IniData(Pipeline.ServerConfig).from_file(config_file, "PORCH") + email_config = IniData(EmailConfig).from_file(config_file, "MAIL") pipeline = Pipeline( ContactEmail, name="ont-event-email", uri="https://github.com/wtsi/npg_notifications.git", version=npg_notify.version(), - config=config, + config=pipeline_config, ) num_processed, num_succeeded, num_errors = 0, 0, 0 @@ -437,8 +479,14 @@ def main(): elif action == "run": with get_connection(config_file, MYSQL_MLWH_CONFIG_FILE_SECTION) as session: num_processed, num_succeeded, num_errors = run_email_tasks( - pipeline, session + pipeline, session, email_config.domain ) + elif action == "register": + pipeline.register() + elif action == "token": + print(pipeline.new_token("ont-event-email")) + else: + raise ValueError(f"Unknown action: {action}") if num_errors > 0: log.error( diff --git a/tests/ont/test_generate_email.py b/tests/ont/test_generate_email.py index e07fa36..88e09bc 100644 --- a/tests/ont/test_generate_email.py +++ b/tests/ont/test_generate_email.py @@ -2,6 +2,7 @@ from pytest import mark as m +from npg_notify.db.mlwh import Study from npg_notify.ont.event import ContactEmail, EventType @@ -43,7 +44,9 @@ def test_generate_email(self): flowcell_id = "FAKE12345" path = f"/testZone/home/irods/{expt}_{slot}_{flowcell_id}" event_type = EventType.UPLOADED - studies = ["study1", "study2"] + + domain = "no-such-domain.sanger.ac.uk" + studies = [Study(name="study1"), Study(name="study2")] event = ContactEmail( experiment_name=expt, @@ -58,16 +61,19 @@ def test_generate_email(self): == f"Update: ONT run {expt} flowcell {flowcell_id} has been {event_type}" ) - study_lines = "\n".join(studies) + study_descs = [f"{s.id_study_lims} ({s.name})" for s in studies] + study_lines = "\n".join(study_descs) - assert event.body(*studies) == ( - f"The ONT run for experiment {expt}, flowcell {flowcell_id} has been {event_type}.\n" - "The data are available in iRODS at the following path:\n" + assert event.body(studies, domain=domain) == ( + f"The ONT run for experiment {expt}, flowcell {flowcell_id} has been {event_type}. The data are available in iRODS at the following path:\n" "\n" f"{path}\n" "\n" - "This is an automated email from NPG. You are receiving it because you are registered\n" - "as a contact for one or more of the Studies listed below:\n" + "This is an automated email from NPG. You are receiving it because you are registered as a contact for one or more of the Studies listed below:\n" "\n" f"{study_lines}\n" + "\n" + f"If you have any questions or need further assistance, please feel free to contact a Scientific Service Representative at dnap-ssr@{domain}.\n" + "\n" + "NPG on behalf of DNA Pipelines.\n" )