Welcome to the Movie Theater API project! This educational assignment is designed to help you develop and refine your skills in creating robust web applications using FastAPI, SQLAlchemy, and Docker. Here's what the project offers:
-
Database setup:
- PostgreSQL for development: The application uses PostgreSQL as the main database for the development environment, configured via Docker Compose.
- SQLite for testing: A lightweight SQLite database is utilized for testing, ensuring fast and isolated test execution.
-
Data population:
- The database can be automatically populated with movie data from a provided dataset. This includes associated entities such as genres, actors, languages, and countries, ensuring a rich and interconnected data structure.
-
Docker integration:
- The project is fully Dockerized, allowing seamless setup and execution of the application and its dependencies. Docker Compose simplifies the orchestration of services like the FastAPI application, PostgreSQL database, and any other required components.
-
Project structure:
- A well-organized and modular project structure is provided, including:
- Database models and schemas for movies and related entities.
- Routing logic for managing API endpoints.
- Utility scripts for tasks like data seeding and database migrations.
- A well-organized and modular project structure is provided, including:
The Movie Theater API project follows a modular and organized structure to simplify development, testing, and deployment. Below is an overview of the main components:
Here is a visual representation of the project structure:
.
├── Dockerfile
├── README.md
├── alembic.ini
├── commands
│ ├── run_migration.sh
│ └── run_web_server_dev.sh
├── docker-compose.yml
├── init.sql
├── poetry.lock
├── pyproject.toml
├── pytest.ini
└── src
├── config
│ ├── __init__.py
│ └── settings.py
├── database
│ ├── __init__.py
│ ├── models.py
│ ├── populate.py
│ ├── session_postgresql.py
│ ├── session_sqlite.py
│ ├── migrations
│ │ ├── README
│ │ ├── env.py
│ │ ├── script.py.mako
│ │ └── versions
│ │ ├── 3cd3b7913067_temp_migration.py
│ │ └── dac4daee7459_initial_migration.py
│ ├── seed_data
│ │ ├── imdb_movies.csv
│ │ └── test_data.csv
│ └── source
│ └── theater.db
├── main.py
├── routes
│ ├── __init__.py
│ └── movies.py
├── schemas
│ ├── __init__.py
│ └── movies.py
└── tests
├── __init__.py
├── conftest.py
└── test_integration
├── __init__.py
└── test_movies.py
Dockerfile
: Defines the container configuration for the FastAPI application.docker-compose.yml
: Orchestrates the application and its dependencies, such as PostgreSQL, in a development environment.README.md
: Provides detailed documentation for the project.alembic.ini
: Configuration file for Alembic, the tool used for database migrations.init.sql
: Contains initial SQL commands for setting up the PostgreSQL database.poetry.lock
andpyproject.toml
: Manage dependencies and project configurations using Poetry.pytest.ini
: Configuration file for pytest, specifying test settings and options.
run_migration.sh
: A script to execute database migrations.run_web_server_dev.sh
: A script to start the FastAPI development server.
The src
directory contains the core application logic and is organized as follows:
-
config
:settings.py
: Defines project configurations, including database connections for both development and testing environments.
-
database
:models.py
: Contains SQLAlchemy models for movies and related entities.populate.py
: Provides logic for seeding the database with movie data from datasets.session_postgresql.py
: Manages the PostgreSQL database session for development.session_sqlite.py
: Manages the SQLite database session for testing.migrations/
: Houses Alembic migration scripts.seed_data/
: Contains datasets (imdb_movies.csv
andtest_data.csv
) used for populating the database.source/
: Stores the SQLite database file (theater.db
) for testing.
-
routes
:movies.py
: Defines API endpoints related to movies, such as retrieving, creating, updating, and deleting movies.
-
schemas
:movies.py
: Contains Pydantic models for request and response validation.
-
tests
:conftest.py
: Defines shared fixtures for test configuration.test_integration/
: Contains integration tests for API endpoints.
-
main.py
:- The entry point for the application, responsible for initializing and running the FastAPI app.
The get_db
function is a generator that provides a SQLAlchemy session for interacting with the database. This function
is particularly useful in FastAPI as it can be injected into route handlers using the Depends
mechanism. Here’s how
you can use it effectively:
from fastapi import Depends, APIRouter
from sqlalchemy.orm import Session
from database import get_db # Import the get_db generator
router = APIRouter()
@router.get("/example")
def example_route(db: Session = Depends(get_db)):
# Use the db session here to interact with the database
- The
Depends
function simplifies injecting dependencies like the database session into your route handlers. - The
get_db
function ensures proper session handling: the session is created before the route logic executes and is closed automatically afterward. - This approach promotes cleaner, more testable code by separating dependency setup from business logic.
You can use this pattern across your application for any routes that need database access.
The project is configured with the following services in the docker-compose.yml
file. These services collectively
support the development, testing, and deployment of the Movie Theater API:
- Image:
postgres:latest
- Purpose: Acts as the primary database for the project, running a PostgreSQL instance.
- Configuration:
- Loads initial SQL setup from
init.sql
. - Stores persistent data using a named Docker volume
postgres_theater_data
. - Exposes PostgreSQL on port
5432
.
- Loads initial SQL setup from
- Health Check: Ensures the database is ready by using the
pg_isready
command. - Network: Attached to
theater_network
.
- Image:
dpage/pgadmin4
- Purpose: Provides a web-based interface for managing and monitoring the PostgreSQL database.
- Configuration:
- Exposes pgAdmin on port
3333
. - Stores pgAdmin data in the
pgadmin_theater_data
volume.
- Exposes pgAdmin on port
- Dependency: Starts only after the
db
service is healthy. - Network: Attached to
theater_network
.
- Build Context: Builds the FastAPI application from the local directory (
.
). - Purpose: Runs the FastAPI application, serving the Movie Theater API backend.
- Configuration:
- Exposes the backend on port
8000
. - Uses the
run_web_server_dev.sh
script to start the development server. - Watches for file changes in the
src
directory usingWATCHFILES_FORCE_POLLING=true
. - Mounts the
src
directory to/usr/src/fastapi
inside the container for live development.
- Exposes the backend on port
- Dependency: Starts only after the
db
service is healthy. - Network: Attached to
theater_network
.
- Build Context: Shares the same build context as the backend service.
- Purpose: Runs database migrations using Alembic to ensure the schema is up-to-date.
- Configuration:
- Executes the
run_migration.sh
script to apply migrations. - Uses the
src
directory as the source for migration scripts.
- Executes the
- Dependency: Starts only after the
db
service is healthy. - Network: Attached to
theater_network
.
postgres_theater_data
:- Stores persistent PostgreSQL data.
pgadmin_theater_data
:- Stores persistent data for pgAdmin.
theater_network
:- A bridge network connecting all services, enabling inter-service communication.
Follow these steps to set up and run the Movie Theater API project on your local machine.
Start by cloning the project repository from GitHub:
git clone <repository-url>
cd <repository-folder>
It is recommended to use a virtual environment to isolate project dependencies:
# Create a virtual environment
python -m venv venv
# Activate the virtual environment
# On Windows
venv\Scripts\activate
# On macOS/Linux
source venv/bin/activate
This project uses Poetry for dependency management. Install dependencies as follows:
# Install Poetry if not already installed
pip install poetry
# Install project dependencies
poetry install
Create a .env
file in the project root directory with the following variables. Customize the values as needed:
# PostgreSQL
POSTGRES_DB=movies_db
POSTGRES_DB_PORT=5432
POSTGRES_USER=admin
POSTGRES_PASSWORD=some_password
POSTGRES_HOST=postgres_theater
# pgAdmin
PGADMIN_DEFAULT_EMAIL=[email protected]
PGADMIN_DEFAULT_PASSWORD=admin
The project is Dockerized for easy setup. To start all the required services (PostgreSQL, pgAdmin, FastAPI app, and Alembic migrator), run:
docker-compose up --build
Note: On the first run, the database will be populated with data from the dataset. This process may take some time, so please be patient.
- API: The Movie Theater API will be available at
http://localhost:8000
. - pgAdmin: The pgAdmin web interface will be available at
http://localhost:3333
. Use the credentials you defined in the.env
file to log in.
After all services are running, you can test the API by accessing the OpenAPI documentation:
http://localhost:8000/docs
The project defines the following entities and relationships using SQLAlchemy. Each entity represents a table in the database and maps to a specific domain concept in the Movie Theater API.
Represents a movie in the database.
-
Table Name:
movies
-
Fields:
id
(Primary Key): Unique identifier for each movie.name
: Name of the movie.date
: Release date of the movie.score
: Movie rating score (e.g., IMDb score).overview
: A short description or synopsis of the movie.status
: Production status of the movie (e.g., Released, In Production).budget
: The budget of the movie (stored as a decimal value).revenue
: The revenue generated by the movie.country_id
: Foreign key linking to thecountries
table.
-
Relationships:
country
: Links to theCountryModel
.genres
: Many-to-many relationship withGenreModel
.actors
: Many-to-many relationship withActorModel
.languages
: Many-to-many relationship withLanguageModel
.
-
Constraints:
- Unique constraint on
name
anddate
to prevent duplicate entries.
- Unique constraint on
Represents a genre (e.g., Action, Comedy).
-
Table Name:
genres
-
Fields:
id
(Primary Key): Unique identifier for each genre.name
: Name of the genre (e.g., Action, Drama).
-
Relationships:
movies
: Many-to-many relationship withMovieModel
.
Represents an actor in the database.
-
Table Name:
actors
-
Fields:
id
(Primary Key): Unique identifier for each actor.name
: Name of the actor.
-
Relationships:
movies
: Many-to-many relationship withMovieModel
.
Represents a country associated with a movie (e.g., production country).
-
Table Name:
countries
-
Fields:
id
(Primary Key): Unique identifier for each country.code
: ISO 3166-1 alpha-3 country code (e.g., USA, FRA).name
: Full name of the country.
-
Relationships:
movies
: One-to-many relationship withMovieModel
.
Represents a language spoken in a movie.
-
Table Name:
languages
-
Fields:
id
(Primary Key): Unique identifier for each language.name
: Name of the language (e.g., English, French).
-
Relationships:
movies
: Many-to-many relationship withMovieModel
.
Used to establish many-to-many relationships between entities.
-
MoviesGenresModel
:- Links
movies
andgenres
. - Fields:
movie_id
,genre_id
.
- Links
-
ActorsMoviesModel
:- Links
movies
andactors
. - Fields:
movie_id
,actor_id
.
- Links
-
MoviesLanguagesModel
:- Links
movies
andlanguages
. - Fields:
movie_id
,language_id
.
- Links
In this assignment, you are tasked with continuing the development of the cinema application. Your objective is to
implement two endpoints in the schemas/movies.py
and routes/movies.py
files. The rest of the application, including
database models and utility functions, has already been provided.
Your task is to implement an endpoint in the routes/movies.py
file that retrieves a paginated list of movies from
the database. The required response structure is detailed below.
- HTTP Method:
GET
- Path:
/movies/
-
page
(integer):- Specifies the page number to retrieve.
- Default:
1
- Constraints: Must be greater than or equal to
1
.
-
per_page
(integer):- Specifies the number of items to display per page.
- Default:
10
- Constraints: Must be between
1
and20
(inclusive).
The endpoint should return the following JSON response:
{
"movies": [
{
"id": 9936,
"name": "Avatar new",
"date": "2022-12-15",
"score": 78.0,
"overview": "Set more than a decade after the events of the first film..."
}
],
"prev_page": null,
"next_page": "/theater/movies/?page=2&per_page=1",
"total_pages": 9934,
"total_items": 9934
}
- Sorting: Movies must be sorted in descending order by their
id
. - Pagination:
- The
page
andper_page
parameters control pagination. - The
prev_page
andnext_page
fields should provide links to the previous and next pages, respectively, ornull
if no such page exists.
- The
- Empty Results:
- If no movies are found for the specified page, the endpoint should return a
404 Not Found
error with the message:"No movies found."
- If no movies are found for the specified page, the endpoint should return a
- 200 OK: Successfully retrieved a paginated list of movies.
- 404 Not Found: No movies found for the specified page.
This task requires you to implement both the endpoint logic in routes/movies.py
and the response structure in
schemas/movies.py
. Ensure that your implementation handles all edge cases, such as invalid query parameters or empty
results.
Your task is to implement an endpoint in the routes/movies.py
file that allows the creation of a new movie in the database. The required request and response structures are detailed below.
- HTTP Method:
POST
- Path:
/movies/
The endpoint accepts a JSON object with the following fields:
{
"name": "string",
"date": "date",
"score": "float (0-100)",
"overview": "string",
"status": "string (Released | Post Production | In Production)",
"budget": "float (>= 0)",
"revenue": "float (>= 0)",
"country": "string (ISO 3166-1 alpha-3 code)",
"genres": ["string"],
"actors": ["string"],
"languages": ["string"]
}
The endpoint returns the created movie's details in the following format:
{
"id": "integer",
"name": "string",
"date": "date",
"score": "float",
"overview": "string",
"status": "string",
"budget": "float",
"revenue": "float",
"country": {
"id": "integer",
"code": "string",
"name": "string or null"
},
"genres": [
{
"id": "integer",
"name": "string"
}
],
"actors": [
{
"id": "integer",
"name": "string"
}
],
"languages": [
{
"id": "integer",
"name": "string"
}
]
}
-
Validation:
- The
name
must not exceed 255 characters. - The
date
must not be more than one year in the future. - The
score
must be between 0 and 100. - The
budget
andrevenue
must be non-negative.
- The
-
Entity Linking:
- The endpoint automatically links or creates related entities (country, genres, actors, languages) if they do not already exist in the database.
-
Duplicate Check:
- If a movie with the same
name
anddate
already exists, the endpoint returns a409 Conflict
error with an appropriate message. - Message: "A movie with the name '{name}' and release date '{date}' already exists."
- If a movie with the same
-
Error Handling:
- Returns a
400 Bad Request
if the input data is invalid.
- Returns a
- 201 Created: The movie was successfully created, and its details are returned in the response.
- 400 Bad Request: The input data is invalid (e.g., missing required fields, invalid values).
- 409 Conflict: A movie with the same
name
anddate
already exists.
-
Request:
POST /movies/ Content-Type: application/json { "name": "Inception", "date": "2010-07-16", "score": 8.8, "overview": "A mind-bending thriller about dreams within dreams.", "status": "Released", "budget": 160000000.00, "revenue": 829895144.00, "country": "USA", "genres": ["Action", "Sci-Fi"], "actors": ["Leonardo DiCaprio", "Joseph Gordon-Levitt"], "languages": ["English", "Japanese"] }
Response:
{ "id": 1, "name": "Inception", "date": "2010-07-16", "score": 8.8, "overview": "A mind-bending thriller about dreams within dreams.", "status": "Released", "budget": 160000000.00, "revenue": 829895144.00, "country": { "id": 1, "code": "USA", "name": "United States" }, "genres": [ { "id": 1, "name": "Action" }, { "id": 2, "name": "Sci-Fi" } ], "actors": [ { "id": 1, "name": "Leonardo DiCaprio" }, { "id": 2, "name": "Joseph Gordon-Levitt" } ], "languages": [ { "id": 1, "name": "English" }, { "id": 2, "name": "Japanese" } ] }
-
Request:
POST /movies/ Content-Type: application/json { "name": "Inception", "date": "2010-07-16" }
Response:
{ "detail": "Invalid input data." }
Your task is to implement an endpoint in the routes/movies.py
file that retrieves detailed information about a specific movie by its unique ID. The required response structure is detailed below.
- HTTP Method:
GET
- Path:
/movies/{movie_id}/
movie_id
(integer):
The unique identifier of the movie to retrieve.
The endpoint should return the details of the movie in the following format:
{
"id": "integer",
"name": "string",
"date": "date",
"score": "float",
"overview": "string",
"status": "string (Released | Post Production | In Production)",
"budget": "float",
"revenue": "float",
"country": {
"id": "integer",
"code": "string",
"name": "string or null"
},
"genres": [
{
"id": "integer",
"name": "string"
}
],
"actors": [
{
"id": "integer",
"name": "string"
}
],
"languages": [
{
"id": "integer",
"name": "string"
}
]
}
-
Validation:
- The
movie_id
must be a valid integer corresponding to an existing movie in the database.
- The
-
Entity Linking:
- The endpoint should include all related entities (country, genres, actors, languages) in the response.
-
Error Handling:
- If the movie with the given
movie_id
is not found, the endpoint must return a404 Not Found
error with the message:"Movie with the given ID was not found."
- If the movie with the given
- 200 OK: Successfully retrieved the movie details.
- 404 Not Found: The movie with the specified ID does not exist.
-
Request:
GET /movies/1/
Response:
{ "id": 1, "name": "Inception", "date": "2010-07-16", "score": 8.8, "overview": "A mind-bending thriller about dreams within dreams.", "status": "Released", "budget": 160000000.00, "revenue": 829895144.00, "country": { "id": 1, "code": "USA", "name": "United States" }, "genres": [ { "id": 1, "name": "Action" }, { "id": 2, "name": "Sci-Fi" } ], "actors": [ { "id": 1, "name": "Leonardo DiCaprio" }, { "id": 2, "name": "Joseph Gordon-Levitt" } ], "languages": [ { "id": 1, "name": "English" }, { "id": 2, "name": "Japanese" } ] }
-
Request:
GET /movies/999/
Response:
{ "detail": "Movie with the given ID was not found." }
Your task is to implement an endpoint in the routes/movies.py
file that deletes a specific movie by its unique ID. The required details for this endpoint are provided below.
- HTTP Method:
DELETE
- Path:
/movies/{movie_id}/
movie_id
(integer):
The unique identifier of the movie to delete.
The endpoint does not return a response body on success. Instead, it responds with the following status codes:
- 204 No Content: The movie was successfully deleted.
- 404 Not Found: The movie with the specified ID does not exist.
-
Validation:
- The
movie_id
must correspond to an existing movie in the database.
- The
-
Error Handling:
- If the movie with the given
movie_id
does not exist, the endpoint must return a404 Not Found
error with the message:"Movie with the given ID was not found."
- If the movie with the given
-
Successful Deletion:
- The movie is removed from the database, and the endpoint returns a
204 No Content
status.
- The movie is removed from the database, and the endpoint returns a
-
Request:
DELETE /movies/1/
Response:
- Status Code:
204 No Content
- Status Code:
-
Request:
DELETE /movies/999/
Response:
{ "detail": "Movie with the given ID was not found." }
Your task is to implement an endpoint in the routes/movies.py
file that updates the details of a specific movie by its unique ID. The details for this endpoint are provided below.
- HTTP Method:
PATCH
- Path:
/movies/{movie_id}/
movie_id
(integer):
The unique identifier of the movie to update.
The endpoint accepts a JSON object with the following optional fields. Any fields not provided will remain unchanged in the database:
{
"name": "string",
"date": "date",
"score": "float (0-100)",
"overview": "string",
"status": "string (Released | Post Production | In Production)",
"budget": "float (>= 0)",
"revenue": "float (>= 0)"
}
The endpoint returns a response indicating the result of the update operation:
-
200 OK: Successfully updated the movie.
{ "detail": "Movie updated successfully." }
-
404 Not Found: The movie with the specified ID does not exist.
{ "detail": "Movie with the given ID was not found." }
-
400 Bad Request: The input data is invalid (e.g., violates constraints).
{ "detail": "Invalid input data." }
-
Validation:
- Only the provided fields in the request body are updated.
- The
score
must be between 0 and 100. - The
budget
andrevenue
must be non-negative.
-
Entity Retrieval:
- The endpoint checks if a movie with the given
movie_id
exists. If not, it returns a404 Not Found
error.
- The endpoint checks if a movie with the given
-
Partial Update:
- Updates only the fields provided in the request body, leaving all other fields unchanged.
-
Error Handling:
- If invalid input data is provided, the endpoint returns a
400 Bad Request
error.
- If invalid input data is provided, the endpoint returns a
-
Request:
PATCH /movies/1/ Content-Type: application/json { "name": "Updated Movie Name", "score": 95.0 }
Response:
{ "detail": "Movie updated successfully." }
-
Request:
PATCH /movies/999/ Content-Type: application/json { "name": "Nonexistent Movie" }
Response:
{ "detail": "Movie with the given ID was not found." }
-
Request:
PATCH /movies/1/ Content-Type: application/json { "score": 150.0 }
Response:
{ "detail": "Invalid input data." }
If you’re unsure about the expected behavior or need clarification, refer to the provided test suite. Running the tests will:
- Show the expected logic and flow for each endpoint.
- Help you identify edge cases and handle errors correctly.
- Ensure your implementation aligns with the project's requirements.
To run the tests, use the following command in the project root directory:
pytest
The test results will indicate any discrepancies between your implementation and the expected behavior, providing clear guidance on how to fix them.