Python Context Managers
One of my current projects involves some improvements to code that I wrote last year – specifically enhancing the “Patient Identity Fixer”, which is my pet project of sorts. I plan on adding support for a new API that was recently released, but an improvement that I was looking forward to the most was one to boost performance by adding HTTP sessions and connection pooling. A patient with a single MRN in a single system involves at least 11 calls to the EMPI API. Each additional MRN adds 5 more calls, and some patients have a dozen MRNs from the various facilities in the Health System. Creating an HTTPS session for a single API call was taking between 600 and 1000 milliseconds – almost a full second for each API call! You can see how this adds up quickly!
Using connection pooling with the Python requests
library isn’t difficult at all, but it involves initializing a Session
object and closing it at the termination of the program. I’m sure there’s a measure of idiot-proofing within Python that will terminate any open network connections when a given program terminates, but to be a good developer, we should do our housekeeping. The easiest way to do that is with Context Managers.
What’s a context manager? If you’ve opened a File Handle object, you’ve used a Connection Manager. What’s a File Handle? Ok, let me give an example so we’re all on the same page… this with
block is a Context Manager:
with open('my_text_file') as file:
contents = file.read()
print(contents)
We use these so that we don’t have to worry about closing our file when we exit the loop, and they’re incredibly easy to implement in classes that involve things like clients. In fact, I think it was the requests
library that inspired me to make this change.
Adding a context manager involves adding two “special” instance methods to a class similar to the __init__
constructor method that we’re already familiar with. The first is __enter__
. It cannot take any parameters - as there isn’t anywhere to specify any, and its return value will be bound to the variable specified in the as
clause - that is, file
in the example above. Unless you need to do something specific when inside a context manager, this is likely just going to be a simple return self
without much else. When the context manager code block begins, the constructor is called as it would if we were assigning the part after the keyword with
to the variable defined after the keyword as
.
The function that’s likely to have more to it is the __exit__
method. In simple terms, this method is called when code execution leaves the Context Manager, so this is where you call your .close()
methods on your objects so that files can be closed, connections can be gracefully terminated, and any needed housekeeping tasks can be tidily handled. In more elaborate terms, the __exit__
method is also called when an exception occurs in the Context Manager and can opt to suppress the exception. The three required parameters to the __exit__
method are the exception’s type, value, and traceback. The __exit__
method’s return type is bool
. If the context exited without causing an exception, the value is ignored. Otherwise, a true
value indicates that the exception should be suppressed.
While there are a few more nuances, the gist is that we could also write our operation this way:
try:
file = open('my_text_file')
contents = file.read()
except:
file.close()
print(contents)
Back to my project and context managers… keeping the Session
in a class variable and initializing it in the constructor makes sense. Then all I needed to do was search-and-replace all of my calls to requests.post()
to use self.session.post()
and so on.
Bringing everything together, the class definition and special methods look like this:
class EmpiClient:
"""
Enterprise Master Patient Index (EMPI) REST API Client
"""
logger: logging.Logger
config: EmpiConfig
auth: HTTPBasicAuth
session: requests.Session
def __init__(self, config: EmpiConfig) -> None:
self.logger = logging.getLogger(__name__)
self.config = config
self.session = requests.Session()
self.session.auth = HTTPBasicAuth(self.config.user, self.config.password)
def __enter__(self):
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.session.close()
And a snippet of the code where I use this in a different class:
with EmpiClient(self.configuration.empi) as empi_client:
while patient_list:
this_source, this_id = patient_list.pop().split(':', maxsplit=1)
empi_id, corrected_patient = self.build_patient(empi_client, this_source, this_id)