-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsts-query.py
executable file
·718 lines (628 loc) · 22.4 KB
/
sts-query.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (c) 2022 Nick Semenkovich <[email protected]>.
# https://nick.semenkovich.com/
#
# This software is released under the MIT License:
# <https://opensource.org/licenses/mit-license.php>
#
# Source: https://github.com/semenko/sts-risk-calculator-cli
import argparse
import csv
import datetime
import os
import sys
import time
import requests
import tqdm
# Modern python please (esp for | operator, https://peps.python.org/pep-0584/)
assert sys.version_info >= (3, 9)
STS_API_URL = "https://riskcalc.sts.org/stswebriskcalc/v1/calculate/stsall"
# The required API parameters
STS_PARAMS_REQUIRED = [
"age",
"gender",
"raceasian",
"raceblack",
"racenativeam",
"racnativepacific",
"ethnicity",
"payorprim",
"payorsecond",
"surgdt",
"weightkg",
"heightcm",
"hct",
"wbc",
"platelets",
"creatlst",
"dialysis",
"hypertn",
"immsupp",
"pvd",
"cvd",
"cvdtia",
"cvdpcarsurg",
"mediastrad",
"cancer",
"fhcad",
"slpapn",
"liverdis",
"unrespstat",
"syncope",
"diabetes",
"diabctrl",
"infendo",
"infendty",
"cva",
"cvawhen",
"chrlungd",
"cvdstenrt",
"cvdstenlft",
"ivdrugab",
"alcohol",
"pneumonia",
"tobaccouse",
"hmo2",
"prcvint",
"prcab",
"prvalve",
"prvalveproc1",
"prvalveproc2",
"prvalveproc3",
"prvalveproc4",
"prvalveproc5",
"poc",
"pocint1",
"pocint2",
"pocint3",
"pocint4",
"pocint5",
"pocint6",
"pocint7",
"pocpci",
"pocpciwhen",
"pocpciin",
"miwhen",
"heartfailtmg",
"classnyh",
"cardsymptimeofadm",
"carshock",
"arrhythatrfib",
"arrhythafib",
"arrhythaflutter",
"arrhyththird",
"arrhythsecond",
"arrhythsss",
"arrhythvv",
"medinotr",
"medadp5days",
"medadpidis",
"medacei48",
"medbeta",
"medster",
"medgp",
"resusc",
"numdisv",
"stenleftmain",
"laddiststenpercent",
"hdef",
"vdstena",
"vdstenm",
"vdinsufa",
"vdinsufm",
"vdinsuft",
"vdaoprimet",
"incidenc",
"status",
"iabpwhen",
"cathbasassistwhen",
"ecmowhen",
"calculatedbmi",
"procid",
]
# An empty API query with all required parmeters defined (but empty)
STS_QUERY_STUB = {key: "" for key in STS_PARAMS_REQUIRED}
# The optional API parameters
# TODO: Add all other optional params from STS? Is the above actually exhaustive?
STS_PARAMS_OPTIONAL = ["id"] # ID is an internal parameter we allow
STS_EXPECTED_RESULTS = [
"predmort",
"predmm",
"preddeep",
"pred14d",
"predstro",
"predvent",
"predrenf",
"predreop",
"pred6d",
]
def query_sts_api(sts_query_dict):
"""
Encode a dict as .json and pass it to the STS API with a small delay.
Input: a dict of STS query parameters:
Outout: the STS results dict
"""
# The STS client side computes BMI (why?)
sts_query_dict["calculatedbmi"] = round(
(
float(sts_query_dict["weightkg"])
/ ((float(sts_query_dict["heightcm"]) / 100.0) ** 2)
),
2,
)
response = requests.post(url=STS_API_URL, json=sts_query_dict)
time.sleep(0.2)
if response.status_code != requests.codes.ok:
# This error may be thrown if your data are invalid, and we didn't catch that
# in our validation below. This might happen if you specify prior cardiac procedure types,
# which I don't fully validate yet, and you misspelled one of them.
raise Exception("STS API returned status code %d" % response.status_code)
sts_response = response.json()
assert all(
k in STS_EXPECTED_RESULTS for k in sts_response.keys()
), f"API returned an unexpected value in: {sts_response.keys()}"
# NOTE: Some values can be empty -- e.g. predrenf (at least) is sometimes blank
# I choose to only validate overall mortality, which seems to always result.
if not (0.0 <= sts_response["predmort"] <= 1.0):
print("NOTE: Odd numeric value returned by STS -- maybe your data are invalid.")
print("Please double-check the results carefully, and open a GitHub issue if this occurs.")
return response.json()
def validate_and_return_csv_data(csv_entry):
"""
Extensively validate the dict we're about to pass to the STS API.
Note that we validate ~most fields, but not all prior procedures if those are entered.
(Please open a Github issue if you find a problem with your data validation.)
Input: patient_data (dict)
Output: returns a dict that passed validation, union with the STS_QUERY_STUB
"""
# Check the required parameters are present
# assert all(key in data.keys() for key in STS_PARAMS_REQUIRED)
# Check the parameters are all allowable
provided_keys = csv_entry.keys()
valid_keys = STS_PARAMS_REQUIRED + STS_PARAMS_OPTIONAL
key_difference = set(provided_keys) - set(valid_keys)
assert (
len(key_difference) == 0
), f"You have one or more columns that are not defined STS keys: {key_difference}"
# Union of these two dicts, overwriting keys from the user supplied CSV entries where able
data = STS_QUERY_STUB | csv_entry
# Procedure
procedure_dict = {
"CAB": "1",
"AVR": "2",
"MVR": "3",
"AVR+CAB": "4",
"MVR+CAB": "5",
# There's no 6 ?!
"MVRepair": "7",
"MVRepair+CAB": "8",
}
assert data["procid"] in procedure_dict.values(), "Invalid procid"
# Key Patient Data
assert int(data["age"]) in range(1, 111), "Invalid age"
assert data["gender"] in ["Male", "Female", ""], "Invalid gender"
yes_or_empty = ["Yes", ""]
## Race/Ethnicity Parameters
assert data["raceasian"] in yes_or_empty, "Invalid raceasian"
assert data["raceblack"] in yes_or_empty, "Invalid raceblack"
assert data["racenativeam"] in yes_or_empty, "Invalid racenativeam"
# Missing an "e" in race -- on STS's end.
assert data["racnativepacific"] in yes_or_empty, "Invalid racnativepacific"
# Hispanic/latino
assert data["ethnicity"] in yes_or_empty, "Invalid ethnicity"
## Patient Metadata
payors = [
"None / self",
"Medicare (includes commercially managed options)",
"Medicaid (includes commercially managed options)",
"Commercial Health Insurance",
"Health Maintenance Organization",
"Non-U.S. Plan",
"Other",
"",
]
assert data["payorprim"] in payors, "Invalid payorprim"
assert data["payorsecond"] in payors, "Invalid payorsecond"
if data["payorsecond"] != "":
assert (
data["payorprim"] != ""
), "If payorsecond is set, payorprim must also be set"
# TODO: Double check the STS API handles 0 padding
data["surgdt"] = datetime.datetime.strptime(data["surgdt"], "%m/%d/%Y").strftime(
"%m/%d/%Y"
)
## Biometrics
# We require weight/height for BMI calc later
assert 10 <= float(data["weightkg"]) <= 250, "Invalid weightkg"
assert 20 <= float(data["heightcm"]) <= 251, "Invalid heightcm"
## Labs
assert (data["hct"] == "") or (1 <= int(data["hct"]) <= 100), "Invalid hct"
assert (data["wbc"] == "") or (0.1 <= float(data["wbc"]) <= 100), "Invalid wbc"
assert (data["platelets"] == "") or (
int(data["platelets"]) in range(1000, 900001)
), "Invalid platelets"
assert (data["creatlst"] == "") or (
0.10 <= float(data["creatlst"]) <= 30
), "Invalid creatlst"
## Comorbidities
assert data["dialysis"] in yes_or_empty, "Invalid dialysis"
assert data["hypertn"] in yes_or_empty, "Invalid hypertn"
assert data["immsupp"] in yes_or_empty, "Invalid immsupp"
assert data["pvd"] in yes_or_empty, "Invalid pvd"
assert data["cvd"] in yes_or_empty, "Invalid cvd"
assert data["cvdtia"] in yes_or_empty, "Invalid cvdtia"
assert data["cvdpcarsurg"] in yes_or_empty, "Invalid cvdpcarsurg"
if data["cvdtia"] != "" or data["cvdpcarsurg"] != "":
assert (
data["cvd"] == "Yes"
), "Invalid cvdpcarsurg, cvd must be set if a TIA or prior cartoid procedure is true."
assert data["cva"] in yes_or_empty, "Invalid cva"
assert data["cvawhen"] in ["<= 30 days", "> 30 days", ""], "Invalid when"
if data["cvawhen"] != "":
assert (
data["cva"] != ""
), "Invalid cvawhen: cva must be set to Yes if cvdwhen is defined"
assert data["mediastrad"] in yes_or_empty, "Invalid mediastrad"
assert data["cancer"] in yes_or_empty, "Invalid cancer"
assert data["fhcad"] in yes_or_empty, "Invalid fhcad"
assert data["slpapn"] in yes_or_empty, "Invalid slpapn"
assert data["liverdis"] in yes_or_empty, "Invalid liverdis"
assert data["unrespstat"] in yes_or_empty, "Invalid unrespstat"
assert data["syncope"] in yes_or_empty, "Invalid syncope"
assert data["diabetes"] in yes_or_empty, "Invalid diabetes"
assert data["diabctrl"] in [
"None",
"Diet only",
"Oral",
"Insulin",
"Other SubQ",
"Other",
"Unknown",
"",
], "Invalid diabctrl"
if data["diabctrl"] != "":
assert (
data["diabetes"] != ""
), "Invalid diabctrl: diabetes must be set if diabctrl is set"
assert data["infendo"] in yes_or_empty, "Invalid infendo"
assert data["infendty"] in ["Treated", "Active", ""], "Invalid infendty"
if data["infendty"] != "":
assert (
data["infendo"] != ""
), "Invalid infendty; infendo must be set if infendty is set"
assert data["chrlungd"] in [
"No",
"Mild",
"Moderate",
"Severe",
"Lung disease documented, severity unknown",
"Unknown",
"",
], "Invalid chrlungd"
stenosis_pct = ["50% to 79%", "80% to 99%", "100%", "Not documented", ""]
assert data["cvdstenrt"] in stenosis_pct, "Invalid cvdstenrt"
assert data["cvdstenlft"] in stenosis_pct, "Invalid cvdstenlft"
assert data["ivdrugab"] in yes_or_empty, "Invalid ivdrugab"
assert data["alcohol"] in [
"<= 1 drink/week",
"2-7 drinks/week",
">= 8 drinks/week",
"None",
"Unknown",
"",
], "Invalid alcohol"
assert data["pneumonia"] in [
"Recent",
"Remote",
"No",
"Unknown",
"",
], "Invalid pneumonia"
assert data["tobaccouse"] in [
"Never smoker",
"Current every day smoker",
"Current some day smoker",
"Smoker, current status (frequency) unknown",
"Former smoker",
"Smoking status unknown",
"",
], "Invalid tobaccouse"
assert data["hmo2"] in [
"Yes, PRN",
"Yes, oxygen dependent",
"No",
"Unknown",
"",
], "Invalid hmo2"
## Previous interventions
assert data["prcvint"] in yes_or_empty, "Invalid prcvint"
assert data["prcab"] in yes_or_empty, "Invalid prcab" # cabg
assert data["prvalve"] in yes_or_empty, "Invalid prvalve" # valve
assert data["poc"] in yes_or_empty, "Invalid poc" # other cardiac
assert data["pocpci"] in yes_or_empty, "Invalid pocpci" # pci
## Details of previous procedures are not validated yet
"""
"prvalveproc1": "",
"prvalveproc2": "",
"prvalveproc3": "",
"prvalveproc4": "",
"prvalveproc5": "",
"pocint1": "",
"pocint2": "",
"pocint3": "",
"pocint4": "",
"pocint5": "",
"pocint6": "",
"pocint7": "",
"pocpciwhen": "",
"""
assert data["pocpciin"] in ["<= 6 Hours", ">6 Hours", ""], "Invalid pocpciin"
assert data["miwhen"] in [
"<=6 Hrs",
">6 Hrs but <24 Hrs",
"1 to 7 Days",
"8 to 21 Days",
">21 Days",
"",
], "Invalid miwhen"
assert data["heartfailtmg"] in [
"Acute",
"Chronic",
"Both",
"",
], "Invalid heartfailtmg"
assert data["classnyh"] in [
"Class I",
"Class II",
"Class III",
"Class IV",
"Not documented",
"",
], "Invalid classnyh"
# Angina sx not yet validated
# "cardsymptimeofadm": "Stable Angina",
assert data["carshock"] in [
"Yes - At the time of the procedure",
"Yes, not at the time of the procedure but within prior 24 hours",
"",
], "Invalid carshock"
rhythm_onset = ["None", "Remote (> 30 days preop)", "Recent (<= 30 days preop)", ""]
assert data["arrhythatrfib"] in rhythm_onset, "Invalid arrhythatrfib"
assert data["arrhythafib"] in ["Persistent", "Paroxysmal", ""], "Invalid arrhythafib"
assert data["arrhythaflutter"] in rhythm_onset, "Invalid arrhythaflutter"
assert data["arrhyththird"] in rhythm_onset, "Invalid arrhyththird"
assert data["arrhythsecond"] in rhythm_onset, "Invalid arrhythsecond"
assert data["arrhythsss"] in rhythm_onset, "Invalid arrhythsss"
assert data["arrhythvv"] in rhythm_onset, "Invalid arrhythvv"
## Medications
assert data["medinotr"] in yes_or_empty, "Invalid medinotr" # Inotropes
# NOTE: The API allows contraindicated & unknown, which we ignore. Yes/No/Empty.
assert data["medadp5days"] in yes_or_empty, "Invalid medadp5days" # ADPi
assert (data["medadpidis"] == "") or 1 <= int(
float(data["medadpidis"])
) <= 5, "Invalid medadpidis"
assert data["medacei48"] in yes_or_empty, "Invalid medacei48" # ACE
assert data["medbeta"] in yes_or_empty, "Invalid medbeta" # BB
assert data["medster"] in yes_or_empty, "Invalid medster" # Steroids
assert data["medgp"] in yes_or_empty, "Invalid medgp" # GP2B3A
assert data["resusc"] in [
"Yes - Within 1 hour of the start of the procedure",
"Yes - More than 1 hour but less than 24 hours of the start of the procedure",
"",
], "Invalid resusc"
# Why is this not an int?!!
assert data["numdisv"] in ["None", "One", "Two", "Three", ""], "Invalid numdisv"
assert data["stenleftmain"] in ["Yes", "No", "N/A", ""], "Invalid stenleftmain"
# Left main > 50%
assert data["laddiststenpercent"] in [
"50 - 69%",
">=70%",
"",
], "Invalid laddiststenpercent"
# EF
assert (data["hdef"] == "") or (1.0 <= float(data["hdef"]) <= 99.0), "Invalid hdef"
## Valves
assert data["vdstena"] in yes_or_empty, "Invalid vdstena" # AS
assert data["vdstenm"] in yes_or_empty, "Invalid vdstenm" # MS
valve_severity = [
"Trivial/Trace",
"Mild",
"Moderate",
"Severe",
"Not documented",
"",
]
assert data["vdinsufa"] in valve_severity, "Invalid vdinsufa" # AI
assert data["vdinsufm"] in valve_severity, "Invalid vdinsufm" # MR
assert data["vdinsuft"] in valve_severity, "Invalid vdinsuft" # TR
valve_indications = [
"Bicuspid valve disease",
"Unicuspid valve disease",
"Quadricuspid valve disease",
"Congenital (other than Bicuspid, Unicuspid, or Quadricuspid)",
"Degenerative- Calcified",
"Degenerative- Leaflet prolapse with or without annular dilatation",
"Degenerative- Pure annular dilatation without leaflet prolapse",
"Degenerative - Commissural Rupture",
"Degenerative - Extensive Fenestration",
"Degenerative - Leaflet perforation / hole",
"Endocarditis, native valve with root abscess",
"Endocarditis, native valve without root abscess",
"Endocarditis, prosthetic valve with root abscess",
"Endocarditis, prosthetic valve without root abscess",
"LV Outflow Tract Pathology, HOCM",
"LV Outflow Tract Pathology, Sub-aortic membrane",
"LV Outflow Tract Pathology, Sub-aortic tunnel",
"LV Outflow Tract Pathology, Other",
"Primary Aortic Disease, Aortic Dissection",
"Primary Aortic Disease, Atherosclerotic Aneurysm",
"Primary Aortic Disease, Ehler-Danlos Syndrome",
"Primary Aortic Disease, Hypertensive Aneurysm",
"Primary Aortic Disease, Idiopathic Root dilatation",
"Primary Aortic Disease, Inflammatory",
"Primary Aortic Disease, Loeys-Dietz Syndrome",
"Primary Aortic Disease, Marfan Syndrome",
"Primary Aortic Disease, Other Connective tissue disorder",
"Radiation induced heart disease",
"Reoperation - Failure of previous AV repair or replacement",
"Rheumatic",
"Supravalvular Aortic Stenosis",
"Trauma",
"Carcinoid",
"Tumor, Myxoma",
"Tumor, Papillary Fibroelastoma",
"Tumor, Other",
"Mixed Etiology",
"Not documented",
"",
]
assert data["vdaoprimet"] in valve_indications, "Invalid vdaoprimet"
assert data["incidenc"] in [
"First cardiovascular surgery",
"First re-op cardiovascular surgery",
"Second re-op cardiovascular surgery",
"Third re-op cardiovascular surgery",
"Fourth or more re-op cardiovascular surgery",
"NA - Not a cardiovascular surgery",
"",
], "Invalid incidenc"
assert data["status"] in [
"Elective",
"Urgent",
"Emergent",
"Emergent Salvage",
"",
], "Invalid status"
assert data["iabpwhen"] in ["Preop", "Intraop", "Postop", ""], "Invalid iabpwhen"
assert data["cathbasassistwhen"] in [
"Preop",
"Intraop",
"Postop",
"",
], "Invalid cathbasassistwhen"
assert data["ecmowhen"] in [
"Preop",
"Intraop",
"Postop",
"Non-operative",
"",
], "Invalid ecmowhen"
return data
def main():
"""
Essentially all heavy lifting happens here -- the argparse parameters encode the right STS API variable names,
and then calls the minimalist query_sts_api() function to get the results.
Note that the STS API is a little odd -- it requires *all* parameters to be passed (even if they're empty)
it includes oddities (race is sometimes "rac"), it does almost nothing client side (except BMI calculation?).
If you use this code, please consider citing me and this repository.
"""
parser = argparse.ArgumentParser(
description="Query the STS Short-Term Risk Calculator (v4.2) via a CSV."
+ "\nPlease cite this repository if you're using in a publication.",
epilog="Written by Nick Semenkovich <[email protected]> https://nick.semenkovich.com",
usage="%(prog)s [options]",
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
parser.add_argument(
"--csv",
dest="csv_file",
metavar="patient-data.csv",
type=argparse.FileType("r", encoding="utf-8-sig"),
required=True,
help="Your input patient data .csv.",
)
parser.add_argument(
"--dry-run",
dest="dryrun",
action="store_true",
help="Only validate data, do not query the STS API.",
)
parser.add_argument(
"--output",
dest="output_csv_file",
metavar="results.csv",
type=str,
help="Where to store results.",
default="results.csv",
)
parser.add_argument(
"--override",
dest="override",
nargs="+",
help="Override values sent to the STS API, e.g. "
+ "make all patients the same age with --override age=50",
metavar="stsvariable=value",
)
if len(sys.argv) == 1:
parser.print_help(sys.stderr)
sys.exit(1)
args = parser.parse_args()
assert not os.path.exists(
args.output_csv_file
), f"Output file already exists: {args.output_csv_file}"
## Parse potential override values, which will take priority over anything passed in the .csv
override_dict = {}
if args.override:
print(
"NOTE: Override values supplied -- these will be sent to the STS API instead of the values in your .csv"
)
for entry in args.override:
split_entry = entry.split("=")
assert len(split_entry) == 2, "Can't handle multiple = in override value."
assert split_entry[0] != "id", "Cannot override patient ID."
override_dict[split_entry[0]] = split_entry[1]
assert all(
key in STS_PARAMS_REQUIRED + STS_PARAMS_OPTIONAL
for key in override_dict.keys()
), "Override value is not one of the defined STS keys."
print(f"\tOverriding: {override_dict}")
print("Validating CSV entries...")
# NOTE: Other than an "ID" column your CSV header must be the same as the STS API parameters,
# and your CSV entries must *exactly* match the STS query parameters.
validated_patient_data = []
csv_dictreader = csv.DictReader(args.csv_file)
errors_exist = False
for line_num, row in enumerate(csv_dictreader, start=1):
overriden_row = row | override_dict
try:
assert (row["id"]) != "", "No ID exists for this row. (Is it empty?)"
validated_patient_data.append(validate_and_return_csv_data(overriden_row))
except (AssertionError, ValueError) as error_val:
print(
f"\tError in .csv line: {line_num}, patient ID: {row['id']}: {error_val}"
)
errors_exist = True
if errors_exist:
print("Errors exist in your input .csv, unable to query STS API.")
sys.exit()
else:
print("Valid!\n")
## Actually query the STS API (if not a dry run)
if not args.dryrun:
# A dict of patient_id to the STS risk values
# e.g. '1': {pred6d: 0.37929, pred14d: 0.04021 …}
sts_results = {}
print("Querying STS API.")
# Query the API for all CSV entries
for entry in tqdm.tqdm(validated_patient_data):
patient_id = entry["id"]
# Don't send the ID to STS
del entry["id"]
assert patient_id not in sts_results, "Your patient IDs were not unique!"
sts_results[patient_id] = query_sts_api(entry)
with open(args.output_csv_file, "w") as csv_output:
writer = csv.DictWriter(
csv_output, fieldnames=["id"] + STS_EXPECTED_RESULTS
)
writer.writeheader()
for patient_id, patient_results in sts_results.items():
# Shove the ID back in for DictWriter
patient_results["id"] = patient_id
writer.writerow(patient_results)
print(f"\nDone!\nResults written to: {args.output_csv_file}")
else:
print(f"(Dry run requested, STS API not queried.)")
if __name__ == "__main__":
main()