Within the field of web development, APIs (Application Programming Interfaces) are essential for facilitating communication across various software components. Because of its simplicity, scalability, and statelessness, REST (Representational State Transfer) APIs stand out among the other types of APIs. In this blog, we will dive into basics of building a REST API.
What is a REST API?
REST is an architectural style that outlines a collection of guidelines for building web services. A REST API offers a straightforward and adaptable method to interact with web services without the need for any processing. REST is often favored for its lower bandwidth usage, simplicity, and flexibility, making it better suited for internet applications. REST is utilized to retrieve or transmit data from a web service with all interactions through a REST API occurring via HTTP requests.
Key Principles of REST
The four key principles of REST API design, derived from the broader principles of REST architecture, are:
1. Stateless: Every client request to the server must include all necessary information for comprehension and processing. The server should not retain any data about the client's latest HTTP request.
2. Client-Server Architecture: The client handles the user interface and experience, while the server manages request processing and resource management.
3. Cacheable: Clients can cache responses from the server, enhancing performance by reducing the server load and minimizing client-server interactions.
4. Uniform Interface: The API has a consistent interface, which simplifies the architecture and enhances visibility of interactions.
REST API Architecture
REST APIs utilize a client-server architecture, communicating over HTTP. The client sends requests to the server, which processes them and sends back responses. Communication is stateless, requiring each client request to contain all necessary information for processing.
Components of REST API Architecture
Resources: In REST, resources are identified by URIs, representing anything that can be named and addressed, such as documents, images, temporal services, collections of other resources, non-virtual objects, and more.
HTTP Methods: REST APIs utilize standard HTTP methods for operations on resources. Common methods include GET (to retrieve a resource), POST (to create a new resource), PUT (to update a resource), and DELETE (to remove a resource).
Representations: Resources are represented in a format easily understood by both client and server, commonly JSON (JavaScript Object Notation) or XML (eXtensible Markup Language).
Understanding Endpoints in REST API
Endpoints are the points of interaction between the client and the server in a RESTful API. Each endpoint is associated with a specific URI through which an API is accessed, and it defines a specific URI at which the resource can be accessed or manipulated. Endpoints are crucial for REST APIs as they provide a structured and standardized way for clients to interact with the API's resources.
Characteristics of Endpoints
- URI Structure: Endpoints are defined by their URI structure, which typically includes the base URI of the API and the path to the resource.
- HTTP Methods: Each endpoint is associated with one or more HTTP methods (GET, POST, PUT, DELETE, etc.), which define the type of operation that can be performed on the resource.
- Resource Representation: The data returned by an endpoint is typically in a structured format, such as JSON or XML, which represents the resource or resources associated with the endpoint.
Implementing a REST API
Now, let's start to build a REST API. Before we start, make sure you have Python installed on your system. You will need a minimum Python version of 3.6. You can download Python from python.org. We will also need Flask, which you can install using pip:
pip install Flask
We will build a REST API for managing a list of tasks. Each task will have a title and a status indicating whether it's completed or not.
Create a new Python file:
app.py
Import Flask and initialize the app:
from flask import Flask, jsonify, request
app = Flask(__name__)
- Define the data structure for our tasks. For simplicity, we'll use a list of dictionaries:
tasks = [
{"id": 1, "title": "Buy groceries", "completed": False},
{"id": 2, "title": "Clean the house", "completed": True}
]
GET Requests
A GET request is used to retrieve data from a server. We can get all the available data or specific data using a GET request.
- To retrieve all tasks:
@app.route("/tasks", methods=["GET"])
def get_tasks():
return jsonify({"tasks": tasks}), 200
- The
@app.route
decorator provided by Flask binds a URL pattern to the function that follows it. In this case, the URL pattern in/tasks
. -
methods=["GET]
specifies that this route should respond to the HTTP GET requests. This means that when a client sends a GET request to the/tasks
URL, Flask will call the associated function. -
get_tasks()
function will return a JSON response containing all tasks. jsonify({"tasks": tasks})
is a Flask function that converts the Python dictionary{"tasks": tasks}
into a JSON response.To retrieve a specific task:
@app.route("/tasks/<int:task_id>", methods=["GET"])
def get_task(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
return jsonify({"task": task}), 200
-
@app.route
binds a URL pattern/tasks/<int:task_id>
to the function that follows it. -
<int:task_id>
is a variable part of the URL that will be captured and passed to the function as an argument. -
methods=["GET"]
specifies that this route should respond to the HTTP GET requests. This means that when a client sends a GET request to thetasks/<int:task_id>
URL, Flask will call the associated function. -
get_task()
function will taketask_id
as the argument that is extracted from/tasks/<int:task_id>
URL. The function will then search for a task with that ID in thetasks
list. If a task is found, it will return the task in JSON format. If no task is found, it will return a404
status code with an error message. -
return jsonify({"task": task})
returns a JSON response containing the task.
POST Request
A POST request is used to send data to a server to create a new resource. To create a new task, implement a route that handles POST requests:
@app.route("/tasks", methods=["POST"])
def create_task():
new_task = request.get_json()
tasks.append(new_task)
return jsonify({"task": new_task}), 201
-
@app.route
binds a URL pattern/tasks
to the function that follows it. -
methods=["POST"]
specifies that this route should respond to the HTTP POST requests. This means when a client sends a POST request to the/tasks
URL, Flask will call the associated function. -
request.get_json()
is a Flask function that parses the incoming request data as JSON. This function is used to extract the JSON payload sent by the client in the POST request. -
create_task()
function will parse the JSON payload, add the new task to thetasks
list, and return a JSON response containing the newly created task. The status code201
indicates that the task was successfully created. -
tasks.append(new_task)
appends thenew_task
dictionary to the tasks list. -
jsonify({"task": new_task})
is a Flask function that converts thenew_task
dictionary into a JSON response. This response includes the newly created task.
PUT Request
A PUT request is used to update an existing resource on a server. To update a task, implement a route that handles PUT requests:
@app.route("/tasks/<int:task_id>", methods=["PUT"])
def update_task(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
task.update(request.get_json())
return jsonify({"task": task}), 201
-
@app.route
binds a URL pattern/tasks/<int:task_id>
to the function that follows it. -
<int:task_id>
is a variable part of the URL that will be captured and passed to the function as an argument. -
methods=["PUT"]
specifies that this route should respond to the HTTP PUT requests. This means that when a client sends a PUT request to the/tasks/<int:task_id>
URL, Flask will call the associated function. -
update_task()
function will taketask_id
as the argument that is extracted from/tasks/<int:task_id>
URL. The function will then search for a task with that ID in the tasks list. If a task is found, it will update the task with the new data provided in the request and return the updated task in JSON format. If no task is found, it will return a404
status code with an error message. -
task.update(request.get_json())
updates the task with the JSON data sent by the client in the PUT request. Therequest.get_json()
function parses the incoming request data as JSON, and theupdate()
method merges this data into thetask
dictionary. -
return jsonify({"task": task})
returns a JSON response containing the updated task.
DELETE Request
A DELETE request is used to remove a resource from a server. To delete a task, implement a route that handles DELETE requests:
@app.route("/tasks/<int:task_id>", methods=["DELETE"])
def delete_tasks(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
tasks.remove(task)
return jsonify({"result": "Task deleted"}), 200
-
@app.route
binds a URL pattern/tasks/<int:task_id>
to the function that follows it. -
<int:task_id>
is a variable part of the URL that will be captured and passed to the function as an argument. -
methods=["DELETE"]
specifies that this route should respond to the HTTP DELETE requests. This means that when a client sends a DELETE request to the/tasks/<int:task_id>
URL, Flask will call the associated function. -
delete_task()
function will taketask_id
as the argument that is extracted from/tasks/<int:task_id>
URL. The function will then search for a task with that ID in thetasks
list. If a task is found, it will remove the task from the list and return a JSON response indicating that the task was deleted. If no task is found, it will return a404
status code with an error message. -
tasks.remove(task)
removes the task from thetasks
list. -
return jsonify({"result": "Task deleted"})
returns a JSON response indicating that the task was successfully deleted.
Running the Application
To run your application, add the following line:
if __name__ == "__main__":
app.run(debug=True)
The entire app.py
will look like this:
from flask import Flask, jsonify, request
app = Flask(__name__)
tasks = [
{"id": 1, "title": "Buy groceries", "completed": False},
{"id": 2, "title": "Clean the house", "completed": True}
]
@app.route("/tasks", methods=["GET"])
def get_tasks():
return jsonify({"tasks": tasks}), 200
@app.route("/tasks/<int:task_id>", methods=["GET"])
def get_task(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
return jsonify({"task": task}), 200
@app.route("/tasks", methods=["POST"])
def create_task():
new_task = request.get_json()
tasks.append(new_task)
return jsonify({"task": new_task}), 201
@app.route("/tasks/<int:task_id>", methods=["PUT"])
def update_task(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
task.update(request.get_json())
return jsonify({"task": task}), 201
@app.route("/tasks/<int:task_id>", methods=["DELETE"])
def delete_tasks(task_id):
task = next((item for item in tasks if item["id"] == task_id), None)
if task is None:
return jsonify({"error": "Task not found"}), 404
tasks.remove(task)
return jsonify({"result": "Task deleted"}), 200
if __name__ == "__main__":
app.run(debug=True)
Then open your terminal and run the following command from your project directory:
python3 app.py
Your REST API is now up and running on http://127.0.0.1:5000
Testing Your API
You can test your API using tools like Postman or cURL.
1. List All Tasks
You can use the following cURL command to list all the available tasks:
curl http://127.0.0.1:5000/tasks
Expected Result:
{
"tasks": [
{
"completed": false,
"id": 1,
"title": "Buy groceries"
},
{
"completed": true,
"id": 2,
"title": "Clean the house"
}
]
}
2. List a Specific Task
To retrieve a specific task by its ID, you can use the following cURL command, replacing <task_id>
with the ID of the task you want to retrieve:
curl http://127.0.0.1:5000/tasks/<task_id>
Example:
curl http://127.0.0.1:5000/tasks/1
The curl
command sends an HTTP GET request to http://127.0.0.1:5000/tasks/1
, retrieving the task with ID 1
from a local server.
Expected Result:
{
"task": {
"completed": false,
"id": 1,
"title": "Buy groceries"
}
}
3. Create a New Task
To create a new task, you can use following cURL command, replacing <task_data>
with the JSON data for the new task:
curl -X POST -H "Content-Type: application/json" -d '<json_data>' http://127.0.0.1:5000/tasks
Example:
curl -X POST -H "Content-Type: application/json" -d '{"id":3,"task":"Finish the report","completed":false}' http://127.0.0.1:5000/tasks
The curl
command sends an HTTP POST request with JSON data to http://127.0.0.1:5000/tasks
, creating a new task.
Expected Result:
{
"task": {
"completed": false,
"id": 3,
"task": "Finish the report"
}
}
4. Update a Task
To update an existing task, use the following cURL command, replacing <task_id>
with the id of the task, and <updated_data>
with the JSON data you want to update:
curl -X PUT -H "Content-Type: application/json" -d '<updated_data>' http://127.0.0.1:5000/tasks/<task_id>
Example:
curl -X PUT -H "Content-Type: application/json" -d '{"completed":true}' http://127.0.0.1:5000/tasks/3
The curl
command sends an HTTP PUT request with JSON data to http://127.0.0.1:5000/tasks/3
, updating the task with ID 3
to mark it as completed.
Expected Result:
{
"task": {
"completed": true,
"id": 3,
"task": "Finish the report"
}
}
5. Delete a Task
To delete a task, use the following cURL command, replacing <task_id>
with the id of the task you want to delete:
curl -X DELETE http://127.0.0.1:5000/tasks/<task_id>
Example:
curl -X DELETE http://127.0.0.1:5000/tasks/3
The curl
command sends an HTTP DELETE request to http://127.0.0.1:5000/tasks/3
, deleting the task with ID 3
.
Expected Result:
{
"result": "Task deleted"
}
Let's try to retrieve a task that doesn't exist in the list of tasks. For instance, we just deleted a task with ID 3
. Now, let's try to retrieve the task with ID 3
and see the result.
curl http://127.0.0.1:5000/tasks/3
Expected result:
{
"error": "Task not found"
}
We see that the JSON response returns the error Task not found
, which indicates that the task doesn't exist in the list of tasks.
Common Causes of API Request Failures
Now, let's explore scenarios where API requests might fail. API requests can fail for various reasons, including sending invalid requests. This could happen if an incorrect endpoint is used, if the JSON data is invalid, or if the request is made with an unsupported HTTP method. Let's examine some instances where API requests might fail.
1. Invalid Endpoint:
With the REST API that we built, we have the following valid API endpoints:
-
GET /tasks
: Retrieves a list of all available tasks -
GET /tasks/<int:task_id>
: Retrieves a specific task by its ID -
POST /tasks
: Creates a new task -
PUT /tasks/<int:task_id>
: Updates a specific task by its ID -
DELETE /tasks/<int:task_id>
: Deletes a specific task by its ID
If we send an API request to any endpoint that is not listed above, the request will fail with a 404
status code. For instance, let's try to get all the tasks with following cURL command:
curl http://127.0.0.1:5000/task
Here, we have used task
instead of tasks
in our endpoint, which makes our endpoint invalid. With this request, we get the following error with 404
status code.
Expected Result:
The requested URL was not found on the server. If you entered the URL manually please check your spelling and try again.
2. Invalid JSON Data:
Invalid JSON data can lead to API request failures due to syntax or structure errors. Here are some of the most common syntax and structure errors that can occur in JSON data:
- Missing Quotes Around Keys: In JSON, keys must be strings, and strings are defined by enclosing them in double quotes. Omitting quotes around keys will result in invalid JSON. Invalid:
{ key: "value" }
Valid:
{ "key": "value" }
- Single Quotes Instead of Double Quotes: JSON strings must be enclosed in double quotes. Using single quotes will result in invalid JSON. Invalid:
{ 'key': 'value' }
Valid:
{ "key": "value" }
- Trailing Commas: JSON does not allow trailing commas after the last element in an object or array. Including a comma after the last element will make the JSON invalid. Invalid:
{
"key1": "value1",
"key2": "value2",
}
Valid:
{
"key1": "value1",
"key2": "value2"
}
- Missing Comma Between Elements: In JSON objects and arrays, elements must be separated by commas. Omitting a comma between elements will make the JSON invalid. Invalid:
{
"key1": "value1"
"key2": "value2"
}
Valid:
{
"key1": "value1",
"key2": "value2"
}
- Incorrect Data Types: JSON supports several data types, including strings, numbers, objects, arrays, booleans, and null. Using an incorrect data type or a value that does not match the expected type can lead to invalid JSON. Invalid:
{
"key": "value",
"number": "123"
}
Valid:
{
"key": "value",
"number": 123
}
Let's add a new task with invalid JSON data with a missing quote around one of the keys:
curl -X POST -H "Content-Type: application/json" -d '{"id":3,task:"Finish the report","completed":false}' http://127.0.0.1:5000/tasks
Here, the JSON data in the request body is invalid as the key task
is not enclosed in double quotes. With this request, we get the following error with status code 400
.
Expected Result:
Failed to decode JSON object: Expecting property name enclosed in double quotes: line 1 column 9 (char 8)
3. Unsupported HTTP method
When a API request is sent to an endpoint with an unsupported HTTP method, then the API request will fail with an error message Method Not Allowed
. For example, trying to POST to an endpoint to get a specific task:
curl -X POST http://127.0.0.1:5000/tasks/1
Here, we have sent an API request with POST method to retrieve a specific task with ID 1
. The problem with using the POST
method to retrieve a specific task by its ID is that it violates the standard conventions of REST API design. In REST APIs, the POST method is intended for creating new resources, not for retrieving existing ones, due to which this API request will fail with error message Method Not Allowed
.
Expected Result:
Method Not Allowed
Conclusion
In this blog, we've explored the basics of REST APIs and how to implement them using Python Flask. This simple example demonstrates the power and flexibility of REST APIs in building scalable and maintainable web applications. You can expand this basic structure to include more complex data models, authentication, and more advanced features as needed. Building a REST API is just the beginning. The real power comes from how you integrate it with other services, databases, and front-end applications. Happy coding!
Top comments (0)