page_type | languages | products | description | urlFragment | |||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|
sample |
|
|
Creating a modern REST API with Python and Azure SQL, using Flask and Visual Studio Code |
azure-sql-db-python-rest-api |
Thanks to native JSON support, creating a REST API with Azure SQL and Python is really a matter of a few lines of code. Take a look at app.py
to easy it is!
Wondering what's the magic behind? The sample uses the well known Flask micro-framework and the flask-restful package to easily implement REST APIs. Beside that the native JSON support that Azure SQL provides does all the heavy lifting so sending data back and forth to the database is as easy as sending a JSON message.
In order to run this sample, the WideWorldImporters database is needed. Install WideWorldImporters sample database:
Restore WideWorldImporters Database
Once the sample database has been installed, you need to add some stored procedures that will be called from Python. The SQL code is available here:
./sql/WideWorldImportersUpdates.sql
If you need any help in executing the SQL script, you can find a Quickstart here: Quickstart: Use Azure Data Studio to connect and query Azure SQL database
Make sure you have Python 3.7 installed on your machine. Clone this repo in a directory on our computer and then create a virtual environment. For example:
virtualenv venv --python C:\Python37\
then activate the created virtual environment. For example, on Windows:
.\venv\Scripts\activate
and then install all the required packages:
pip install -r requirements
The connections string is not saved in the python code for security reasons, so you need to assign it to an environment variable in order to run the sample successfully. You also want to enable development environment for Flask:
Linux:
export FLASK_ENV="development"
export SQLAZURECONNSTR_WWIF="<your-connection-string>"
Windows:
$Env:FLASK_ENV="development"
$Env:SQLAZURECONNSTR_WWIF="<your-connection-string>"
Your connection string is something like:
DRIVER={ODBC Driver 17 for SQL Server};SERVER=<your-server-name>.database.windows.net;DATABASE=<your-database-name>;UID=PythonWebApp;PWD=a987REALLY#$%TRONGpa44w0rd
Just replace <your-server-name>
and <your-database-name>
with the correct values for your environment.
To run and test the Python REST API local, just run
flask run
Python will start the HTTP server and when everything is up and running you'll see something like
* Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)
Using a REST Client (like Insomnia, Postman or curl), you can now call your API, for example:
curl -X GET http://localhost:5000/customer/123
and you'll get info on Customer 123:
[
{
"CustomerID": 123,
"CustomerName": "Tailspin Toys (Roe Park, NY)",
"PhoneNumber": "(212) 555-0100",
"FaxNumber": "(212) 555-0101",
"WebsiteURL": "http://www.tailspintoys.com/RoePark",
"Delivery": {
"AddressLine1": "Shop 219",
"AddressLine2": "528 Persson Road",
"PostalCode": "90775"
}
}
]
Check out more samples to test all implemented verbs here:
Debugging from Visual Studio Code is fully supported. Make sure you create an .env
file the look like the following one (making sure you add your connection string)
FLASK_ENV="development"
SQLAZURECONNSTR_WWIF=""
and you'll be good to go.
Now that your REST API solution is ready, it's time to deploy it on Azure so that anyone can take advantage of it. A detailed article on how you can that that is here:
- Deploying Python web apps to Azure App Services
- Quickstart: Create a Python app in Azure App Service on Linux
The only thing you have do in addition to what explained in the above articles is to add the connection string to the Azure Web App configuration. Using AZ CLI, for example:
appName="azure-sql-db-python-rest-api"
resourceGroup="my-resource-group"
az webapp config connection-string set \
-g $resourceGroup \
-n $appName \
--settings WWIF=$SQLAZURECONNSTR_WWIF \
--connection-string-type=SQLAzure
Just make sure you correctly set $appName
and $resourceGroup
to match your environment and also that the variable $SQLAZURECONNSTR_WWIF
as also been set, as mentioned in section "Run sample locally". An example of a full script that deploy the REST API is available here: azure-deploy.sh
.
Please note that connection string are accessible as environment variables from Python when running on Azure, but they are prefixed as documented here:
https://docs.microsoft.com/en-us/azure/app-service/configure-common#connection-strings
That's why the Python code in the sample look for SQLAZURECONNSTR_WWIF
but the Shell script write the WWIF
connection string name.
As per best practices, code implement a retry logic to make sure connections to Azure SQL are resilient and che nicely handle those cases in which the database may not be available. One of these case is when database is being scale up or down. This is usually a pretty fast operation (with Azure SQL Hyperscale it happens in something around 10 seconds), but still graceful management of connection is needed.
The sample uses the Tenacity library to implement a simple retry-logic in case the error "Communication link failure" happens (see ODBC Error Codes).
If you need more details aside the general error generated by the ODBC call, you can take a look at the detailed errors that Azure SQL will return here: Troubleshooting connectivity issues and other errors with Microsoft Azure SQL Database
To test connection resiliency you can using testing tools like Locust.io, K6 or Apache JMeter. IF you want something really simple to try right away, a while loop will do. For example:
while :; do curl -s -X GET http://localhost:5000/customer/$((RANDOM % 1000)); sleep 1; done
To get the best performances, Connection Pooling should be used so that each time the code tries to open and close a connection, it can just take an existing connection from the pool, reset it, and use that one. Using a connection from the pool is way less expensive than creating a new connection, so by using connection pooling performance can be greatly improved.
Unfortunately as of today (March 2020) ODBC connection pooling in Linux does work as expected to due an issue in unixODBC library. To work around that, I had to implement a manual technique to pool connection, that you can find in the ConnectionManager
class.
Once the issue will be fixed, or if you are using Windows, you can completely remove that part of the code, and just open and close the connection as you would do normally, as connection pooling will be used automatically behind the scenes.
If you're new to Python and want to learn more, there is a full free Python curse here:
It will teach you not only how to use Python, but also how to take advantage the a great editor like Visual Studio Code.
Flask is a very common (and amazing!) framework. Learn how to use it right from Visual Studio Code with this tutorial:
Flask Tutorial in Visual Studio Code
This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.opensource.microsoft.com.
When you submit a pull request, a CLA bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., status check, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA.
This project has adopted the Microsoft Open Source Code of Conduct. For more information see the Code of Conduct FAQ or contact opencode@microsoft.com with any additional questions or comments.