Compare commits
30 Commits
e99defc533
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| dba06e642a | |||
| 5868c1649b | |||
| a14184cddc | |||
| 6587f8c39c | |||
| ae6663572c | |||
| 3eb456e999 | |||
| cfa5773e62 | |||
| ddee2a6029 | |||
| 5e2b4f3408 | |||
| c238278df6 | |||
| a9c25c8c10 | |||
| 5ff25b3eb6 | |||
| c396ba8ae9 | |||
| d12f14a994 | |||
| a4b622bfd3 | |||
| 005c76119f | |||
| b1e2b266f4 | |||
| 6475a117e7 | |||
| 28a193078a | |||
| 07bbe88784 | |||
| 59554f6f17 | |||
| 02f609d829 | |||
| c8df3d0460 | |||
| 5afee97259 | |||
| ebbcbeeaa7 | |||
| 044ec67aa3 | |||
| e871d19a9f | |||
| 0ec562e0d2 | |||
| 276d177c4d | |||
| ed0076a34e |
49
README.md
49
README.md
@@ -1,6 +1,6 @@
|
|||||||
# Skywipe
|
# Skywipe
|
||||||
|
|
||||||
Skywipe is a work-in-progress Python 3.13+ CLI that helps you wipe data from your Bluesky account using the AT Protocol.
|
Skywipe is a work-in-progress Python 3.13+ CLI that helps you wipe data from your Bluesky account using the AT Protocol SDK.
|
||||||
|
|
||||||
## Warning
|
## Warning
|
||||||
|
|
||||||
@@ -8,7 +8,7 @@ This tool performs destructive operations. Only use it if you intend to erase da
|
|||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
Check `pyproject.toml`.
|
Check [pyproject.toml](pyproject.toml).
|
||||||
|
|
||||||
You can use `uv` to install dependencies:
|
You can use `uv` to install dependencies:
|
||||||
|
|
||||||
@@ -20,31 +20,21 @@ uv sync
|
|||||||
|
|
||||||
## How to run
|
## How to run
|
||||||
|
|
||||||
When installation will be worked out, you'll be able to :
|
|
||||||
|
|
||||||
```bash
|
|
||||||
skywipe all # target everything
|
|
||||||
skywipe configure # create configuration
|
|
||||||
skywipe posts # only posts
|
|
||||||
skywipe medias # only posts with medias
|
|
||||||
skywipe likes # only likes
|
|
||||||
skywipe reposts # only reposts
|
|
||||||
skywipe follows # only follows
|
|
||||||
```
|
|
||||||
|
|
||||||
While it's being developed, you can use the tool using `uv` :
|
While it's being developed, you can use the tool using `uv` :
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
uv run main.py all # target everything
|
uv run python -m skywipe.cli all # target everything
|
||||||
uv run main.py configure # create configuration
|
uv run python -m skywipe.cli configure # create configuration
|
||||||
uv run main.py posts # only posts
|
uv run python -m skywipe.cli posts # delete posts
|
||||||
uv run main.py medias # only posts with medias
|
uv run python -m skywipe.cli medias # delete posts with medias
|
||||||
uv run main.py likes # only likes
|
uv run python -m skywipe.cli likes # undo likes
|
||||||
uv run main.py reposts # only reposts
|
uv run python -m skywipe.cli reposts # undo reposts
|
||||||
uv run main.py follows # only follows
|
uv run python -m skywipe.cli quotes # delete quotes
|
||||||
|
uv run python -m skywipe.cli follows # unfollow all
|
||||||
|
uv run python -m skywipe.cli bookmarks # delete bookmarks
|
||||||
```
|
```
|
||||||
|
|
||||||
### Configuration
|
## Configuration
|
||||||
|
|
||||||
If you run the tool for the first time, it will prompt you to use `skywipe configure` to create the configuration file, which is located in `~/.config/skywipe/config.yml` :
|
If you run the tool for the first time, it will prompt you to use `skywipe configure` to create the configuration file, which is located in `~/.config/skywipe/config.yml` :
|
||||||
|
|
||||||
@@ -65,15 +55,14 @@ BE SURE TO USE A [BLUESKY APP PASSWORD](https://blueskyfeeds.com/faq-app-passwor
|
|||||||
- [x] sign in to at protocol
|
- [x] sign in to at protocol
|
||||||
- [x] delete posts in batch
|
- [x] delete posts in batch
|
||||||
- [x] only delete posts with media
|
- [x] only delete posts with media
|
||||||
- [ ] remove likes
|
- [x] undo likes
|
||||||
- [ ] remove reposts
|
- [x] undo reposts
|
||||||
- [ ] unfollow accounts
|
- [x] delete quotes
|
||||||
- [ ] make `all` run the other commands
|
- [x] unfollow accounts
|
||||||
|
- [x] remove bookmarks
|
||||||
|
- [x] 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 (confirmation, dry-run flag)
|
||||||
|
|
||||||
Once it's done, we'll think:
|
|
||||||
|
|
||||||
- [ ] decent code architecture
|
- [ ] decent code architecture
|
||||||
- [ ] installation and run process
|
- [ ] installation and run process
|
||||||
|
|
||||||
|
|||||||
@@ -7,4 +7,4 @@ requires-python = ">=3.13"
|
|||||||
dependencies = ["atproto>=0.0.65", "pyyaml>=6.0"]
|
dependencies = ["atproto>=0.0.65", "pyyaml>=6.0"]
|
||||||
|
|
||||||
[project.scripts]
|
[project.scripts]
|
||||||
skywipe = "main:main"
|
skywipe = "skywipe.cli:main"
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
"""Authentication module for Skywipe"""
|
"""Authentication module for Skywipe"""
|
||||||
|
|
||||||
from atproto import Client
|
from atproto import Client
|
||||||
from skywipe.configure import Configuration
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
class Auth:
|
class Auth:
|
||||||
|
|||||||
75
skywipe/bookmarks.py
Normal file
75
skywipe/bookmarks.py
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
"""Bookmark deletion module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from atproto import models
|
||||||
|
from .auth import Auth
|
||||||
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
def delete_bookmarks():
|
||||||
|
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 bookmark deletion with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
cursor = None
|
||||||
|
total_deleted = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
get_params = models.AppBskyBookmarkGetBookmarks.Params(
|
||||||
|
limit=batch_size,
|
||||||
|
cursor=cursor
|
||||||
|
)
|
||||||
|
|
||||||
|
response = client.app.bsky.bookmark.get_bookmarks(params=get_params)
|
||||||
|
|
||||||
|
bookmarks = response.bookmarks
|
||||||
|
if not bookmarks:
|
||||||
|
break
|
||||||
|
|
||||||
|
for bookmark in bookmarks:
|
||||||
|
try:
|
||||||
|
bookmark_uri = None
|
||||||
|
if hasattr(bookmark, "uri"):
|
||||||
|
bookmark_uri = bookmark.uri
|
||||||
|
else:
|
||||||
|
for attr_name in ("subject", "record", "post", "item"):
|
||||||
|
if hasattr(bookmark, attr_name):
|
||||||
|
nested = getattr(bookmark, attr_name)
|
||||||
|
if hasattr(nested, "uri"):
|
||||||
|
bookmark_uri = nested.uri
|
||||||
|
break
|
||||||
|
|
||||||
|
if not bookmark_uri:
|
||||||
|
if verbose:
|
||||||
|
print(f"Skipping bookmark: unable to find uri")
|
||||||
|
continue
|
||||||
|
|
||||||
|
delete_data = models.AppBskyBookmarkDeleteBookmark.Data(
|
||||||
|
uri=bookmark_uri
|
||||||
|
)
|
||||||
|
client.app.bsky.bookmark.delete_bookmark(data=delete_data)
|
||||||
|
total_deleted += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Deleted bookmark: {bookmark_uri}")
|
||||||
|
except Exception as e:
|
||||||
|
bookmark_uri = getattr(bookmark, "uri", "unknown")
|
||||||
|
if verbose:
|
||||||
|
print(f"Error deleting bookmark {bookmark_uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Deleted {total_deleted} bookmarks.")
|
||||||
@@ -3,8 +3,8 @@
|
|||||||
import sys
|
import sys
|
||||||
import argparse
|
import argparse
|
||||||
|
|
||||||
from skywipe.commands import registry
|
from .commands import registry
|
||||||
from skywipe.configure import Configuration
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
def create_parser():
|
def create_parser():
|
||||||
@@ -1,9 +1,14 @@
|
|||||||
"""Command implementations for Skywipe"""
|
"""Command implementations for Skywipe"""
|
||||||
|
|
||||||
from typing import Callable, Dict, Optional
|
from typing import Callable, Dict, Optional
|
||||||
from skywipe.configure import Configuration
|
from .configure import Configuration
|
||||||
from skywipe.posts import delete_posts
|
from .posts import delete_all_posts
|
||||||
from skywipe.medias import delete_posts_with_medias
|
from .medias import delete_posts_with_medias
|
||||||
|
from .likes import undo_likes
|
||||||
|
from .reposts import undo_reposts
|
||||||
|
from .quotes import delete_quotes_posts
|
||||||
|
from .follows import unfollow_all
|
||||||
|
from .bookmarks import delete_bookmarks
|
||||||
|
|
||||||
|
|
||||||
CommandHandler = Callable[[], None]
|
CommandHandler = Callable[[], None]
|
||||||
@@ -55,7 +60,7 @@ def run_configure():
|
|||||||
|
|
||||||
|
|
||||||
def run_posts():
|
def run_posts():
|
||||||
delete_posts()
|
delete_all_posts()
|
||||||
|
|
||||||
|
|
||||||
def run_medias():
|
def run_medias():
|
||||||
@@ -63,23 +68,34 @@ def run_medias():
|
|||||||
|
|
||||||
|
|
||||||
def run_likes():
|
def run_likes():
|
||||||
print("Command 'likes' is not yet implemented.")
|
undo_likes()
|
||||||
|
|
||||||
|
|
||||||
def run_reposts():
|
def run_reposts():
|
||||||
print("Command 'reposts' is not yet implemented.")
|
undo_reposts()
|
||||||
|
|
||||||
|
|
||||||
|
def run_quotes():
|
||||||
|
delete_quotes_posts()
|
||||||
|
|
||||||
|
|
||||||
def run_follows():
|
def run_follows():
|
||||||
print("Command 'follows' is not yet implemented.")
|
unfollow_all()
|
||||||
|
|
||||||
|
|
||||||
|
def run_bookmarks():
|
||||||
|
delete_bookmarks()
|
||||||
|
|
||||||
|
|
||||||
def run_all():
|
def run_all():
|
||||||
registry.execute("posts")
|
commands = ["posts", "likes", "reposts", "follows", "bookmarks"]
|
||||||
registry.execute("medias")
|
|
||||||
registry.execute("likes")
|
for cmd in commands:
|
||||||
registry.execute("reposts")
|
try:
|
||||||
registry.execute("follows")
|
registry.execute(cmd)
|
||||||
|
except Exception as e:
|
||||||
|
print(f"Error running '{cmd}': {e}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
|
||||||
registry.register("configure", run_configure,
|
registry.register("configure", run_configure,
|
||||||
@@ -88,5 +104,7 @@ registry.register("posts", run_posts, "only posts")
|
|||||||
registry.register("medias", run_medias, "only posts with medias")
|
registry.register("medias", run_medias, "only posts with medias")
|
||||||
registry.register("likes", run_likes, "only likes")
|
registry.register("likes", run_likes, "only likes")
|
||||||
registry.register("reposts", run_reposts, "only reposts")
|
registry.register("reposts", run_reposts, "only reposts")
|
||||||
|
registry.register("quotes", run_quotes, "only quotes")
|
||||||
registry.register("follows", run_follows, "only follows")
|
registry.register("follows", run_follows, "only follows")
|
||||||
|
registry.register("bookmarks", run_bookmarks, "only bookmarks")
|
||||||
registry.register("all", run_all, "target everything")
|
registry.register("all", run_all, "target everything")
|
||||||
|
|||||||
69
skywipe/follows.py
Normal file
69
skywipe/follows.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
"""Follow undoing module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from atproto import models
|
||||||
|
from .auth import Auth
|
||||||
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
FOLLOW_COLLECTION = "app.bsky.graph.follow"
|
||||||
|
|
||||||
|
|
||||||
|
def unfollow_all():
|
||||||
|
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 unfollow operation with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
did = client.me.did
|
||||||
|
cursor = None
|
||||||
|
total_unfollowed = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
list_params = models.ComAtprotoRepoListRecords.Params(
|
||||||
|
repo=did,
|
||||||
|
collection=FOLLOW_COLLECTION,
|
||||||
|
limit=batch_size,
|
||||||
|
cursor=cursor
|
||||||
|
)
|
||||||
|
|
||||||
|
response = client.com.atproto.repo.list_records(params=list_params)
|
||||||
|
|
||||||
|
records = response.records
|
||||||
|
if not records:
|
||||||
|
break
|
||||||
|
|
||||||
|
for record in records:
|
||||||
|
try:
|
||||||
|
record_uri = record.uri
|
||||||
|
rkey = record_uri.rsplit("/", 1)[-1]
|
||||||
|
delete_data = {
|
||||||
|
"repo": did,
|
||||||
|
"collection": FOLLOW_COLLECTION,
|
||||||
|
"rkey": rkey
|
||||||
|
}
|
||||||
|
client.com.atproto.repo.delete_record(data=delete_data)
|
||||||
|
total_unfollowed += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Unfollowed: {record_uri}")
|
||||||
|
except Exception as e:
|
||||||
|
record_uri = getattr(record, "uri", "unknown")
|
||||||
|
if verbose:
|
||||||
|
print(f"Error unfollowing {record_uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Unfollowed {total_unfollowed} accounts.")
|
||||||
69
skywipe/likes.py
Normal file
69
skywipe/likes.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
"""Like undoing module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from atproto import models
|
||||||
|
from .auth import Auth
|
||||||
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
LIKE_COLLECTION = "app.bsky.feed.like"
|
||||||
|
|
||||||
|
|
||||||
|
def undo_likes():
|
||||||
|
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 like deletion with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
did = client.me.did
|
||||||
|
cursor = None
|
||||||
|
total_undone = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
list_params = models.ComAtprotoRepoListRecords.Params(
|
||||||
|
repo=did,
|
||||||
|
collection=LIKE_COLLECTION,
|
||||||
|
limit=batch_size,
|
||||||
|
cursor=cursor
|
||||||
|
)
|
||||||
|
|
||||||
|
response = client.com.atproto.repo.list_records(params=list_params)
|
||||||
|
|
||||||
|
records = response.records
|
||||||
|
if not records:
|
||||||
|
break
|
||||||
|
|
||||||
|
for record in records:
|
||||||
|
try:
|
||||||
|
record_uri = record.uri
|
||||||
|
rkey = record_uri.rsplit("/", 1)[-1]
|
||||||
|
delete_data = {
|
||||||
|
"repo": did,
|
||||||
|
"collection": LIKE_COLLECTION,
|
||||||
|
"rkey": rkey
|
||||||
|
}
|
||||||
|
client.com.atproto.repo.delete_record(data=delete_data)
|
||||||
|
total_undone += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Undone like: {record_uri}")
|
||||||
|
except Exception as e:
|
||||||
|
record_uri = getattr(record, "uri", "unknown")
|
||||||
|
if verbose:
|
||||||
|
print(f"Error undoing like {record_uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Undone {total_undone} likes.")
|
||||||
@@ -1,43 +1,8 @@
|
|||||||
"""Media post deletion module for Skywipe"""
|
"""Media post deletion module for Skywipe"""
|
||||||
|
|
||||||
import time
|
import time
|
||||||
from skywipe.auth import Auth
|
from .auth import Auth
|
||||||
from skywipe.configure import Configuration
|
from .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():
|
def delete_posts_with_medias():
|
||||||
@@ -59,11 +24,8 @@ def delete_posts_with_medias():
|
|||||||
total_deleted = 0
|
total_deleted = 0
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
if cursor:
|
|
||||||
response = client.get_author_feed(
|
response = client.get_author_feed(
|
||||||
actor=did, limit=batch_size, cursor=cursor)
|
actor=did, limit=batch_size, cursor=cursor)
|
||||||
else:
|
|
||||||
response = client.get_author_feed(actor=did, limit=batch_size)
|
|
||||||
|
|
||||||
posts = response.feed
|
posts = response.feed
|
||||||
if not posts:
|
if not posts:
|
||||||
@@ -72,7 +34,41 @@ def delete_posts_with_medias():
|
|||||||
for post in posts:
|
for post in posts:
|
||||||
post_record = post.post
|
post_record = post.post
|
||||||
|
|
||||||
if not has_media_embed(post_record):
|
embed = getattr(post_record, 'embed', None)
|
||||||
|
has_media = False
|
||||||
|
if embed:
|
||||||
|
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:
|
||||||
|
has_media = True
|
||||||
|
|
||||||
|
if not has_media and 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:
|
||||||
|
has_media = True
|
||||||
|
|
||||||
|
if not has_media:
|
||||||
|
for attr in ('images', 'video', 'external'):
|
||||||
|
if hasattr(embed, attr):
|
||||||
|
has_media = True
|
||||||
|
break
|
||||||
|
if isinstance(embed, dict) and embed.get(attr):
|
||||||
|
has_media = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if not has_media:
|
||||||
if verbose:
|
if verbose:
|
||||||
print(f"Skipping post without media: {post_record.uri}")
|
print(f"Skipping post without media: {post_record.uri}")
|
||||||
continue
|
continue
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
"""Post deletion module for Skywipe"""
|
"""Post deletion module for Skywipe"""
|
||||||
|
|
||||||
import time
|
import time
|
||||||
from skywipe.auth import Auth
|
from .auth import Auth
|
||||||
from skywipe.configure import Configuration
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
def delete_posts():
|
def delete_all_posts():
|
||||||
auth = Auth()
|
auth = Auth()
|
||||||
client = auth.login()
|
client = auth.login()
|
||||||
config = Configuration()
|
config = Configuration()
|
||||||
|
|||||||
75
skywipe/quotes.py
Normal file
75
skywipe/quotes.py
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
"""Quote post deletion module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from .auth import Auth
|
||||||
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
def delete_quotes_posts():
|
||||||
|
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 quote post deletion with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
did = client.me.did
|
||||||
|
cursor = None
|
||||||
|
total_deleted = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
response = client.get_author_feed(
|
||||||
|
actor=did, limit=batch_size, cursor=cursor)
|
||||||
|
|
||||||
|
posts = response.feed
|
||||||
|
if not posts:
|
||||||
|
break
|
||||||
|
|
||||||
|
for post in posts:
|
||||||
|
post_record = post.post
|
||||||
|
|
||||||
|
embed = getattr(post_record, 'embed', None)
|
||||||
|
has_quote = False
|
||||||
|
if embed:
|
||||||
|
embed_type = getattr(embed, 'py_type', None)
|
||||||
|
if embed_type:
|
||||||
|
embed_type_base = embed_type.split('#')[0]
|
||||||
|
quote_types = {
|
||||||
|
'app.bsky.embed.record',
|
||||||
|
'app.bsky.embed.recordWithMedia',
|
||||||
|
'app.bsky.embed.record_with_media'
|
||||||
|
}
|
||||||
|
if embed_type_base in quote_types:
|
||||||
|
has_quote = True
|
||||||
|
|
||||||
|
if not has_quote and (hasattr(embed, 'record') or (isinstance(embed, dict) and embed.get('record'))):
|
||||||
|
has_quote = True
|
||||||
|
|
||||||
|
if not has_quote:
|
||||||
|
if verbose:
|
||||||
|
print(f"Skipping post without quote: {post_record.uri}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
client.delete_post(post_record.uri)
|
||||||
|
total_deleted += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Deleted quote post: {post_record.uri}")
|
||||||
|
except Exception as e:
|
||||||
|
if verbose:
|
||||||
|
print(f"Error deleting quote post {post_record.uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Deleted {total_deleted} quote posts.")
|
||||||
69
skywipe/reposts.py
Normal file
69
skywipe/reposts.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
"""Repost undoing module for Skywipe"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from atproto import models
|
||||||
|
from .auth import Auth
|
||||||
|
from .configure import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
REPOST_COLLECTION = "app.bsky.feed.repost"
|
||||||
|
|
||||||
|
|
||||||
|
def undo_reposts():
|
||||||
|
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 repost deletion with batch_size={batch_size}, delay={delay}s")
|
||||||
|
|
||||||
|
did = client.me.did
|
||||||
|
cursor = None
|
||||||
|
total_undone = 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
list_params = models.ComAtprotoRepoListRecords.Params(
|
||||||
|
repo=did,
|
||||||
|
collection=REPOST_COLLECTION,
|
||||||
|
limit=batch_size,
|
||||||
|
cursor=cursor
|
||||||
|
)
|
||||||
|
|
||||||
|
response = client.com.atproto.repo.list_records(params=list_params)
|
||||||
|
|
||||||
|
records = response.records
|
||||||
|
if not records:
|
||||||
|
break
|
||||||
|
|
||||||
|
for record in records:
|
||||||
|
try:
|
||||||
|
record_uri = record.uri
|
||||||
|
rkey = record_uri.rsplit("/", 1)[-1]
|
||||||
|
delete_data = {
|
||||||
|
"repo": did,
|
||||||
|
"collection": REPOST_COLLECTION,
|
||||||
|
"rkey": rkey
|
||||||
|
}
|
||||||
|
client.com.atproto.repo.delete_record(data=delete_data)
|
||||||
|
total_undone += 1
|
||||||
|
if verbose:
|
||||||
|
print(f"Undone repost: {record_uri}")
|
||||||
|
except Exception as e:
|
||||||
|
record_uri = getattr(record, "uri", "unknown")
|
||||||
|
if verbose:
|
||||||
|
print(f"Error undoing repost {record_uri}: {e}")
|
||||||
|
|
||||||
|
cursor = response.cursor
|
||||||
|
if not cursor:
|
||||||
|
break
|
||||||
|
|
||||||
|
if delay > 0:
|
||||||
|
time.sleep(delay)
|
||||||
|
|
||||||
|
print(f"Undone {total_undone} reposts.")
|
||||||
Reference in New Issue
Block a user