Bandcamp Async API
Project description
Bandcamp Async API
A modern, asynchronous Python client for the Bandcamp API.
This project was created to implement a Bandcamp music provider for Music Assistant, enabling seamless integration of Bandcamp's music catalog into home audio systems.
- Repository: https://github.com/ALERTua/bandcamp_async_api
- Changelog: https://github.com/ALERTua/bandcamp_async_api/releases
- PyPI: https://pypi.org/project/bandcamp_async_api/
- Music Assistant: https://github.com/music-assistant
Features
- Search: Search for artists, albums, and tracks across Bandcamp
- Albums: Retrieve detailed album information including track listings
- Tracks: Get individual track details and streaming information
- Artists: Access artist profiles, discographies, and metadata
- Collections: Browse user collections and wishlists (auth required for private data)
- Following: Access following bands, following fans, and followers
- Feed: Get personalized music feed with new releases from followed artists
- Async: Fully asynchronous API using aiohttp
- Type-safe: Complete type hints for all models and methods
- Well-tested: Comprehensive test suite with real API data
Installation
Install from PyPI:
pip install bandcamp-async-api
Or using uv:
uv add bandcamp-async-api
Quick Start
import asyncio
from bandcamp_async_api import BandcampAPIClient
async def main():
async with BandcampAPIClient(identity_token='7%09optional_identity_token%7D') as client:
# Search for music
results = await client.search("radiohead")
print(f"Found {len(results)} results")
# Get album details
if results:
album_result = next(r for r in results if r.type == "album")
album = await client.get_album(album_result.artist_id, album_result.id)
print(f"Album: {album.title} by {album.artist.name}")
# Get artist information
artist_result = next(r for r in results if r.type == "artist")
artist = await client.get_artist(artist_result.id)
print(f"Artist: {artist.name} - {artist.bio}")
if __name__ == '__main__':
asyncio.run(main())
Authentication
For accessing user collections, you need to obtain an identity token from Bandcamp cookies:
from bandcamp_async_api import BandcampAPIClient
client = BandcampAPIClient(identity_token="your_identity_token")
Music Feed
The get_feed() method retrieves a personalized music feed containing new releases from followed artists, fan purchases, and fan picks. This endpoint requires authentication - you must provide an identity token.
import asyncio
from bandcamp_async_api import BandcampAPIClient, BandcampMustBeLoggedInError
async def main():
async with BandcampAPIClient(identity_token='your_identity_token') as client:
# Get your music feed
feed = await client.get_feed()
print(f"New stories: {len(feed.stories)}")
print(f"Has more: {feed.has_more}")
# Iterate through feed stories
for story in feed.stories:
print(f" - {story.story_type}: {story.item_title} by {story.band_name}")
# Access tracks with streaming URLs
for track in feed.track_list:
print(f" Track: {track.title} - {track.streaming_url}")
# Paginate through older stories
if feed.has_more and feed.oldest_story_date:
older_feed = await client.get_feed(older_than=feed.oldest_story_date)
print(f"Older stories: {len(older_feed.stories)}")
if __name__ == '__main__':
asyncio.run(main())
Feed Story Types
The feed contains different story types:
np- New track/track releasenr- New album releasep- Fan purchasefp- Fan pick
Error Handling
The feed endpoint requires authentication. If you try to access it without an identity token, you'll receive a BandcampMustBeLoggedInError:
from bandcamp_async_api import BandcampAPIClient, BandcampMustBeLoggedInError
async def safe_get_feed():
client = BandcampAPIClient() # No identity token
try:
feed = await client.get_feed()
except BandcampMustBeLoggedInError:
print("Feed requires authentication - provide an identity token")
API Reference
Core Client
BandcampAPIClient()- Main API clientsearch(query: str)- Search Bandcampget_album(artist_id, album_id)- Get album detailsget_track(artist_id, track_id)- Get track detailsget_artist(artist_id)- Get artist detailsget_collection_summary()- Get collection overviewget_collection_items(collection_type, older_than_token, count, fan_id)- Get collection/wishlist/following items with paginationget_artist_discography(artist_id)- Get artist's complete discographyget_feed(older_than)- Get personalized music feed with pagination support
Data Models
SearchResultItem- Base search resultBCAlbum- Album with tracks and metadataBCTrack- Individual track informationBCArtist- Artist/band profileCollectionSummary- User's collection dataCollectionItem- Individual collection itemFollowingItem- Band/artist from following listFanItem- Fan/user from following_fans or followersFeedResponse- User's music feed with stories and tracksFeedStory- Individual feed story (new release, fan purchase, etc.)FeedTrack- Track from feed with streaming URLFeedBandInfo- Band information referenced in feedFeedFanInfo- Fan information referenced in feed
Exceptions
BandcampAPIError- Base API errorBandcampNotFoundError- Resource not foundBandcampBadQueryError- Invalid search queryBandcampRateLimitError- Rate limit exceeded (includesretry_afterattribute)
Error Handling
The client provides specific exception types for different error conditions:
from bandcamp_async_api import (
BandcampAPIClient,
BandcampNotFoundError,
BandcampAPIError
)
async def safe_get_album(client, artist_id, album_id):
try:
return await client.get_album(artist_id, album_id)
except BandcampNotFoundError:
print("Album not found")
return None
except BandcampAPIError as e:
print(f"API error: {e}")
return None
Rate Limiting
When Bandcamp's API rate limit is exceeded, a BandcampRateLimitError is raised with a retry_after attribute indicating how many seconds to wait before retrying:
import asyncio
from bandcamp_async_api import BandcampAPIClient, BandcampRateLimitError
async def get_album_with_retry(client, artist_id, album_id, max_retries=3):
for attempt in range(max_retries):
try:
return await client.get_album(artist_id, album_id)
except BandcampRateLimitError as e:
if attempt < max_retries - 1:
wait_time = e.retry_after or 30
print(f"Rate limited. Waiting {wait_time} seconds...")
await asyncio.sleep(wait_time)
else:
raise
For automatic retries with exponential backoff, you can use the tenacity library:
from tenacity import retry, retry_if_exception_type, wait_exponential
from bandcamp_async_api import BandcampAPIClient, BandcampRateLimitError
@retry(
retry=retry_if_exception_type(BandcampRateLimitError),
wait=wait_exponential(multiplier=1, min=30, max=300)
)
async def get_album(client, artist_id, album_id):
return await client.get_album(artist_id, album_id)
Development
Setup
# Clone the repository
git clone https://github.com/ALERTua/bandcamp_async_api.git
cd bandcamp_async_api
# Install dependencies
uv sync --dev
# Run tests
uv run pytest
# Run linting
uv run ruff check
Testing
The project includes comprehensive tests:
# Run all tests
uv run pytest
# Run integration tests (requires real API access)
echo "BANDCAMP_IDENTITY_TOKEN=7%09identity_token%7D" > .env
uv run pytest tests/real_data/
Contributing
Contributions are welcome! Please:
- Fork the repository
- Create a feature branch
- Add tests for new functionality
- Ensure all tests pass
- Submit a pull request
This project is built based on data from:
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 bandcamp_async_api-0.1.1.tar.gz.
File metadata
- Download URL: bandcamp_async_api-0.1.1.tar.gz
- Upload date:
- Size: 13.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.2 {"installer":{"name":"uv","version":"0.11.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c1707cc937c54d7da1ddd6c3e11787afac616e2f26c808ee260e8d52debfc3b0
|
|
| MD5 |
c12af0fd3dd69dc8dd01f837e11b2c91
|
|
| BLAKE2b-256 |
7a8d5fa9ba91ff7cd7115a19b1e5678c3f7a8c45ace14f75aef62e95c3ee45ec
|
File details
Details for the file bandcamp_async_api-0.1.1-py3-none-any.whl.
File metadata
- Download URL: bandcamp_async_api-0.1.1-py3-none-any.whl
- Upload date:
- Size: 18.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.2 {"installer":{"name":"uv","version":"0.11.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8aab748f4af3424b7d79d9c5883a08d28cb7cf33ce8d135decbf722b3ef027a7
|
|
| MD5 |
37a726b7d52a34afb63a4e26b78bcbd6
|
|
| BLAKE2b-256 |
7e56e681a57ac422ff8ffb9007d723afe9d0d197297cc6241c8dbb7f50ebc016
|