A production-ready Python wrapper for the Pexels REST API featuring robust Pydantic input validation.
Project description
Pexels API Python Wrapper
Disclaimer
This project is an independent, community-maintained Python wrapper for the Pexels API. It is not affiliated with, endorsed by, or maintained by Pexels.
An elegant, developer-friendly Python client library for interacting seamlessly with the Pexels REST API.
Unlike naive wrappers, this package uses Pydantic v2 to perform robust client-side validation on your parameters before hitting the network—saving you latency, bandwidth, and unhandled remote API errors.
Features
- Pydantic-powered Validation: Caught invalid page bounds, dimensions, orientations, or hexadecimal colors locally before triggering HTTP requests.
- Clean Interface: Consistent method signatures exposing Photo, Video, and Curated/Featured Collection endpoints.
- Fail-Safe Response Analysis: Safe management of common HTTP statuses (
400,401,404) with programmatic reporting. - Transparent Rate Limiting: Automatic tracking and console reporting of monthly quotas (
X-Ratelimit-*) parsed exclusively from successful hits.
Installation
Install the package via pip:
pip install pexels-sdk
Quick Start
1. Initializing the Client
Obtain an API key from the Pexels API Documentation portal.
from pexels import Client
# Note: Pexels uses raw keys. The client automatically manages
# the header authorization for you without a "Bearer" prefix.
client = Client(api_key="YOUR_PEXELS_API_KEY")
Photo Endpoints
Search Photos
Search for photos using a required query and optional filters such as orientation, size, color, locale, page, and per_page.
search_results = client.search_photos(
query="nature",
orientation="landscape",
size="large",
page=1,
per_page=10
)
for photo in search_results["photos"]:
print(photo["photographer"])
print(photo["src"]["original"])
Curated Photos
Retrieve curated photos from the Pexels curated feed.
curated = client.curated_photos(page=1, per_page=10)
for photo in curated["photos"]:
print(photo["photographer"])
Get a Single Photo
Fetch a specific photo by its unique ID.
photo = client.get_a_photo(photo_id=2014422)
print(photo)
Video Endpoints
Search Videos
Search for videos using keyword-based filtering and supported video parameters.
videos = client.search_for_videos(
query="mountains",
orientation="landscape",
size="large",
page=1,
per_page=10
)
for video in videos["videos"]:
print(video["user"]["name"])
Popular Videos
Retrieve popular videos with optional dimension and duration filters.
popular = client.popular_videos(
min_width=1280,
min_height=720,
min_duration=5,
max_duration=30,
page=1,
per_page=10
)
for video in popular["videos"]:
print(video["id"])
Get a Single Video
Fetch a specific video by its unique ID.
video = client.get_a_video(video_id=123456)
print(video)
Collection Endpoints Featured Collections
Get a list of featured collections.
featured = client.featured_collections(page=1, per_page=10)
for collection in featured["collections"]:
print(collection["title"])
My Collections
Retrieve the authenticated user’s collections.
collections = client.my_collections(page=1, per_page=10)
for collection in collections["collections"]:
print(collection["title"])
Collection Media
Get the media items inside a specific collection.
media = client.collection_media(
collection_id=12345,
collection_type="photos",
sort="asc",
page=1,
per_page=10
)
print(media)
Validation
This package uses Pydantic v2 to validate input parameters before sending any HTTP request.
Invalid values are rejected locally, which helps reduce unnecessary network traffic and prevents avoidable API errors.
Examples of validation-covered parameters include:
page bounds
image dimensions
orientations
colors in hexadecimal format
video duration filters
from pydantic import ValidationError
try:
client.search_photos(query="city", page=-1)
except ValidationError as e:
print(f"Validation error: {e}")
Error Handling
The client includes a ResponseAnalyzer layer that inspects API responses and handles common HTTP statuses such as:
400 Bad Request
401 Unauthorized
404 Not Found
This gives you a cleaner and more predictable error-handling flow.
from pexels.exceptions import PexelsAPIError
try:
client.search_photos(query="ocean")
except PexelsAPIError as e:
print(f"API Error ({e.status_code}): {e.message}")
Rate Limiting
The client also tracks rate-limit information returned by Pexels headers such as:
X-Ratelimit-Limit
X-Ratelimit-Remaining
X-Ratelimit-Reset
This makes quota usage more transparent during development and production use.
client.search_photos(query="architecture")
print(client.rate_limit_remaining)
Project Highlights
Pydantic-powered validation before API calls
Clean endpoint coverage for photos, videos, and collections
Safe response analysis for common HTTP failures
Transparent quota awareness via rate-limit headers
Simple authorization flow using raw Pexels API keys
Contributing
Contributions are welcome and appreciated.
Fork the Project
Create your Feature Branch:
git checkout -b feature/AmazingFeature
Commit your Changes:
git commit -m "Add some AmazingFeature"
Push to the Branch:
git push origin feature/AmazingFeature
License
Distributed under the MIT License. See the LICENSE file for more information.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pexels_sdk-1.1.0.tar.gz.
File metadata
- Download URL: pexels_sdk-1.1.0.tar.gz
- Upload date:
- Size: 14.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0a545c5db77a40b36661abbfc2a5e984e0db734e4ea0cbfcae81f86e29602f46
|
|
| MD5 |
0c3c5fc89ef0619a7ac7948128bc64f0
|
|
| BLAKE2b-256 |
784d9134c1499bf4b9f5f475ce1d18c685f782375be429011396abbe589773e4
|
Provenance
The following attestation bundles were made for pexels_sdk-1.1.0.tar.gz:
Publisher:
publish.yml on Stink-Po/pexels-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pexels_sdk-1.1.0.tar.gz -
Subject digest:
0a545c5db77a40b36661abbfc2a5e984e0db734e4ea0cbfcae81f86e29602f46 - Sigstore transparency entry: 1721553157
- Sigstore integration time:
-
Permalink:
Stink-Po/pexels-python@0b0c23e63f1da2af30c43545efdfbd20919758e3 -
Branch / Tag:
refs/tags/v1.1.0 - Owner: https://github.com/Stink-Po
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@0b0c23e63f1da2af30c43545efdfbd20919758e3 -
Trigger Event:
release
-
Statement type:
File details
Details for the file pexels_sdk-1.1.0-py3-none-any.whl.
File metadata
- Download URL: pexels_sdk-1.1.0-py3-none-any.whl
- Upload date:
- Size: 15.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
da151628f050e667d2b918965f94fb207e2d703e188ff2e529aa98abc61170ea
|
|
| MD5 |
7645ee19eab9034bea8e43aee21e9954
|
|
| BLAKE2b-256 |
0a1942d5816b451f278792258bcaedf860808b4cdfddc3de29361516e2590383
|
Provenance
The following attestation bundles were made for pexels_sdk-1.1.0-py3-none-any.whl:
Publisher:
publish.yml on Stink-Po/pexels-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pexels_sdk-1.1.0-py3-none-any.whl -
Subject digest:
da151628f050e667d2b918965f94fb207e2d703e188ff2e529aa98abc61170ea - Sigstore transparency entry: 1721553224
- Sigstore integration time:
-
Permalink:
Stink-Po/pexels-python@0b0c23e63f1da2af30c43545efdfbd20919758e3 -
Branch / Tag:
refs/tags/v1.1.0 - Owner: https://github.com/Stink-Po
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@0b0c23e63f1da2af30c43545efdfbd20919758e3 -
Trigger Event:
release
-
Statement type: