Unofficial Python client for Meta Threads.net API
Project description
Meta's Threads.net API
Unofficial, Reverse-Engineered Python client for Meta's Threads.
Inspired by NPM Threads-API
Threads API - Python
Threads API is an unofficial Python client for Meta's Threads API. It allows you to interact with the API to login, read and publish posts, view who liked a post, retrieve user profile information, follow/unfollow and much more.
It allows you to configure the session object. Choose between:
aiohttp
- Python library to ease asynchronous execution of the API, for ⚡ super-fast ⚡ results. (default)requests
- Python library for standard ease of use (supportsHTTP_PROXY
env var functionality)instagrapi
- utilize the same connection all the way for private api- (Advanced) Implement your own and call ThreadsAPI like this:
ThreadsAPI(http_class=YourOwnHTTPSessionImpl)
Note Since v1.1.3 we are using
instagrapi
package to login.
Note Since v1.1.10 you can use
requests
orinstagrapi
as HTTP clients, not justaiohttp
.
Note Since v1.1.12 a
.session.json
file will be created by-default to save default settings (to reduce risk of being flagged). You can disable it by passingThreadsAPI(settings_path=None)
Important Tip Use the same
cached_token_path
for connections, to reduce the number of actual login attempts. When needed, threads-api will reconnect and update the file incached_token_path
.
Table of content:
Demo
Getting Started
📦 Installation
pip install threads-api
or
poetry add threads-api
Example using threads-api to post to Threads.net:
from threads_api.src.threads_api import ThreadsAPI
import asyncio
import os
from dotenv import load_dotenv
load_dotenv()
async def post():
api = ThreadsAPI()
await api.login(os.environ.get('INSTAGRAM_USERNAME'), os.environ.get('INSTAGRAM_PASSWORD'), cached_token_path=".token")
result = await api.post(caption="Posting this from the Danie1/threads-api!", image_path=".github/logo.jpg")
if result:
print("Post has been successfully posted")
else:
print("Unable to post.")
await api.close_gracefully()
async def main():
await post()
# Create an event loop and run the main function
loop = asyncio.get_event_loop()
loop.run_until_complete(main())
Customize HTTP Client
Each HTTP client brings to the table different functionality. Use whichever you like, or implement your own wrapper.
Usage:
api = ThreadsAPI(http_session_class=AioHTTPSession) # default
# or
api = ThreadsAPI(http_session_class=RequestsSession)
# or
api = ThreadsAPI(http_session_class=InstagrapiSession)
Set Desired Log Level
Threads-API reads the environment variable LOG_LEVEL
and sets the log-level according to its value.
Possible values include: DEBUG, INFO, WARNING, ERROR, CRITICAL
Log Level defaults to WARNING when not set.
Useful to know:
# Set Info (Prints general flow)
export LOG_LEVEL=INFO
# Set Debug (Prints HTTP Requests + HTTP Responses)
export LOG_LEVEL=DEBUG
Contributing to Danie1/threads-api
Getting Started
With Poetry (Recommended)
# Step 1: Clone the project
git clone git@github.com:Danie1/threads-api.git
# Step 2: Install dependencies to virtual environment
poetry install
# Step 3: Activate virtual environment
poetry shell
or
Without Poetry
# Step 1: Clone the project
git clone git@github.com:Danie1/threads-api.git
# Step 2: Create virtual environment
python3 -m venv env
# Step 3 (Unix/MacOS): Activate virtual environment
source env/bin/activate # Unix/MacOS
# Step 3 (Windows): Activate virtual environment
.\env\Scripts\activate # Windows
# Step 4: Install dependencies
pip install -r requirements.txt
Supported Features
- ✅ Login functionality, including 2FA 🔒
- ✅ Cache login token securely (reduce login requests / due to restrictive limits)
- ✅ Read recommended posts from timeline (Requires Login 🔒)
- ✅ Write Posts (Requires Login 🔒)
- ✅ Posts with just text
- ✅ Posts with text and an image
- ✅ Posts with text that share a url
- ✅ Reply to Posts
- ✅ Perform Actions (Requires Login 🔒)
- ✅ Like Posts
- ✅ Unlike Posts
- ✅ Delete post
- ✅ Follow User
- ✅ Unfollow User
- ✅ Block User
- ✅ Unblock User
- ✅ Restrict User
- ✅ Unrestrict User
- ✅ Mute User
- ✅ Unmute User
- ✅ Read Public Data
- ✅ Read a user_id (eg.
314216
) via username(eg.zuck
) - ✅ Read a user's profile info
- ✅ Read list of a user's Threads
- ✅ Read list of a user's Replies
- ✅ Read Post and a list of its Replies
- ✅ View who liked a post
- ✅ Read a user_id (eg.
- ✅ Read Private Data (Requires Login 🔒)
- ✅ Read a user's followers list
- ✅ Read a user's following list
- ✅ CI/CD
- ✅ GitHub Actions Pipeline
Usage Examples
View example.py for code examples. At the end of the file you will be able to uncomment and run the individual examples with ease.
Then simply run as:
# Pass the credentials as environment variables
USERNAME=<Instagram Username> PASSWORD=<Instagram Password> python3 example.py
Samples
"get_user_id_from_username" Function
from threads_api.src.threads_api import ThreadsAPI
import asyncio
async def get_user_id_from_username():
threads_api = ThreadsAPI()
username = "zuck"
user_id = await threads_api.get_user_id_from_username(username)
if user_id:
print(f"The user ID for username '{username}' is: {user_id}")
else:
print(f"User ID not found for username '{username}'")
Example Output:
The user ID for username 'zuck' is: 314216
"get_user_profile" Function
async def get_user_profile():
threads_api = ThreadsAPI()
username = "zuck"
user_id = await threads_api.get_user_id_from_username(username)
if user_id:
user_profile = await threads_api.get_user_profile(user_id)
print(f"User profile for '{username}':")
print(f"Name: {user_profile['username']}")
print(f"Bio: {user_profile['biography']}")
print(f"Followers: {user_profile['follower_count']}")
else:
print(f"User ID not found for username '{username}'")
Example Output:
User profile for 'zuck':
Name: zuck
Bio:
Followers: 2288633
"get_user_threads" Function
async def get_user_threads():
threads_api = ThreadsAPI()
username = "zuck"
user_id = await threads_api.get_user_id_from_username(username)
if user_id:
threads = await threads_api.get_user_threads(user_id)
print(f"The threads for user '{username}' are:")
for thread in threads:
print(f"Text: {thread['thread_items'][0]['post']['caption']} || Likes: {thread['thread_items'][0]['post']['like_count']}")
else:
print(f"User ID not found for username '{username}'")
Example Output:
The threads for user 'zuck' are:
zuck's Post: {'text': '70 million sign ups on Threads as of this morning. Way beyond our expectations.'} || Likes: 159293
zuck's Post: {'text': 'Lots of work on basic capabilities this morning.'} || Likes: 217148
zuck's Post: {'text': "Wow, 30 million sign ups as of this morning. Feels like the beginning of something special, but we've got a lot of work ahead to build out the app."} || Likes: 340098
zuck's Post: {'text': '10 million sign ups in seven hours 🤯'} || Likes: 357105
zuck's Post: {'text': 'Just passed 5 million sign ups in the first four hours...'} || Likes: 156277
zuck's Post: {'text': 'Threads just passed 2 million sign ups in the first two hours.'} || Likes: 132504
zuck's Post: {'text': "Glad you're all here on day one. Let's build something great together!"} || Likes: 175563
zuck's Post: {'text': "Let's do this. Welcome to Threads. 🔥"} || Likes: 166987
"get_user_replies" Function
async def get_user_replies():
threads_api = ThreadsAPI()
username = "zuck"
user_id = await threads_api.get_user_id_from_username(username)
if user_id:
threads = await threads_api.get_user_replies(user_id)
print(f"The replies for user '{username}' are:")
for thread in threads:
print(f"-\n{thread['thread_items'][0]['post']['user']['username']}'s Post: {thread['thread_items'][0]['post']['caption']} || Likes: {thread['thread_items'][0]['post']['like_count']}")
if len(thread["thread_items"]) > 1:
print(f"{username}'s Reply: {thread['thread_items'][1]['post']['caption']} || Likes: {thread['thread_items'][1]['post']['like_count']}\n-")
else:
print(f"-> You will need to sign up / login to see more.")
else:
print(f"User ID not found for username '{username}'")
Example Output:
mosseri's Post: {'text': 'I joined Meta, then Facebook, 15 years ago today. We were four years old, had ~450 employees, had just translated the site, and had ~70M people.\n\nToday we hit that many signups on Threads. Now signups and retained users are different, and we built Threads on top of an amazing foundation provided by Instagram and by Meta, but there is something elegant about that symmetry.\n\nThank you to the team that actually built this app, thank you to the company and @zuck for trusting me all these years, 🙏🏼'} || Likes: 25523
zuck's Reply: {'text': "Congrats! Great milestone to celebrate 15 years. I'm grateful for everything you do."} || Likes: 5506
-
-
adidas's Post: {'text': 'to sock and slide or not to sock and slide today…'} || Likes: 7425
zuck's Reply: {'text': 'No socks for life'} || Likes: 9976
-
-
evachen212's Post: {'text': 'This is a good first Thread 🙌🏼'} || Likes: 8739
zuck's Reply: {'text': 'Believe when I say, I want it that way.'} || Likes: 23991
-
-
iamsamyrlaine's Post: {'text': "Can't remember the last time I even had the Twitter app on my phone, let alone posted something there; I'm definitely down with Threads though!"} || Likes: 4876
zuck's Reply: {'text': '🙌'} || Likes: 7928
...
"get_post_id_from_url" Function
async def get_post_id_from_url():
threads_api = ThreadsAPI()
post_url = "https://www.threads.net/t/CuZsgfWLyiI"
post_id = await threads_api.get_post_id_from_url(post_url)
print(f"'Thread post {post_id}':")
Example Output:
Thread post_id is 3141737961795561608
"get_post" Function
async def get_post():
threads_api = ThreadsAPI()
post_url = "https://www.threads.net/t/CuZsgfWLyiI"
post_id = await threads_api.get_post_id_from_url(post_url)
thread = await threads_api.get_post(post_id)
print(f"'Thread post {thread['containing_thread']['thread_items'][0]['post']['caption']}':")
for thread in thread["reply_threads"]:
print(f"-\n{thread['thread_items'][0]['post']['user']['username']}'s Post: {thread['thread_items'][0]['post']['caption']} || Likes: {thread['thread_items'][0]['post']['like_count']}")
Example Output:
zuck's post {'text': '70 million sign ups on Threads as of this morning. Way beyond our expectations.'}:
-
luclevesque's Reply: {'text': 'Wow 🤯'} || Likes: 167
-
jasminericegirl's Reply: {'text': 'you are doing amazing sweetie'} || Likes: 391
-
zhra.ghalenoei's Reply: {'text': 'نصفشون ایرانین یَره🤣'} || Likes: 0
-
a.llisterthomas's Reply: {'text': 'elon finna drop this guy😭🥊'} || Likes: 0
-
_vormund_'s Reply: None || Likes: 0
-
sri_ty_'s Reply: {'text': '🐸So nice'} || Likes: 0
-
_william.carrera_'s Reply: {'text': 'Where’s the porn here Mr Zuck'} || Likes: 0
-
kal_blogs's Reply: {'text': 'When you said ‘our’, was it the ‘royal our’?'} || Likes: 0
-
nasheet's Reply: {'text': 'That is crazy road to 100M'} || Likes: 19
-
dsb.don's Reply: {'text': 'You did it 🇰🇪♥️'} || Likes: 0
-
pisceansoulx's Reply: {'text': 'Wohoo. You the man Zucker'} || Likes: 0
-
winchester_757's Reply: {'text': 'If only the meta verse was this good LMAO'} || Likes: 0
-
winchester_757's Reply: {'text': 'Only 10 mil more to match the big guy'} || Likes: 0
"get_post_likes" Function
async def get_post_likes():
api = ThreadsAPI()
post_url = "https://www.threads.net/t/CuZsgfWLyiI"
post_id = await api.get_post_id_from_url(post_url)
likes = await api.get_post_likes(post_id)
number_of_likes_to_display = 10
for user_info in likes[:number_of_likes_to_display]:
print(f'Username: {user_info["username"]} || Full Name: {user_info["full_name"]} || Follower Count: {user_info["follower_count"]} ')
Example Output:
Username: andrew_votava || Full Name: Andrew Votava || Follower Count: 19
Username: herson_theeog || Full Name: Herson_theeOG || Follower Count: 323
Username: dhruv___kanojia || Full Name: Dhruv🌟 || Follower Count: 38
Username: codecrusadepk || Full Name: Code Crusade || Follower Count: 9
Username: toxicated_jeshim_007 || Full Name: Jeshim Akhtar Choudhury || Follower Count: 6
Username: jay.rex.official || Full Name: Jay Rex || Follower Count: 30
Username: jessy.servin || Full Name: Jessica Servín || Follower Count: 343
Username: joshxmadrid || Full Name: Josh Madrid || Follower Count: 1092
Username: ganjipro || Full Name: Song Ganji || Follower Count: 1649
Username: bilalmuhamadi || Full Name: B I L A L M U H A M A D I || Follower Count: 111
"post" Function
async def post():
threads_api = ThreadsAPI()
# either set USERNAME and PASSWORD as environment variables, or replace these with your actual credentials
await threads_api.login(os.environ.get('USERNAME'), os.environ.get('PASSWORD'))
result = await threads_api.post("Hello World!")
if result:
print("Post has been successfully posted")
else:
print("Unable to post.")
Example Output:
Post has been successfully posted
📌 Roadmap
- 🚧 Read feed, notifications
- 🚧 Post text and share a video
- 🚧 Documentation Improvements
- 🚧 CI/CD Improvements
- 🚧 Add coverage Pytest
License
This project is licensed under the MIT license.
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
File details
Details for the file threads-api-1.1.12.tar.gz
.
File metadata
- Download URL: threads-api-1.1.12.tar.gz
- Upload date:
- Size: 24.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.10.6
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 3afc1f2c217945a1c36595cafe3bcf24628352a3ccf3aa0ff3aaf278c0fc61ce |
|
MD5 | d284742f2e4cfb066ba30d3ac47730f4 |
|
BLAKE2b-256 | f14821a94f0498fd9774bf33f4d122ad7feeff352e0d8f2e62cc1fdaeaca7c1c |
Provenance
File details
Details for the file threads_api-1.1.12-py3-none-any.whl
.
File metadata
- Download URL: threads_api-1.1.12-py3-none-any.whl
- Upload date:
- Size: 22.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.10.6
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 5a7fb763e15be17fdfd4400273d77c03bd42308f8617d9f2f3b41ee62ed5fc4a |
|
MD5 | 5512c69ba6d4af94372afe5346f617d0 |
|
BLAKE2b-256 | f1b2a2ee7cb84026f3bfbe1bfaa9d9e151f485518f4e34febe8b22928bd6f5aa |