2017-04-15 12:53:08 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2017-04-14 14:41:09 +00:00
|
|
|
|
2018-01-14 14:34:41 +00:00
|
|
|
import os
|
2017-04-12 14:42:04 +00:00
|
|
|
import sys
|
2017-04-19 12:47:30 +00:00
|
|
|
import logging
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-18 14:40:26 +00:00
|
|
|
from argparse import ArgumentParser, FileType
|
2017-04-19 12:47:30 +00:00
|
|
|
from collections import namedtuple
|
2018-06-12 10:22:16 +00:00
|
|
|
from toot import config, commands, CLIENT_NAME, CLIENT_WEBSITE, __version__
|
2017-12-30 12:32:52 +00:00
|
|
|
from toot.exceptions import ApiError, ConsoleError
|
2017-05-08 07:09:20 +00:00
|
|
|
from toot.output import print_out, print_err
|
2017-04-18 14:40:26 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
VISIBILITY_CHOICES = ['public', 'unlisted', 'private', 'direct']
|
2017-04-18 14:40:26 +00:00
|
|
|
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
def visibility(value):
|
|
|
|
"""Validates the visibilty parameter"""
|
|
|
|
if value not in VISIBILITY_CHOICES:
|
|
|
|
raise ValueError("Invalid visibility value")
|
2017-04-18 14:40:26 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
return value
|
2017-04-18 14:40:26 +00:00
|
|
|
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
Command = namedtuple("Command", ["name", "description", "require_auth", "arguments"])
|
|
|
|
|
|
|
|
|
2018-06-30 07:44:36 +00:00
|
|
|
# Aruguments added to every command
|
2017-05-08 07:09:20 +00:00
|
|
|
common_args = [
|
|
|
|
(["--no-color"], {
|
|
|
|
"help": "don't use ANSI colors in output",
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
2017-08-26 13:12:32 +00:00
|
|
|
}),
|
2018-06-15 07:02:19 +00:00
|
|
|
(["--quiet"], {
|
|
|
|
"help": "don't write to stdout on success",
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
|
|
|
}),
|
2017-08-26 13:12:32 +00:00
|
|
|
(["--debug"], {
|
|
|
|
"help": "show debug log in console",
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
2018-06-30 07:44:36 +00:00
|
|
|
}),
|
|
|
|
]
|
|
|
|
|
|
|
|
# Arguments added to commands which require authentication
|
|
|
|
common_auth_args = [
|
|
|
|
(["-u", "--using"], {
|
|
|
|
"help": "the account to use, overrides active account",
|
|
|
|
}),
|
2017-05-08 07:09:20 +00:00
|
|
|
]
|
|
|
|
|
2017-04-26 09:49:21 +00:00
|
|
|
account_arg = (["account"], {
|
2018-01-02 09:44:32 +00:00
|
|
|
"help": "account name, e.g. 'Gargron@mastodon.social'",
|
2017-04-26 09:49:21 +00:00
|
|
|
})
|
|
|
|
|
2017-08-26 12:39:53 +00:00
|
|
|
instance_arg = (["-i", "--instance"], {
|
|
|
|
"type": str,
|
|
|
|
"help": 'mastodon instance to log into e.g. "mastodon.social"',
|
|
|
|
})
|
|
|
|
|
|
|
|
email_arg = (["-e", "--email"], {
|
|
|
|
"type": str,
|
|
|
|
"help": 'email address to log in with',
|
|
|
|
})
|
|
|
|
|
2018-12-25 01:20:30 +00:00
|
|
|
scheme_arg = (["--disable-https"], {
|
|
|
|
"help": "disable HTTPS and use insecure HTTP",
|
|
|
|
"dest": "scheme",
|
|
|
|
"default": "https",
|
|
|
|
"action": "store_const",
|
|
|
|
"const": "http",
|
|
|
|
})
|
|
|
|
|
2017-04-26 09:49:21 +00:00
|
|
|
|
2017-04-26 10:11:52 +00:00
|
|
|
AUTH_COMMANDS = [
|
2017-04-19 12:47:30 +00:00
|
|
|
Command(
|
|
|
|
name="login",
|
2018-06-15 07:39:28 +00:00
|
|
|
description="Log into a mastodon instance using your browser (recommended)",
|
2018-12-25 01:20:30 +00:00
|
|
|
arguments=[instance_arg, scheme_arg],
|
2017-08-26 12:39:53 +00:00
|
|
|
require_auth=False,
|
|
|
|
),
|
|
|
|
Command(
|
2018-06-15 07:39:28 +00:00
|
|
|
name="login_cli",
|
|
|
|
description="Log in from the console, does NOT support two factor authentication",
|
2018-12-25 01:20:30 +00:00
|
|
|
arguments=[instance_arg, email_arg, scheme_arg],
|
2018-01-02 09:44:32 +00:00
|
|
|
require_auth=False,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="activate",
|
|
|
|
description="Switch between logged in accounts.",
|
|
|
|
arguments=[account_arg],
|
2017-04-19 12:47:30 +00:00
|
|
|
require_auth=False,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="logout",
|
|
|
|
description="Log out, delete stored access keys",
|
2018-01-02 09:44:32 +00:00
|
|
|
arguments=[account_arg],
|
2017-04-19 12:47:30 +00:00
|
|
|
require_auth=False,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="auth",
|
2018-01-02 09:44:32 +00:00
|
|
|
description="Show logged in accounts and instances",
|
2017-04-19 12:47:30 +00:00
|
|
|
arguments=[],
|
|
|
|
require_auth=False,
|
|
|
|
),
|
2017-04-26 10:11:52 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
READ_COMMANDS = [
|
2017-04-19 12:47:30 +00:00
|
|
|
Command(
|
|
|
|
name="whoami",
|
|
|
|
description="Display logged in user details",
|
|
|
|
arguments=[],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
2017-04-19 13:29:40 +00:00
|
|
|
Command(
|
|
|
|
name="whois",
|
2017-04-26 10:11:52 +00:00
|
|
|
description="Display account details",
|
2017-04-19 13:29:40 +00:00
|
|
|
arguments=[
|
|
|
|
(["account"], {
|
|
|
|
"help": "account name or numeric ID"
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
2017-12-29 13:26:40 +00:00
|
|
|
Command(
|
|
|
|
name="instance",
|
|
|
|
description="Display instance details",
|
|
|
|
arguments=[
|
|
|
|
(["instance"], {
|
|
|
|
"help": "instance domain (e.g. 'mastodon.social') or blank to use current",
|
|
|
|
"nargs": "?",
|
|
|
|
}),
|
2018-12-30 08:53:12 +00:00
|
|
|
scheme_arg,
|
2017-12-29 13:26:40 +00:00
|
|
|
],
|
|
|
|
require_auth=False,
|
|
|
|
),
|
2017-04-26 10:11:52 +00:00
|
|
|
Command(
|
|
|
|
name="search",
|
|
|
|
description="Search for users or hashtags",
|
|
|
|
arguments=[
|
|
|
|
(["query"], {
|
|
|
|
"help": "the search query",
|
|
|
|
}),
|
|
|
|
(["-r", "--resolve"], {
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
|
|
|
"help": "Resolve non-local accounts",
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="timeline",
|
2018-06-12 08:40:36 +00:00
|
|
|
description="Show recent items in a timeline (home by default)",
|
2018-03-05 18:13:45 +00:00
|
|
|
arguments=[
|
2018-06-12 08:40:36 +00:00
|
|
|
(["-p", "--public"], {
|
|
|
|
"action": "store_true",
|
|
|
|
"default": False,
|
|
|
|
"help": "Show public timeline.",
|
|
|
|
}),
|
|
|
|
(["-t", "--tag"], {
|
|
|
|
"type": str,
|
|
|
|
"help": "Show timeline for given hashtag.",
|
|
|
|
}),
|
2018-06-12 09:53:10 +00:00
|
|
|
(["-i", "--list"], {
|
2018-06-12 08:40:36 +00:00
|
|
|
"type": int,
|
|
|
|
"help": "Show timeline for given list ID.",
|
|
|
|
}),
|
2018-03-05 18:13:45 +00:00
|
|
|
(["-l", "--local"], {
|
2018-06-12 08:40:36 +00:00
|
|
|
"action": "store_true",
|
2018-03-05 18:13:45 +00:00
|
|
|
"default": False,
|
2018-06-12 08:40:36 +00:00
|
|
|
"help": "Show only statuses from local instance (public and tag timelines only).",
|
2018-03-05 18:13:45 +00:00
|
|
|
}),
|
2018-07-25 18:40:59 +00:00
|
|
|
(["-r", "--reverse"], {
|
|
|
|
"action": "store_true",
|
|
|
|
"default": False,
|
|
|
|
"help": "Reverse the order of the shown timeline (to new posts at the bottom)",
|
|
|
|
}),
|
2018-03-05 18:13:45 +00:00
|
|
|
],
|
2017-04-26 10:11:52 +00:00
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="curses",
|
2017-09-09 07:54:13 +00:00
|
|
|
description="An experimental timeline app (doesn't work on Windows)",
|
2018-01-06 10:25:05 +00:00
|
|
|
arguments=[
|
|
|
|
(["-p", "--public"], {
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
|
|
|
"help": "Resolve non-local accounts",
|
|
|
|
}),
|
|
|
|
(["-i", "--instance"], {
|
|
|
|
"type": str,
|
|
|
|
"help": 'instance from which to read (for public timeline only)',
|
|
|
|
})
|
|
|
|
],
|
|
|
|
require_auth=False,
|
2017-04-26 10:11:52 +00:00
|
|
|
),
|
|
|
|
]
|
|
|
|
|
|
|
|
POST_COMMANDS = [
|
2017-04-19 12:47:30 +00:00
|
|
|
Command(
|
|
|
|
name="post",
|
|
|
|
description="Post a status text to your timeline",
|
|
|
|
arguments=[
|
|
|
|
(["text"], {
|
|
|
|
"help": "The status text to post.",
|
2017-12-30 15:42:52 +00:00
|
|
|
"nargs": "?",
|
2017-04-19 12:47:30 +00:00
|
|
|
}),
|
|
|
|
(["-m", "--media"], {
|
|
|
|
"type": FileType('rb'),
|
|
|
|
"help": "path to the media file to attach"
|
|
|
|
}),
|
|
|
|
(["-v", "--visibility"], {
|
|
|
|
"type": visibility,
|
|
|
|
"default": "public",
|
|
|
|
"help": 'post visibility, one of: %s' % ", ".join(VISIBILITY_CHOICES),
|
2018-06-07 08:04:50 +00:00
|
|
|
}),
|
|
|
|
(["-s", "--sensitive"], {
|
|
|
|
"action": 'store_true',
|
|
|
|
"default": False,
|
|
|
|
"help": "mark the media as NSFW",
|
|
|
|
}),
|
|
|
|
(["-p", "--spoiler-text"], {
|
|
|
|
"type": str,
|
|
|
|
"help": 'text to be shown as a warning before the actual content',
|
2018-06-13 10:43:31 +00:00
|
|
|
}),
|
|
|
|
(["-r", "--reply-to"], {
|
|
|
|
"type": int,
|
|
|
|
"help": 'local ID of the status you want to reply to',
|
|
|
|
}),
|
2017-04-19 12:47:30 +00:00
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="upload",
|
|
|
|
description="Upload an image or video file",
|
|
|
|
arguments=[
|
|
|
|
(["file"], {
|
|
|
|
"help": "Path to the file to upload",
|
|
|
|
"type": FileType('rb')
|
|
|
|
})
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
2018-06-14 08:40:16 +00:00
|
|
|
Command(
|
|
|
|
name="delete",
|
|
|
|
description="Delete an existing status",
|
|
|
|
arguments=[
|
|
|
|
(["status_id"], {
|
|
|
|
"help": "ID of the status to delete",
|
|
|
|
"type": int,
|
|
|
|
})
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
2017-04-26 10:11:52 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
ACCOUNTS_COMMANDS = [
|
2017-04-19 12:47:30 +00:00
|
|
|
Command(
|
|
|
|
name="follow",
|
|
|
|
description="Follow an account",
|
|
|
|
arguments=[
|
2017-04-26 09:49:21 +00:00
|
|
|
account_arg,
|
2017-04-19 12:47:30 +00:00
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="unfollow",
|
|
|
|
description="Unfollow an account",
|
|
|
|
arguments=[
|
2017-04-26 09:49:21 +00:00
|
|
|
account_arg,
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="mute",
|
|
|
|
description="Mute an account",
|
|
|
|
arguments=[
|
|
|
|
account_arg,
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="unmute",
|
|
|
|
description="Unmute an account",
|
|
|
|
arguments=[
|
|
|
|
account_arg,
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="block",
|
|
|
|
description="Block an account",
|
|
|
|
arguments=[
|
|
|
|
account_arg,
|
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
Command(
|
|
|
|
name="unblock",
|
|
|
|
description="Unblock an account",
|
|
|
|
arguments=[
|
|
|
|
account_arg,
|
2017-04-19 12:47:30 +00:00
|
|
|
],
|
|
|
|
require_auth=True,
|
|
|
|
),
|
|
|
|
]
|
2017-04-18 14:40:26 +00:00
|
|
|
|
2017-04-26 10:11:52 +00:00
|
|
|
COMMANDS = AUTH_COMMANDS + READ_COMMANDS + POST_COMMANDS + ACCOUNTS_COMMANDS
|
|
|
|
|
2017-04-18 14:40:26 +00:00
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
def print_usage():
|
2017-04-26 10:11:52 +00:00
|
|
|
max_name_len = max(len(command.name) for command in COMMANDS)
|
|
|
|
|
|
|
|
groups = [
|
|
|
|
("Authentication", AUTH_COMMANDS),
|
|
|
|
("Read", READ_COMMANDS),
|
|
|
|
("Post", POST_COMMANDS),
|
|
|
|
("Accounts", ACCOUNTS_COMMANDS),
|
|
|
|
]
|
|
|
|
|
2017-05-08 07:09:20 +00:00
|
|
|
print_out("<green>{}</green>".format(CLIENT_NAME))
|
2018-06-12 10:22:16 +00:00
|
|
|
print_out("<blue>v{}</blue>".format(__version__))
|
2017-04-19 12:47:30 +00:00
|
|
|
|
2017-04-26 10:11:52 +00:00
|
|
|
for name, cmds in groups:
|
2017-05-08 07:09:20 +00:00
|
|
|
print_out("")
|
|
|
|
print_out(name + ":")
|
2017-04-19 12:47:30 +00:00
|
|
|
|
2017-04-26 10:11:52 +00:00
|
|
|
for cmd in cmds:
|
2017-05-08 07:09:20 +00:00
|
|
|
cmd_name = cmd.name.ljust(max_name_len + 2)
|
|
|
|
print_out(" <yellow>toot {}</yellow> {}".format(cmd_name, cmd.description))
|
2017-04-19 12:47:30 +00:00
|
|
|
|
2017-05-08 07:09:20 +00:00
|
|
|
print_out("")
|
|
|
|
print_out("To get help for each command run:")
|
|
|
|
print_out(" <yellow>toot <command> --help</yellow>")
|
|
|
|
print_out("")
|
|
|
|
print_out("<green>{}</green>".format(CLIENT_WEBSITE))
|
2017-04-15 10:00:05 +00:00
|
|
|
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
def get_argument_parser(name, command):
|
|
|
|
parser = ArgumentParser(
|
|
|
|
prog='toot %s' % name,
|
|
|
|
description=command.description,
|
|
|
|
epilog=CLIENT_WEBSITE)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2018-06-30 07:44:36 +00:00
|
|
|
combined_args = command.arguments + common_args
|
2018-01-02 09:44:32 +00:00
|
|
|
if command.require_auth:
|
2018-06-30 07:44:36 +00:00
|
|
|
combined_args += common_auth_args
|
|
|
|
|
|
|
|
for args, kwargs in combined_args:
|
|
|
|
parser.add_argument(*args, **kwargs)
|
2018-01-02 09:44:32 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
return parser
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-15 10:12:33 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
def run_command(app, user, name, args):
|
|
|
|
command = next((c for c in COMMANDS if c.name == name), None)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
if not command:
|
2017-05-08 07:09:20 +00:00
|
|
|
print_err("Unknown command '{}'\n".format(name))
|
2017-04-19 12:47:30 +00:00
|
|
|
print_usage()
|
2017-04-16 15:15:05 +00:00
|
|
|
return
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
parser = get_argument_parser(name, command)
|
|
|
|
parsed_args = parser.parse_args(args)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
2018-01-02 09:44:32 +00:00
|
|
|
# Override the active account if 'using' option is given
|
|
|
|
if command.require_auth and parsed_args.using:
|
|
|
|
user, app = config.get_user_app(parsed_args.using)
|
|
|
|
if not user or not app:
|
|
|
|
raise ConsoleError("User '{}' not found".format(parsed_args.using))
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
if command.require_auth and (not user or not app):
|
2017-05-08 07:09:20 +00:00
|
|
|
print_err("This command requires that you are logged in.")
|
|
|
|
print_err("Please run `toot login` first.")
|
2017-04-13 11:52:28 +00:00
|
|
|
return
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
fn = commands.__dict__.get(name)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
2017-04-19 13:29:40 +00:00
|
|
|
if not fn:
|
|
|
|
raise NotImplementedError("Command '{}' does not have an implementation.".format(name))
|
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
return fn(app, user, parsed_args)
|
2017-04-13 11:52:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
2018-01-14 14:34:41 +00:00
|
|
|
# Enable debug logging if --debug is in args
|
2017-08-26 13:12:32 +00:00
|
|
|
if "--debug" in sys.argv:
|
2018-01-14 14:34:41 +00:00
|
|
|
filename = os.getenv("TOOT_LOG_FILE")
|
|
|
|
logging.basicConfig(level=logging.DEBUG, filename=filename)
|
2017-04-12 14:42:04 +00:00
|
|
|
|
2017-04-19 09:03:44 +00:00
|
|
|
# If something is piped in, append it to commandline arguments
|
|
|
|
if not sys.stdin.isatty():
|
2017-05-07 08:28:11 +00:00
|
|
|
stdin = sys.stdin.read()
|
|
|
|
if stdin:
|
|
|
|
sys.argv.append(stdin)
|
2017-04-19 09:03:44 +00:00
|
|
|
|
2017-04-19 12:47:30 +00:00
|
|
|
command_name = 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-19 12:47:30 +00:00
|
|
|
if not command_name:
|
2017-04-13 11:52:28 +00:00
|
|
|
return print_usage()
|
|
|
|
|
2018-01-02 09:44:32 +00:00
|
|
|
user, app = config.get_active_user_app()
|
2017-04-19 12:47:30 +00:00
|
|
|
|
2017-04-13 11:52:28 +00:00
|
|
|
try:
|
2017-04-19 12:47:30 +00:00
|
|
|
run_command(app, user, command_name, args)
|
2018-06-15 08:01:18 +00:00
|
|
|
except (ConsoleError, ApiError) as e:
|
2017-05-08 07:09:20 +00:00
|
|
|
print_err(str(e))
|
2017-05-08 07:11:20 +00:00
|
|
|
sys.exit(1)
|
2018-06-15 08:01:18 +00:00
|
|
|
except KeyboardInterrupt as e:
|
|
|
|
pass
|