Compare commits
3 Commits
2395f60d11
...
e99defc533
| Author | SHA1 | Date | |
|---|---|---|---|
| e99defc533 | |||
| 50288e9130 | |||
| c7ef63cc05 |
10
README.md
10
README.md
@@ -8,9 +8,9 @@ This tool performs destructive operations. Only use it if you intend to erase da
|
|||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
We're using [`uv`](https://github.com/astral-sh/uv) for dependency and virtual environment management.
|
Check `pyproject.toml`.
|
||||||
|
|
||||||
You can setup the project (aka create a virtual environment and install dependencies) with :
|
You can use `uv` to install dependencies:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://git.kharec.info/Kharec/skywipe.git
|
git clone https://git.kharec.info/Kharec/skywipe.git
|
||||||
@@ -64,13 +64,17 @@ BE SURE TO USE A [BLUESKY APP PASSWORD](https://blueskyfeeds.com/faq-app-passwor
|
|||||||
- [x] handle configuration logic
|
- [x] handle configuration logic
|
||||||
- [x] sign in to at protocol
|
- [x] sign in to at protocol
|
||||||
- [x] delete posts in batch
|
- [x] delete posts in batch
|
||||||
- [ ] only delete posts with media
|
- [x] only delete posts with media
|
||||||
- [ ] remove likes
|
- [ ] remove likes
|
||||||
- [ ] remove reposts
|
- [ ] remove reposts
|
||||||
- [ ] unfollow accounts
|
- [ ] unfollow accounts
|
||||||
- [ ] make `all` run the other commands
|
- [ ] make `all` run the other commands
|
||||||
- [ ] add simple progress and logging
|
- [ ] add simple progress and logging
|
||||||
- [ ] add safeguards like confirmations and clear dry-run info
|
- [ ] add safeguards like confirmations and clear dry-run info
|
||||||
|
|
||||||
|
Once it's done, we'll think:
|
||||||
|
|
||||||
|
- [ ] decent code architecture
|
||||||
- [ ] installation and run process
|
- [ ] installation and run process
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
from typing import Callable, Dict, Optional
|
from typing import Callable, Dict, Optional
|
||||||
from skywipe.configure import Configuration
|
from skywipe.configure import Configuration
|
||||||
from skywipe.posts import delete_posts
|
from skywipe.posts import delete_posts
|
||||||
|
from skywipe.medias import delete_posts_with_medias
|
||||||
|
|
||||||
|
|
||||||
CommandHandler = Callable[[], None]
|
CommandHandler = Callable[[], None]
|
||||||
@@ -58,7 +59,7 @@ def run_posts():
|
|||||||
|
|
||||||
|
|
||||||
def run_medias():
|
def run_medias():
|
||||||
print("Command 'medias' is not yet implemented.")
|
delete_posts_with_medias()
|
||||||
|
|
||||||
|
|
||||||
def run_likes():
|
def run_likes():
|
||||||
|
|||||||
96
skywipe/medias.py
Normal file
96
skywipe/medias.py
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
"""Media post deletion module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from skywipe.auth import Auth
|
||||||
|
from skywipe.configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
def has_media_embed(post_record):
|
||||||
|
embed = getattr(post_record, 'embed', None)
|
||||||
|
if not embed:
|
||||||
|
return False
|
||||||
|
|
||||||
|
embed_type = getattr(embed, 'py_type', None)
|
||||||
|
media_types = {
|
||||||
|
'app.bsky.embed.images',
|
||||||
|
'app.bsky.embed.video',
|
||||||
|
'app.bsky.embed.external'
|
||||||
|
}
|
||||||
|
|
||||||
|
if embed_type:
|
||||||
|
embed_type_base = embed_type.split('#')[0]
|
||||||
|
|
||||||
|
if embed_type_base in media_types:
|
||||||
|
return True
|
||||||
|
|
||||||
|
if embed_type_base in ('app.bsky.embed.recordWithMedia', 'app.bsky.embed.record_with_media'):
|
||||||
|
media = getattr(embed, 'media', None)
|
||||||
|
if media:
|
||||||
|
media_type = getattr(media, 'py_type', None)
|
||||||
|
if media_type:
|
||||||
|
media_type_base = media_type.split('#')[0]
|
||||||
|
if media_type_base in media_types:
|
||||||
|
return True
|
||||||
|
|
||||||
|
for attr in ('images', 'video', 'external'):
|
||||||
|
if hasattr(embed, attr):
|
||||||
|
return True
|
||||||
|
if isinstance(embed, dict) and embed.get(attr):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def delete_posts_with_medias():
|
||||||
|
auth = Auth()
|
||||||
|
client = auth.login()
|
||||||
|
config = Configuration()
|
||||||
|
config_data = config.load()
|
||||||
|
|
||||||
|
batch_size = config_data.get("batch_size", 10)
|
||||||
|
delay = config_data.get("delay", 1)
|
||||||
|
verbose = config_data.get("verbose", False)
|
||||||
|
|
||||||
|
if verbose:
|
||||||
|
print(
|
||||||
|
f"Starting media post deletion with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
did = client.me.did
|
||||||
|
cursor = None
|
||||||
|
total_deleted = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
if cursor:
|
||||||
|
response = client.get_author_feed(
|
||||||
|
actor=did, limit=batch_size, cursor=cursor)
|
||||||
|
else:
|
||||||
|
response = client.get_author_feed(actor=did, limit=batch_size)
|
||||||
|
|
||||||
|
posts = response.feed
|
||||||
|
if not posts:
|
||||||
|
break
|
||||||
|
|
||||||
|
for post in posts:
|
||||||
|
post_record = post.post
|
||||||
|
|
||||||
|
if not has_media_embed(post_record):
|
||||||
|
if verbose:
|
||||||
|
print(f"Skipping post without media: {post_record.uri}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
client.delete_post(post_record.uri)
|
||||||
|
total_deleted += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Deleted post with media: {post_record.uri}")
|
||||||
|
except Exception as e:
|
||||||
|
if verbose:
|
||||||
|
print(f"Error deleting post {post_record.uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Deleted {total_deleted} posts with media.")
|
||||||
Reference in New Issue
Block a user