2017-04-15 12:53:08 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2017-04-14 14:41:09 +00:00
|
|
|
from __future__ import print_function
|
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
import os
|
|
|
|
import sys
|
2017-04-13 11:52:28 +00:00
|
|
|
import logging
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-13 11:14:01 +00:00
|
|
|
from bs4 import BeautifulSoup
|
2017-04-12 14:42:04 +00:00
|
|
|
from builtins import input
|
2017-04-13 11:14:01 +00:00
|
|
|
from datetime import datetime
|
2017-04-15 10:00:05 +00:00
|
|
|
from future.moves.itertools import zip_longest
|
2017-04-12 14:42:04 +00:00
|
|
|
from getpass import getpass
|
2017-04-13 11:14:01 +00:00
|
|
|
from itertools import chain
|
2017-04-16 12:06:16 +00:00
|
|
|
from argparse import ArgumentParser, FileType
|
2017-04-13 11:14:01 +00:00
|
|
|
from textwrap import TextWrapper
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-16 12:14:33 +00:00
|
|
|
from toot import DEFAULT_INSTANCE
|
2017-04-16 13:07:27 +00:00
|
|
|
from toot.api import create_app, login, post_status, timeline_home, upload_media, search
|
2017-04-16 12:14:33 +00:00
|
|
|
from toot.config import save_user, load_user, load_app, save_app, CONFIG_APP_FILE, CONFIG_USER_FILE
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
class ConsoleError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2017-04-14 14:41:09 +00:00
|
|
|
def red(text):
|
|
|
|
return "\033[31m{}\033[0m".format(text)
|
|
|
|
|
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
def green(text):
|
2017-04-14 14:41:09 +00:00
|
|
|
return "\033[32m{}\033[0m".format(text)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
|
2017-04-14 14:41:09 +00:00
|
|
|
def yellow(text):
|
|
|
|
return "\033[33m{}\033[0m".format(text)
|
|
|
|
|
|
|
|
|
|
|
|
def print_error(text):
|
|
|
|
print(red(text), file=sys.stderr)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
def create_app_interactive():
|
2017-04-13 11:52:28 +00:00
|
|
|
instance = input("Choose an instance [%s]: " % green(DEFAULT_INSTANCE))
|
2017-04-12 14:42:04 +00:00
|
|
|
if not instance:
|
|
|
|
instance = DEFAULT_INSTANCE
|
|
|
|
|
|
|
|
base_url = 'https://{}'.format(instance)
|
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
print("Registering application with %s" % green(base_url))
|
|
|
|
try:
|
|
|
|
app = create_app(base_url)
|
|
|
|
except:
|
|
|
|
raise ConsoleError("Failed authenticating application. Did you enter a valid instance?")
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
save_app(app)
|
2017-04-13 11:52:28 +00:00
|
|
|
print("Application tokens saved to: {}".format(green(CONFIG_APP_FILE)))
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-12 15:12:47 +00:00
|
|
|
return app
|
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
def login_interactive(app):
|
|
|
|
print("\nLog in to " + green(app.base_url))
|
|
|
|
email = input('Email: ')
|
|
|
|
password = getpass('Password: ')
|
|
|
|
|
|
|
|
print("Authenticating...")
|
2017-04-13 11:52:28 +00:00
|
|
|
try:
|
|
|
|
user = login(app, email, password)
|
|
|
|
except:
|
|
|
|
raise ConsoleError("Login failed")
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
save_user(user)
|
|
|
|
print("User token saved to " + green(CONFIG_USER_FILE))
|
|
|
|
|
|
|
|
return user
|
|
|
|
|
|
|
|
|
|
|
|
def print_usage():
|
|
|
|
print("toot - interact with Mastodon from the command line")
|
|
|
|
print("")
|
|
|
|
print("Usage:")
|
2017-04-13 11:52:28 +00:00
|
|
|
print(" toot login - log into a Mastodon instance (saves access tokens to `~/.config/toot/`)")
|
|
|
|
print(" toot logout - log out (delete saved access tokens)")
|
|
|
|
print(" toot auth - shows currently logged in user and instance")
|
|
|
|
print(" toot post <msg> - toot a new post to your timeline")
|
2017-04-16 13:07:27 +00:00
|
|
|
print(" toot search - search for accounts or hashtags")
|
2017-04-13 11:52:28 +00:00
|
|
|
print(" toot timeline - shows your public timeline")
|
2017-04-12 14:42:04 +00:00
|
|
|
print("")
|
2017-04-15 10:12:33 +00:00
|
|
|
print("To get help for each command run:")
|
|
|
|
print(" toot <command> --help")
|
|
|
|
print("")
|
2017-04-12 14:42:04 +00:00
|
|
|
print("https://github.com/ihabunek/toot")
|
|
|
|
|
|
|
|
|
2017-04-13 11:14:01 +00:00
|
|
|
def print_timeline(item):
|
|
|
|
def wrap_text(text, width):
|
|
|
|
wrapper = TextWrapper(width=width, break_long_words=False, break_on_hyphens=False)
|
|
|
|
return chain(*[wrapper.wrap(l) for l in text.split("\n")])
|
|
|
|
|
|
|
|
def timeline_rows(item):
|
|
|
|
name = item['name']
|
|
|
|
time = item['time'].strftime('%Y-%m-%d %H:%M%Z')
|
|
|
|
|
|
|
|
left_column = [name, time]
|
|
|
|
if 'reblogged' in item:
|
|
|
|
left_column.append(item['reblogged'])
|
|
|
|
|
|
|
|
text = item['text']
|
|
|
|
|
|
|
|
right_column = wrap_text(text, 80)
|
|
|
|
|
|
|
|
return zip_longest(left_column, right_column, fillvalue="")
|
|
|
|
|
|
|
|
for left, right in timeline_rows(item):
|
|
|
|
print("{:30} │ {}".format(left, right))
|
|
|
|
|
|
|
|
|
|
|
|
def parse_timeline(item):
|
|
|
|
content = item['reblog']['content'] if item['reblog'] else item['content']
|
|
|
|
reblogged = item['reblog']['account']['username'] if item['reblog'] else ""
|
|
|
|
|
|
|
|
name = item['account']['display_name'] + " @" + item['account']['username']
|
|
|
|
soup = BeautifulSoup(content, "html.parser")
|
|
|
|
text = soup.get_text().replace(''', "'")
|
|
|
|
time = datetime.strptime(item['created_at'], "%Y-%m-%dT%H:%M:%S.%fZ")
|
|
|
|
|
|
|
|
return {
|
|
|
|
"name": name,
|
|
|
|
"text": text,
|
|
|
|
"time": time,
|
|
|
|
"reblogged": reblogged,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def cmd_timeline(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot timeline",
|
|
|
|
description="Show recent items in your public timeline",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
|
|
|
|
args = parser.parse_args(args)
|
|
|
|
|
2017-04-13 11:14:01 +00:00
|
|
|
items = timeline_home(app, user)
|
|
|
|
parsed_items = [parse_timeline(t) for t in items]
|
|
|
|
|
|
|
|
print("─" * 31 + "┬" + "─" * 88)
|
|
|
|
for item in parsed_items:
|
|
|
|
print_timeline(item)
|
|
|
|
print("─" * 31 + "┼" + "─" * 88)
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def visibility(value):
|
|
|
|
if value not in ['public', 'unlisted', 'private', 'direct']:
|
|
|
|
raise ValueError("Invalid visibility value")
|
2017-04-15 10:00:05 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
return value
|
2017-04-15 10:00:05 +00:00
|
|
|
|
2017-04-15 10:39:14 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def cmd_post_status(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot post",
|
|
|
|
description="Post a status text to the timeline",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
parser.add_argument("text", help="The status text to post.")
|
|
|
|
parser.add_argument("-m", "--media", type=FileType('rb'),
|
|
|
|
help="path to the media file to attach")
|
|
|
|
parser.add_argument("-v", "--visibility", type=visibility, default="public",
|
|
|
|
help='post visibility, either "public" (default), "direct", "private", or "unlisted"')
|
2017-04-15 10:00:05 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
args = parser.parse_args(args)
|
2017-04-15 10:00:05 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
if args.media:
|
|
|
|
media = do_upload(app, user, args.media)
|
2017-04-15 10:00:05 +00:00
|
|
|
media_ids = [media['id']]
|
|
|
|
else:
|
|
|
|
media_ids = None
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
response = post_status(app, user, args.text, media_ids=media_ids, visibility=args.visibility)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-12 15:12:47 +00:00
|
|
|
print("Toot posted: " + green(response.get('url')))
|
2017-04-12 14:42:04 +00:00
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def cmd_auth(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot auth",
|
|
|
|
description="Show login details",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
parser.parse_args(args)
|
2017-04-15 10:12:33 +00:00
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
if app and user:
|
2017-04-13 11:14:01 +00:00
|
|
|
print("You are logged in to " + green(app.base_url))
|
2017-04-12 14:42:04 +00:00
|
|
|
print("Username: " + green(user.username))
|
2017-04-13 11:14:01 +00:00
|
|
|
print("App data: " + green(CONFIG_APP_FILE))
|
|
|
|
print("User data: " + green(CONFIG_USER_FILE))
|
2017-04-12 14:42:04 +00:00
|
|
|
else:
|
|
|
|
print("You are not logged in")
|
|
|
|
|
|
|
|
|
2017-04-15 10:12:33 +00:00
|
|
|
def cmd_login():
|
2017-04-16 12:06:16 +00:00
|
|
|
parser = ArgumentParser(prog="toot login",
|
|
|
|
description="Log into a Mastodon instance",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
2017-04-15 10:12:33 +00:00
|
|
|
parser.parse_args()
|
|
|
|
|
|
|
|
app = create_app_interactive()
|
|
|
|
user = login_interactive(app)
|
|
|
|
|
|
|
|
return app, user
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def cmd_logout(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot logout",
|
|
|
|
description="Log out, delete stored access keys",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
parser.parse_args(args)
|
2017-04-15 10:12:33 +00:00
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
os.unlink(CONFIG_APP_FILE)
|
|
|
|
os.unlink(CONFIG_USER_FILE)
|
|
|
|
print("You are now logged out")
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def cmd_upload(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot upload",
|
|
|
|
description="Upload an image or video file",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
parser.add_argument("file", help="Path to the file to upload", type=FileType('rb'))
|
2017-04-15 10:12:33 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
args = parser.parse_args(args)
|
2017-04-14 14:41:09 +00:00
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
response = do_upload(app, user, args.file)
|
2017-04-14 14:41:09 +00:00
|
|
|
|
|
|
|
print("\nSuccessfully uploaded media ID {}, type '{}'".format(
|
|
|
|
yellow(response['id']), yellow(response['type'])))
|
|
|
|
print("Original URL: " + green(response['url']))
|
|
|
|
print("Preview URL: " + green(response['preview_url']))
|
|
|
|
print("Text URL: " + green(response['text_url']))
|
|
|
|
|
|
|
|
|
2017-04-16 13:07:27 +00:00
|
|
|
def _print_accounts(accounts):
|
|
|
|
if not accounts:
|
|
|
|
return
|
|
|
|
|
|
|
|
print("\nAccounts:")
|
|
|
|
for account in accounts:
|
|
|
|
acct = green("@{}".format(account['acct']))
|
|
|
|
display_name = account['display_name']
|
|
|
|
print("* {} {}".format(acct, display_name))
|
|
|
|
|
|
|
|
|
|
|
|
def _print_hashtags(hashtags):
|
|
|
|
if not hashtags:
|
|
|
|
return
|
|
|
|
|
|
|
|
print("\nHashtags:")
|
|
|
|
print(", ".join([green("#" + t) for t in hashtags]))
|
|
|
|
|
|
|
|
|
|
|
|
def cmd_search(app, user, args):
|
|
|
|
parser = ArgumentParser(prog="toot serach",
|
|
|
|
description="Search for content",
|
|
|
|
epilog="https://github.com/ihabunek/toot")
|
|
|
|
|
|
|
|
parser.add_argument("query", help="The search query")
|
|
|
|
parser.add_argument("-r", "--resolve", action='store_true', default=False,
|
|
|
|
help="Whether to resolve non-local accounts")
|
|
|
|
|
|
|
|
args = parser.parse_args(args)
|
|
|
|
|
|
|
|
response = search(app, user, args.query, args.resolve)
|
|
|
|
|
|
|
|
_print_accounts(response['accounts'])
|
|
|
|
_print_hashtags(response['hashtags'])
|
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def do_upload(app, user, file):
|
|
|
|
print("Uploading media: {}".format(green(file.name)))
|
|
|
|
return upload_media(app, user, file)
|
2017-04-15 10:00:05 +00:00
|
|
|
|
|
|
|
|
2017-04-16 12:06:16 +00:00
|
|
|
def run_command(command, args):
|
2017-04-13 11:52:28 +00:00
|
|
|
app = load_app()
|
|
|
|
user = load_user()
|
|
|
|
|
|
|
|
# Commands which can run when not logged in
|
|
|
|
if command == 'login':
|
2017-04-15 10:12:33 +00:00
|
|
|
return cmd_login()
|
2017-04-13 11:52:28 +00:00
|
|
|
|
|
|
|
if command == 'auth':
|
2017-04-16 12:06:16 +00:00
|
|
|
return cmd_auth(app, user, args)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
|
|
|
# Commands which require user to be logged in
|
|
|
|
if not app or not user:
|
2017-04-16 13:07:27 +00:00
|
|
|
print_error("You are not logged in.")
|
|
|
|
print_error("Please run `toot login` first.")
|
2017-04-13 11:52:28 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if command == 'logout':
|
2017-04-16 12:06:16 +00:00
|
|
|
return cmd_logout(app, user, args)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
if command == 'post':
|
2017-04-16 12:06:16 +00:00
|
|
|
return cmd_post_status(app, user, args)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
|
|
|
if command == 'timeline':
|
2017-04-16 12:06:16 +00:00
|
|
|
return cmd_timeline(app, user, args)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
2017-04-14 14:41:09 +00:00
|
|
|
if command == 'upload':
|
2017-04-16 12:06:16 +00:00
|
|
|
return cmd_upload(app, user, args)
|
2017-04-14 14:41:09 +00:00
|
|
|
|
2017-04-16 13:07:27 +00:00
|
|
|
if command == 'search':
|
|
|
|
return cmd_search(app, user, args)
|
|
|
|
|
|
|
|
print_error("Unknown command '{}'\n".format(command))
|
2017-04-13 11:52:28 +00:00
|
|
|
print_usage()
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2017-04-12 14:42:04 +00:00
|
|
|
if os.getenv('TOOT_DEBUG'):
|
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
command = sys.argv[1] if len(sys.argv) > 1 else None
|
2017-04-16 12:06:16 +00:00
|
|
|
args = sys.argv[2:]
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
if not command:
|
|
|
|
return print_usage()
|
|
|
|
|
|
|
|
try:
|
2017-04-16 12:06:16 +00:00
|
|
|
run_command(command, args)
|
2017-04-13 11:52:28 +00:00
|
|
|
except ConsoleError as e:
|
2017-04-14 14:41:09 +00:00
|
|
|
print_error(str(e))
|