diff --git a/.envrc b/.envrc
new file mode 100644
index 0000000..3550a30
--- /dev/null
+++ b/.envrc
@@ -0,0 +1 @@
+use flake
diff --git a/.gitignore b/.gitignore
index b8c6716..4dbdfbb 100644
--- a/.gitignore
+++ b/.gitignore
@@ -19,3 +19,4 @@ worker_*.lock
.idea/moa.iml
Pipfile
Pipefile.lock
+/.direnv
diff --git a/.vscode/settings.json b/.vscode/settings.json
new file mode 100644
index 0000000..30e36e9
--- /dev/null
+++ b/.vscode/settings.json
@@ -0,0 +1,3 @@
+{
+ "python.formatting.provider": "yapf"
+}
diff --git a/app.py b/app.py
index 8a53602..7734583 100644
--- a/app.py
+++ b/app.py
@@ -3,12 +3,12 @@ import os
import random
from datetime import datetime, timedelta
from urllib.error import URLError
+import sys
import pandas as pd
import pygal
import twitter
from authlib.common.errors import AuthlibBaseError
-from authlib.integrations._client import MissingRequestTokenError
from flask import Flask, flash, g, redirect, render_template, request, session, url_for
from flask_migrate import Migrate
from authlib.integrations.flask_client import OAuth
@@ -30,12 +30,17 @@ from moa.forms import MastodonIDForm, SettingsForm
from moa.helpers import blacklisted, email_bridge_details, send_blacklisted_email, timespan, FORMAT
from moa.models import Bridge, MastodonHost, TSettings, WorkerStat, metadata, BridgeStat, BridgeMetadata
+
+class MissingRequestTokenError(Exception):
+ pass
+
+
app = Flask(__name__)
formatter = logging.Formatter(FORMAT)
# initialize the log handler
-logHandler = logging.FileHandler('logs/app.log')
+logHandler = logging.StreamHandler(sys.stderr)
logHandler.setFormatter(formatter)
# set the app logger level
@@ -45,7 +50,6 @@ app.logger.addHandler(logHandler)
app.logger.info("Starting up...")
-
config = os.environ.get('MOA_CONFIG', 'DevelopmentConfig')
# this is needed to get this in line with other modules, where we expect MOA_CONFIG to be an object in config.py.
app.config.from_object('config.' + config)
@@ -59,14 +63,15 @@ if app.config['SENTRY_DSN']:
from sentry_sdk.integrations.flask import FlaskIntegration
sentry_logging = LoggingIntegration(
- level=logging.INFO, # Capture info and above as breadcrumbs
- event_level=logging.FATAL # Only send fatal errors as events
+ level=logging.INFO, # Capture info and above as breadcrumbs
+ event_level=logging.FATAL # Only send fatal errors as events
)
- sentry_sdk.init(
- dsn=app.config['SENTRY_DSN'],
- integrations=[FlaskIntegration(), sentry_logging, SqlalchemyIntegration()]
- )
+ sentry_sdk.init(dsn=app.config['SENTRY_DSN'],
+ integrations=[
+ FlaskIntegration(), sentry_logging,
+ SqlalchemyIntegration()
+ ])
db = SQLAlchemy(metadata=metadata)
migrate = Migrate(app, db)
@@ -76,17 +81,17 @@ oauth = OAuth(app)
if app.config.get('TWITTER_CONSUMER_KEY', None):
oauth.register(
- name='twitter',
- client_id=app.config['TWITTER_CONSUMER_KEY'],
- client_secret=app.config['TWITTER_CONSUMER_SECRET'],
- request_token_url='https://api.twitter.com/oauth/request_token',
- request_token_params=None,
- access_token_url='https://api.twitter.com/oauth/access_token',
- access_token_params=None,
- authorize_url='https://api.twitter.com/oauth/authenticate',
- authorize_params=None,
- api_base_url='https://api.twitter.com/1.1/',
- client_kwargs=None,
+ name='twitter',
+ client_id=app.config['TWITTER_CONSUMER_KEY'],
+ client_secret=app.config['TWITTER_CONSUMER_SECRET'],
+ request_token_url='https://api.twitter.com/oauth/request_token',
+ request_token_params=None,
+ access_token_url='https://api.twitter.com/oauth/access_token',
+ access_token_params=None,
+ authorize_url='https://api.twitter.com/oauth/authenticate',
+ authorize_params=None,
+ api_base_url='https://api.twitter.com/1.1/',
+ client_kwargs=None,
)
# mastodon_scopes = ["write:statuses", "write:media", "read:accounts", "read:statuses"]
@@ -116,28 +121,32 @@ def index():
form = SettingsForm(obj=settings)
if 'bridge_id' in session:
- bridge = db.session.query(Bridge).filter_by(id=session['bridge_id']).first()
+ bridge = db.session.query(Bridge).filter_by(
+ id=session['bridge_id']).first()
if bridge:
g.bridge = bridge
settings = bridge.t_settings
- app.logger.debug(f"Existing settings found: {enabled} {settings.__dict__}")
+ app.logger.debug(
+ f"Existing settings found: {enabled} {settings.__dict__}")
form = SettingsForm(obj=settings)
if not bridge.mastodon_access_code or not bridge.twitter_oauth_token:
form.remove_masto_and_twitter_fields()
- return render_template('index.html.j2',
- form=form,
- mform=mform,
- )
+ return render_template(
+ 'index.html.j2',
+ form=form,
+ mform=mform,
+ )
@app.route('/options', methods=["POST"])
def options():
if 'bridge_id' in session:
- bridge = db.session.query(Bridge).filter_by(id=session['bridge_id']).first()
+ bridge = db.session.query(Bridge).filter_by(
+ id=session['bridge_id']).first()
else:
flash('ERROR: Please log in to an account')
return redirect(url_for('index'))
@@ -177,11 +186,11 @@ def catch_up_twitter(bridge):
if bridge.twitter_last_id == 0 and bridge.twitter_oauth_token:
# get twitter ID
twitter_api = twitter.Api(
- consumer_key=app.config['TWITTER_CONSUMER_KEY'],
- consumer_secret=app.config['TWITTER_CONSUMER_SECRET'],
- access_token_key=bridge.twitter_oauth_token,
- access_token_secret=bridge.twitter_oauth_secret,
- tweet_mode='extended' # Allow tweets longer than 140 raw characters
+ consumer_key=app.config['TWITTER_CONSUMER_KEY'],
+ consumer_secret=app.config['TWITTER_CONSUMER_SECRET'],
+ access_token_key=bridge.twitter_oauth_token,
+ access_token_secret=bridge.twitter_oauth_secret,
+ tweet_mode='extended' # Allow tweets longer than 140 raw characters
)
try:
tl = twitter_api.GetUserTimeline()
@@ -190,7 +199,8 @@ def catch_up_twitter(bridge):
else:
if len(tl) > 0:
bridge.twitter_last_id = tl[0].id
- d = datetime.strptime(tl[0].created_at, '%a %b %d %H:%M:%S %z %Y')
+ d = datetime.strptime(tl[0].created_at,
+ '%a %b %d %H:%M:%S %z %Y')
bridge.md.last_tweet = d
else:
bridge.twitter_last_id = 0
@@ -225,13 +235,15 @@ def catch_up_mastodon(bridge):
@app.route('/delete', methods=["POST"])
def delete():
if 'bridge_id' in session:
- bridge = db.session.query(Bridge).filter_by(id=session['bridge_id']).first()
+ bridge = db.session.query(Bridge).filter_by(
+ id=session['bridge_id']).first()
if bridge:
app.logger.info(f"Deleting settings for Bridge {bridge.id}")
settings = bridge.t_settings
md = bridge.md
- db.session.query(BridgeStat).filter_by(bridge_id=bridge.id).delete()
+ db.session.query(BridgeStat).filter_by(
+ bridge_id=bridge.id).delete()
db.session.delete(bridge)
db.session.delete(settings)
db.session.delete(md)
@@ -246,11 +258,9 @@ def delete():
@app.route('/twitter_login')
def twitter_login():
- callback_url = url_for(
- 'twitter_oauthorized',
- _external=True,
- next=request.args.get('next')
- )
+ callback_url = url_for('twitter_oauthorized',
+ _external=True,
+ next=request.args.get('next'))
app.logger.debug(callback_url)
@@ -273,9 +283,12 @@ def twitter_oauthorized():
return redirect(url_for('index'))
if resp is None:
- flash('ERROR: You denied the request to sign in or have cookies disabled.')
+ flash(
+ 'ERROR: You denied the request to sign in or have cookies disabled.'
+ )
- elif blacklisted(resp['screen_name'], app.config.get('TWITTER_BLACKLIST', [])):
+ elif blacklisted(resp['screen_name'],
+ app.config.get('TWITTER_BLACKLIST', [])):
flash('ERROR: Access Denied.')
send_blacklisted_email(app, resp['screen_name'])
@@ -306,18 +319,18 @@ def twitter_oauthorized():
def get_or_create_host(hostname):
- mastodonhost = db.session.query(MastodonHost).filter_by(hostname=hostname).first()
+ mastodonhost = db.session.query(MastodonHost).filter_by(
+ hostname=hostname).first()
if not mastodonhost:
try:
client_id, client_secret = Mastodon.create_app(
- "Moa",
- scopes=mastodon_scopes,
- api_base_url=f"https://{hostname}",
- website="https://moa.party/",
- redirect_uris=url_for("mastodon_oauthorized", _external=True)
- )
+ "Moa",
+ scopes=mastodon_scopes,
+ api_base_url=f"https://{hostname}",
+ website="https://moa.party/",
+ redirect_uris=url_for("mastodon_oauthorized", _external=True))
app.logger.info(f"New host created for {hostname}")
@@ -344,13 +357,11 @@ def mastodon_api(hostname, access_code=None):
mastodonhost = get_or_create_host(hostname)
if mastodonhost:
- api = Mastodon(
- client_id=mastodonhost.client_id,
- client_secret=mastodonhost.client_secret,
- api_base_url=f"https://{mastodonhost.hostname}",
- access_token=access_code,
- debug_requests=False
- )
+ api = Mastodon(client_id=mastodonhost.client_id,
+ client_secret=mastodonhost.client_secret,
+ api_base_url=f"https://{mastodonhost.hostname}",
+ access_token=access_code,
+ debug_requests=False)
return api
return None
@@ -401,6 +412,10 @@ def mastodon_login():
if host in app.config.get('MASTODON_BLACKLIST', []):
flash('Access Denied')
return redirect(url_for('index'))
+ allow_list = app.config.get('MASTODON_ALLOWLIST', [])
+ if len(allow_list) != 0 and host not in allow_list:
+ flash('Access Denied')
+ return redirect(url_for('index'))
session['mastodon_host'] = host
@@ -408,12 +423,11 @@ def mastodon_login():
if api:
return redirect(
- api.auth_request_url(
- scopes=mastodon_scopes,
- force_login=True,
- redirect_uris=url_for("mastodon_oauthorized", _external=True)
- )
- )
+ api.auth_request_url(scopes=mastodon_scopes,
+ force_login=True,
+ redirect_uris=url_for(
+ "mastodon_oauthorized",
+ _external=True)))
else:
flash(f"There was a problem connecting to the mastodon server.")
else:
@@ -435,7 +449,9 @@ def mastodon_oauthorized():
app.logger.info(f"Authorization code {authorization_code} for {host}")
if not host:
- flash('There was an error. Please ensure you allow this site to use cookies.')
+ flash(
+ 'There was an error. Please ensure you allow this site to use cookies.'
+ )
return redirect(url_for('index'))
session.pop('mastodon_host', None)
@@ -445,19 +461,23 @@ def mastodon_oauthorized():
local_scopes = mastodon_scopes
try:
- access_code = api.log_in(
- code=authorization_code,
- scopes=local_scopes,
- redirect_uri=url_for("mastodon_oauthorized", _external=True)
- )
+ access_code = api.log_in(code=authorization_code,
+ scopes=local_scopes,
+ redirect_uri=url_for(
+ "mastodon_oauthorized",
+ _external=True))
except MastodonAPIError as e:
# Possibly a scopes problem?
- flash(f"There was a problem connecting to the mastodon server. The error was {e}")
+ flash(
+ f"There was a problem connecting to the mastodon server. The error was {e}"
+ )
return redirect(url_for('index'))
except MastodonIllegalArgumentError as e:
- flash(f"There was a problem connecting to the mastodon server. The error was {e}")
+ flash(
+ f"There was a problem connecting to the mastodon server. The error was {e}"
+ )
return redirect(url_for('index'))
# app.logger.info(f"Access code {access_code}")
@@ -468,13 +488,17 @@ def mastodon_oauthorized():
creds = api.account_verify_credentials()
except (MastodonUnauthorizedError, MastodonAPIError) as e:
- flash(f"There was a problem connecting to the mastodon server. The error was {e}")
+ flash(
+ f"There was a problem connecting to the mastodon server. The error was {e}"
+ )
return redirect(url_for('index'))
username = creds["username"]
account_id = creds["id"]
- bridge = db.session.query(Bridge).filter_by(mastodon_account_id=account_id, mastodon_host_id=masto_host.id).first()
+ bridge = db.session.query(Bridge).filter_by(
+ mastodon_account_id=account_id,
+ mastodon_host_id=masto_host.id).first()
if bridge:
session['bridge_id'] = bridge.id
@@ -520,7 +544,9 @@ def instagram_activate():
# app.logger.info(redirect_uri)
scope = ["basic"]
- api = InstagramAPI(client_id=client_id, client_secret=client_secret, redirect_uri=redirect_uri)
+ api = InstagramAPI(client_id=client_id,
+ client_secret=client_secret,
+ redirect_uri=redirect_uri)
try:
redirect_uri = api.get_authorize_login_url(scope=scope)
@@ -540,7 +566,9 @@ def instagram_oauthorized():
client_id = app.config['INSTAGRAM_CLIENT_ID']
client_secret = app.config['INSTAGRAM_SECRET']
redirect_uri = url_for('instagram_oauthorized', _external=True)
- api = InstagramAPI(client_id=client_id, client_secret=client_secret, redirect_uri=redirect_uri)
+ api = InstagramAPI(client_id=client_id,
+ client_secret=client_secret,
+ redirect_uri=redirect_uri)
try:
access_token = api.exchange_code_for_access_token(code)
@@ -565,15 +593,18 @@ def instagram_oauthorized():
bridge.instagram_account_id = data['id']
bridge.instagram_handle = data['username']
- user_api = InstagramAPI(access_token=bridge.instagram_access_code, client_secret=client_secret)
+ user_api = InstagramAPI(access_token=bridge.instagram_access_code,
+ client_secret=client_secret)
try:
- latest_media, _ = user_api.user_recent_media(user_id=bridge.instagram_account_id, count=1)
+ latest_media, _ = user_api.user_recent_media(
+ user_id=bridge.instagram_account_id, count=1)
except Exception:
latest_media = []
if len(latest_media) > 0:
- bridge.instagram_last_id = datetime_to_timestamp(latest_media[0].created_time)
+ bridge.instagram_last_id = datetime_to_timestamp(
+ latest_media[0].created_time)
else:
bridge.instagram_last_id = 0
@@ -595,8 +626,7 @@ def logout():
def stats():
hours = request.args.get('hours', 24)
- return render_template('stats.html.j2',
- hours=hours)
+ return render_template('stats.html.j2', hours=hours)
@app.route('/deactivate_account')
@@ -634,9 +664,10 @@ def time_graph():
hours = int(request.args.get('hours', 24))
since = datetime.now() - timedelta(hours=hours)
- stats_query = db.session.query(WorkerStat).filter(WorkerStat.created > since).with_entities(WorkerStat.created,
- WorkerStat.time,
- WorkerStat.worker)
+ stats_query = db.session.query(WorkerStat).filter(
+ WorkerStat.created > since).with_entities(WorkerStat.created,
+ WorkerStat.time,
+ WorkerStat.worker)
df = pd.read_sql(stats_query.statement, stats_query.session.bind)
@@ -682,10 +713,11 @@ def count_graph():
hours = int(request.args.get('hours', 24))
since = datetime.now() - timedelta(hours=hours)
- stats_query = db.session.query(WorkerStat).filter(WorkerStat.created > since).with_entities(WorkerStat.created,
- WorkerStat.toots,
- WorkerStat.tweets,
- WorkerStat.instas)
+ stats_query = db.session.query(WorkerStat).filter(
+ WorkerStat.created > since).with_entities(WorkerStat.created,
+ WorkerStat.toots,
+ WorkerStat.tweets,
+ WorkerStat.instas)
df = pd.read_sql(stats_query.statement, stats_query.session.bind)
df.set_index(['created'], inplace=True)
@@ -700,9 +732,10 @@ def count_graph():
tweets = r['tweets'].tolist()
instas = r['instas'].tolist()
- chart = pygal.StackedBar(title=f"# of Incoming Messages ({timespan(hours)})\n{total} total",
- human_readable=True,
- legend_at_bottom=True)
+ chart = pygal.StackedBar(
+ title=f"# of Incoming Messages ({timespan(hours)})\n{total} total",
+ human_readable=True,
+ legend_at_bottom=True)
chart.add('Toots', toots)
chart.add('Tweets', tweets)
chart.add('Instas', instas)
@@ -715,10 +748,11 @@ def percent_graph():
hours = int(request.args.get('hours', 24))
since = datetime.now() - timedelta(hours=hours)
- stats_query = db.session.query(WorkerStat).filter(WorkerStat.created > since).with_entities(WorkerStat.created,
- WorkerStat.toots,
- WorkerStat.tweets,
- WorkerStat.instas)
+ stats_query = db.session.query(WorkerStat).filter(
+ WorkerStat.created > since).with_entities(WorkerStat.created,
+ WorkerStat.toots,
+ WorkerStat.tweets,
+ WorkerStat.instas)
df = pd.read_sql(stats_query.statement, stats_query.session.bind)
df.set_index(['created'], inplace=True)
@@ -736,9 +770,10 @@ def percent_graph():
tweets_p = r['tweets_p'].tolist()
instas_p = r['instas_p'].tolist()
- chart = pygal.StackedBar(title=f"Ratio of Incoming Messages ({timespan(hours)})",
- human_readable=True,
- legend_at_bottom=True)
+ chart = pygal.StackedBar(
+ title=f"Ratio of Incoming Messages ({timespan(hours)})",
+ human_readable=True,
+ legend_at_bottom=True)
chart.add('Toots', toots_p)
chart.add('Tweets', tweets_p)
chart.add('Instas', instas_p)
@@ -751,7 +786,8 @@ def user_graph():
hours = int(request.args.get('hours', 24))
since = datetime.now() - timedelta(hours=hours)
- stats_query = db.session.query(Bridge).filter(Bridge.created > since).filter(Bridge.enabled == 1).with_entities(
+ stats_query = db.session.query(Bridge).filter(
+ Bridge.created > since).filter(Bridge.enabled == 1).with_entities(
Bridge.created)
base_count_query = db.session.query(func.count(Bridge.id)).scalar()
@@ -793,4 +829,4 @@ def page_not_found(e):
if __name__ == '__main__':
- app.run()
+ app.run(host="::1")
diff --git a/flake.lock b/flake.lock
new file mode 100644
index 0000000..ddbf123
--- /dev/null
+++ b/flake.lock
@@ -0,0 +1,66 @@
+{
+ "nodes": {
+ "flake-utils": {
+ "locked": {
+ "lastModified": 1667395993,
+ "narHash": "sha256-nuEHfE/LcWyuSWnS8t12N1wc105Qtau+/OdUAjtQ0rA=",
+ "owner": "numtide",
+ "repo": "flake-utils",
+ "rev": "5aed5285a952e0b949eb3ba02c12fa4fcfef535f",
+ "type": "github"
+ },
+ "original": {
+ "owner": "numtide",
+ "repo": "flake-utils",
+ "type": "github"
+ }
+ },
+ "nix-packages": {
+ "inputs": {
+ "flake-utils": [
+ "flake-utils"
+ ],
+ "nixpkgs": [
+ "nixpkgs"
+ ]
+ },
+ "locked": {
+ "lastModified": 1667423076,
+ "narHash": "sha256-0QUa2SBk/+S+q4hBpTSembUtHdiWjIex7tgb6jIoPkA=",
+ "ref": "refs/heads/main",
+ "rev": "31d36925f77febc11189dbc015f2593f17b9a4ff",
+ "revCount": 331,
+ "type": "git",
+ "url": "https://git.chir.rs/darkkirb/nix-packages"
+ },
+ "original": {
+ "type": "git",
+ "url": "https://git.chir.rs/darkkirb/nix-packages"
+ }
+ },
+ "nixpkgs": {
+ "locked": {
+ "lastModified": 1667456235,
+ "narHash": "sha256-bhRXd2WJt6uFTNPGVGJ0/KjssSmmz4kls/1Ggm3gZ0s=",
+ "owner": "NixOS",
+ "repo": "nixpkgs",
+ "rev": "60974988b3e1e5d61341a5ef0357e7ae99fcf7f1",
+ "type": "github"
+ },
+ "original": {
+ "owner": "NixOS",
+ "repo": "nixpkgs",
+ "type": "github"
+ }
+ },
+ "root": {
+ "inputs": {
+ "flake-utils": "flake-utils",
+ "nix-packages": "nix-packages",
+ "nixpkgs": "nixpkgs"
+ }
+ }
+ },
+ "root": "root",
+ "version": 7
+}
diff --git a/flake.nix b/flake.nix
new file mode 100644
index 0000000..ff25756
--- /dev/null
+++ b/flake.nix
@@ -0,0 +1,53 @@
+{
+ inputs = {
+ nixpkgs.url = github:NixOS/nixpkgs;
+ flake-utils.url = github:numtide/flake-utils;
+ nix-packages.url = git+https://git.chir.rs/darkkirb/nix-packages;
+ nix-packages.inputs = {
+ nixpkgs.follows = "nixpkgs";
+ flake-utils.follows = "flake-utils";
+ };
+ };
+ outputs = {
+ self,
+ nixpkgs,
+ flake-utils,
+ nix-packages,
+ }:
+ flake-utils.lib.eachDefaultSystem (system: let
+ pkgs = import nixpkgs {inherit system;};
+ nix-pkgs = import nix-packages {inherit pkgs;};
+ in {
+ formatter = pkgs.alejandra;
+ devShells.default = pkgs.mkShell {
+ MOA_CONFIG = "DevelopmentConfig";
+ nativeBuildInputs = [
+ (pkgs.python3.withPackages (ps:
+ with ps; [
+ certifi
+ flask
+ flask_sqlalchemy
+ flask_mail
+ flask_migrate
+ flask_wtf
+ mastodon-py
+ pandas
+ psutil
+ pygal
+ python-twitter
+ pymysql
+ sentry-sdk
+ authlib
+ cairosvg
+ werkzeug
+ wheel
+ setuptools
+ nix-pkgs.python-instagram
+ psycopg2
+ ]))
+ pkgs.yapf
+ pkgs.sqlite
+ ];
+ };
+ });
+}
diff --git a/migrations/alembic.ini b/migrations/alembic.ini
index f8ed480..5301449 100644
--- a/migrations/alembic.ini
+++ b/migrations/alembic.ini
@@ -7,6 +7,7 @@
# set to 'true' to run the environment during
# the 'revision' command, regardless of autogenerate
# revision_environment = false
+script_location = .
# Logging configuration
diff --git a/migrations/env.py b/migrations/env.py
index 690d48b..c996c74 100755
--- a/migrations/env.py
+++ b/migrations/env.py
@@ -18,6 +18,7 @@ logger = logging.getLogger('alembic.env')
# from myapp import mymodel
# target_metadata = mymodel.Base.metadata
from flask import current_app
+
config.set_main_option('sqlalchemy.url',
current_app.config.get('SQLALCHEMY_DATABASE_URI'))
target_metadata = current_app.extensions['migrate'].db.metadata
@@ -82,6 +83,7 @@ def run_migrations_online():
finally:
connection.close()
+
if context.is_offline_mode():
run_migrations_offline()
else:
diff --git a/migrations/versions/b0780999e063_allow_deny_list.py b/migrations/versions/b0780999e063_allow_deny_list.py
new file mode 100644
index 0000000..5344b25
--- /dev/null
+++ b/migrations/versions/b0780999e063_allow_deny_list.py
@@ -0,0 +1,38 @@
+"""Allow/Deny list
+
+Revision ID: b0780999e063
+Revises: 3ac471544742
+Create Date: 2022-11-03 07:53:00
+
+"""
+from alembic import op
+import sqlalchemy as sa
+
+# revision identifiers, used by Alembic.
+revision = 'b0780999e063'
+down_revision = '3ac471544742'
+branch_labels = None
+depends_on = None
+
+
+def upgrade():
+ # ### commands auto generated by Alembic - please adjust! ###
+ op.create_table('block_allow_list',
+ sa.Column('id', sa.Integer(), nullable=False),
+ sa.Column('pattern', sa.String(length=100),
+ nullable=False),
+ sa.Column('allow', sa.Boolean(), nullable=False),
+ sa.Column('settings_id', sa.Integer(), nullable=False),
+ sa.PrimaryKeyConstraint('id'),
+ mysql_charset='utf8mb4',
+ mysql_collate='utf8mb4_general_ci')
+ op.create_foreign_key(None, 'block_allow_list', 'settings',
+ ['settings_id'], ['id'])
+
+ # ### end Alembic commands ###
+
+
+def downgrade():
+ # ### commands auto generated by Alembic - please adjust! ###
+ op.drop_table('block_allow_list')
+ # ### end Alembic commands ###
diff --git a/moa/models.py b/moa/models.py
index e57604b..050e87c 100644
--- a/moa/models.py
+++ b/moa/models.py
@@ -51,22 +51,33 @@ CON_XP_ONLYIF_TAGS = ['moa', 'xp']
CON_XP_UNLESS = 'unless'
CON_XP_UNLESS_TAGS = ['nomoa', 'noxp']
+
class TSettings(Base):
__tablename__ = 'settings'
- __table_args__ = {'mysql_charset': 'utf8mb4', 'mysql_collate': 'utf8mb4_general_ci'}
+ __table_args__ = {
+ 'mysql_charset': 'utf8mb4',
+ 'mysql_collate': 'utf8mb4_general_ci'
+ }
id = Column(Integer, primary_key=True)
bridge = relationship('Bridge', backref='t_settings', lazy='dynamic')
- conditional_posting = Column(String(10), nullable=False, server_default=CON_XP_DISABLED, default=CON_XP_DISABLED)
+ allow_deny_list = relationship('AllowDenyList', lazy='dynamic')
+ conditional_posting = Column(String(10),
+ nullable=False,
+ server_default=CON_XP_DISABLED,
+ default=CON_XP_DISABLED)
# Masto -> Twitter
- post_to_twitter = Column(Boolean, nullable=False, default=True) # This means post public toots
+ post_to_twitter = Column(Boolean, nullable=False,
+ default=True) # This means post public toots
post_private_to_twitter = Column(Boolean, nullable=False, default=False)
post_unlisted_to_twitter = Column(Boolean, nullable=False, default=False)
split_twitter_messages = Column(Boolean, nullable=False, default=True)
post_boosts_to_twitter = Column(Boolean, nullable=False, default=True)
post_sensitive_behind_link = Column(Boolean, nullable=False, default=False)
- sensitive_link_text = Column(String(100), nullable=False, default='(NSFW Image)')
+ sensitive_link_text = Column(String(100),
+ nullable=False,
+ default='(NSFW Image)')
remove_cw = Column(Boolean, nullable=False, default=False)
# Twitter -> Masto
@@ -75,7 +86,9 @@ class TSettings(Base):
post_quotes_to_mastodon = Column(Boolean, nullable=False, default=True)
toot_visibility = Column(String(40), nullable=False, default='public')
tweets_behind_cw = Column(Boolean, nullable=False, default=False)
- tweet_cw_text = Column(String(100), nullable=False, default="From birdsite")
+ tweet_cw_text = Column(String(100),
+ nullable=False,
+ default="From birdsite")
instagram_post_to_twitter = Column(Boolean, nullable=False, default=False)
instagram_post_to_mastodon = Column(Boolean, nullable=False, default=False)
@@ -116,6 +129,19 @@ class TSettings(Base):
self.post_rts_to_mastodon
+class AllowDenyList(Base):
+ __tablename__ = "block_allow_list"
+ __table_args__ = {
+ 'mysql_charset': 'utf8mb4',
+ 'mysql_collate': 'utf8mb4_general_ci'
+ }
+
+ id = Column(Integer, primary_key=True)
+ pattern = Column(String(100), nullable=False)
+ allow = Column(Boolean, nullable=False)
+ settings_id = Column(Integer, ForeignKey('settings.id'), nullable=False)
+
+
class Bridge(Base):
__tablename__ = 'bridge'
@@ -139,7 +165,9 @@ class Bridge(Base):
instagram_handle = Column(String(30))
t_settings_id = Column(Integer, ForeignKey('settings.id'), nullable=True)
- metadata_id = Column(Integer, ForeignKey('bridgemetadata.id'), nullable=True)
+ metadata_id = Column(Integer,
+ ForeignKey('bridgemetadata.id'),
+ nullable=True)
created = Column(DateTime, default=datetime.utcnow)
updated = Column(DateTime)
@@ -258,7 +286,7 @@ if __name__ == '__main__':
import pymysql
engine = create_engine(config.SQLALCHEMY_DATABASE_URI)
- metadata = MetaData(engine, reflect=True)
+ metadata = MetaData(engine)
print("Creating Tables")
Base.metadata.create_all(engine)
diff --git a/moa/toot.py b/moa/toot.py
index 08aa61d..afce494 100644
--- a/moa/toot.py
+++ b/moa/toot.py
@@ -8,11 +8,11 @@ from moa.message import Message
from moa.models import CON_XP_ONLYIF, CON_XP_ONLYIF_TAGS, CON_XP_UNLESS, CON_XP_UNLESS_TAGS
from moa.tweet import HOUR_CUTOFF
-MY_TLDS = [
- "shop"
-]
+MY_TLDS = ["shop"]
-URL_REGEXP = re.compile(r"([--:\w?@%&+~#=]*\.[a-z]{2,4}\/{0,2})((?:[?&](?:\w+)=(?:\w+))+|[--:\w?@%&+~#=]+)?", re.U | re.I)
+URL_REGEXP = re.compile(
+ r"([--:\w?@%&+~#=]*\.[a-z]{2,4}\/{0,2})((?:[?&](?:\w+)=(?:\w+))+|[--:\w?@%&+~#=]+)?",
+ re.U | re.I)
logger = logging.getLogger('worker')
@@ -69,7 +69,8 @@ class Toot(Message):
@property
def is_self_reply(self):
- return self.is_reply and self.data['in_reply_to_account_id'] == self.data['account']['id']
+ return self.is_reply and self.data[
+ 'in_reply_to_account_id'] == self.data['account']['id']
@property
def is_boost(self):
@@ -92,7 +93,7 @@ class Toot(Message):
@property
def media_attachments(self):
if self.is_boost:
- return self.data['reblog']['media_attachments']
+ return []
else:
return self.data['media_attachments']
@@ -111,7 +112,6 @@ class Toot(Message):
@property
def should_skip(self):
-
if self.too_old:
logger.info(f'Skipping because >= {HOUR_CUTOFF} hours old.')
return True
@@ -125,6 +125,30 @@ class Toot(Message):
logger.info(f'Skipping reply.')
return True
+ found = True
+ for entry in self.settings.allow_deny_list:
+ hasEntry = True
+ toot_tags = {x.name for x in self.data['tags']}
+ if entry.allow:
+ if entry.pattern not in self.content:
+ found = False
+ break
+ if entry.pattern.startswith(
+ "#") and entry.pattern[1:] not in toot_tags:
+ found = False
+ break
+ else:
+ if entry.pattern in self.content:
+ found = False
+ break
+ if entry.pattern.startswith(
+ "#") and entry.pattern[1:] in toot_tags:
+ found = False
+ break
+
+ if not found:
+ return True
+
if self.visibility == 'private' and not self.settings.post_private_to_twitter:
logger.info(f'Skipping: Not Posting Private toots.')
return True
@@ -202,12 +226,14 @@ class Toot(Message):
status_length += self.url_length
if self.is_sensitive and self.settings.post_sensitive_behind_link:
- status_length += len(f"\n{self.settings.sensitive_link_text}\n{self.url}")
+ status_length += len(
+ f"\n{self.settings.sensitive_link_text}\n{self.url}")
return status_length
def sanitize_twitter_handles(self):
- self.content = re.sub(r'@?(\w{1,15})@twitter.com', '\g<1>', self.content)
+ self.content = re.sub(r'@?(\w{1,15})@twitter.com', '\g<1>',
+ self.content)
# find possible twitter handles so we can get their ranges
tm = list(re.finditer(r'@(\w{1,15})', self.content))
@@ -250,50 +276,55 @@ class Toot(Message):
@property
def clean_content(self):
- media_regexp = re.compile(re.escape(self.instance_url) + "\/media\/[\w-]+\s?")
+ media_regexp = re.compile(
+ re.escape(self.instance_url) + "\/media\/[\w-]+\s?")
if not self.content:
self.content = self.raw_content
# We trust mastodon to return valid HTML
- self.content = re.sub(r']*href="([^"]+)">[^<]*', '\g<1>', self.content)
+ self.content = re.sub(r']*href="([^"]+)">([^<]*)',
+ '\g<1> (\g<2>)', self.content)
# We replace html br with new lines
- self.content = "\n".join(re.compile(r'
', re.IGNORECASE).split(self.content))
+ self.content = "\n".join(
+ re.compile(r'
', re.IGNORECASE).split(self.content))
# We must also replace new paragraphs with double line skips
- self.content = "\n\n".join(re.compile(r'
', re.IGNORECASE).split(self.content)) + self.content = "\n\n".join( + re.compile(r'
', re.IGNORECASE).split(self.content)) # Then we can delete the other html contents and unescape the string - self.content = html.unescape(str(re.compile(r'<.*?>').sub("", self.content).strip())) + self.content = html.unescape( + str(re.compile(r'<.*?>').sub("", self.content).strip())) # Trim out media URLs self.content = re.sub(media_regexp, "", self.content) # fix up masto mentions for mention in self.mentions: - self.content = re.sub(f'@({mention[0]})(?!@)', f"{mention[1]}", self.content) + self.content = re.sub(f'@({mention[0]})(?!@)', f"{mention[1]}", + self.content) if self.config.SANITIZE_TWITTER_HANDLES: self.sanitize_twitter_handles() else: - self.content = re.sub(r'@(\w{1,15})@twitter.com', '@\g<1>', self.content) + self.content = re.sub(r'@(\w{1,15})@twitter.com', '@\g<1>', + self.content) self.content = self.content.strip() if self.spoiler_text and not self.settings.remove_cw: self.content = f"CW: {self.spoiler_text}\n\n{self.content}" - if self.is_sensitive and self.settings.post_sensitive_behind_link and len(self.media_attachments) > 0: + if self.is_sensitive and self.settings.post_sensitive_behind_link and len( + self.media_attachments) > 0: self.content = f"{self.content}\n{self.settings.sensitive_link_text}\n{self.url}" if self.is_boost: - if len(self.content) > 0: - self.content = f"RT {self.boost_author}\n{self.content}\n{self.url}" - else: - self.content = f"RT {self.boost_author}\n{self.url}\n" + self.content = f"RT {self.boost_author}\n{self.url}\n" # logger.debug(self.content) @@ -318,7 +349,9 @@ class Toot(Message): words = self.clean_content.split(" ") if self.settings.split_twitter_messages: - logger.info(f'Toot bigger than {max_length} characters, need to split...') + logger.info( + f'Toot bigger than {max_length} characters, need to split...' + ) for next_word in words: @@ -348,7 +381,8 @@ class Toot(Message): self.message_parts.append(current_part.strip()) for i, msg in enumerate(self.message_parts): - self.message_parts[i] = msg.replace('XXXXX', f"({i+1}/{len(self.message_parts)})") + self.message_parts[i] = msg.replace( + 'XXXXX', f"({i+1}/{len(self.message_parts)})") logger.debug(self.message_parts[i]) else: logger.info('Truncating toot') diff --git a/moa/tweet.py b/moa/tweet.py index b156967..c5c556a 100644 --- a/moa/tweet.py +++ b/moa/tweet.py @@ -20,6 +20,7 @@ HANDLE_SUFFIX = '' class Tweet(Message): + def __init__(self, settings, data, api): super().__init__(settings, data) @@ -38,7 +39,8 @@ class Tweet(Message): @property def created_at(self): - return datetime.strptime(self.data.created_at, '%a %b %d %H:%M:%S %z %Y') + return datetime.strptime(self.data.created_at, + '%a %b %d %H:%M:%S %z %Y') @property def too_old(self) -> bool: @@ -48,6 +50,8 @@ class Tweet(Message): @property def media(self): + if self.is_retweet: + return [] if not self.__fetched_attachments: @@ -67,13 +71,11 @@ class Tweet(Message): target_id = self.data.id try: - fetched_tweet = self.api.GetStatus( - status_id=target_id, - trim_user=True, - include_my_retweet=False, - include_entities=True, - include_ext_alt_text=True - ) + fetched_tweet = self.api.GetStatus(status_id=target_id, + trim_user=True, + include_my_retweet=False, + include_entities=True, + include_ext_alt_text=True) self.__fetched_attachments = fetched_tweet.media except (TwitterError, ConnectionError) as e: @@ -122,7 +124,7 @@ class Tweet(Message): logger.info(f'Skipping because {local_tags} found') return True - if not self.settings.post_to_mastodon: + elif not self.settings.post_to_mastodon: logger.info(f'Skipping regular tweets.') return True @@ -130,7 +132,7 @@ class Tweet(Message): @property def url(self): - base = "https://twitter.com" + base = "https://twitter.catcatnya.com" user = self.data.user.screen_name status = self.data.id @@ -185,9 +187,11 @@ class Tweet(Message): def mentions(self): if self.is_retweet: - m = [(u.screen_name, u._json['indices']) for u in self.data.retweeted_status.user_mentions] + m = [(u.screen_name, u._json['indices']) + for u in self.data.retweeted_status.user_mentions] else: - m = [(u.screen_name, u._json['indices']) for u in self.data.user_mentions] + m = [(u.screen_name, u._json['indices']) + for u in self.data.user_mentions] return m @@ -195,7 +199,8 @@ class Tweet(Message): def quoted_mentions(self): if self.data.quoted_status: - m = [(u.screen_name, u._json['indices']) for u in self.data.quoted_status.user_mentions] + m = [(u.screen_name, u._json['indices']) + for u in self.data.quoted_status.user_mentions] return m @@ -246,12 +251,14 @@ class Tweet(Message): content = re.sub(r'https://twitter.com/.*$', '', content) quoted_text = self.data.quoted_status.full_text - quoted_text = self.expand_handles(quoted_text, self.quoted_mentions) + quoted_text = self.expand_handles(quoted_text, + self.quoted_mentions) quoted_text = html.unescape(quoted_text) for url in self.data.quoted_status.urls: # Unshorten URLs - quoted_text = re.sub(url.url, url.expanded_url, quoted_text) + quoted_text = re.sub(url.url, url.expanded_url, + quoted_text) else: content = self.data.full_text @@ -271,6 +278,8 @@ class Tweet(Message): content = re.sub(url.url, url.expanded_url, content) if self.is_retweet: + self.__content = f"RT @{self.data.retweeted_status.user.screen_name}{HANDLE_SUFFIX}\n{self.url}" + return self.__content if len(content) > 0: content = f"RT @{self.data.retweeted_status.user.screen_name}{HANDLE_SUFFIX}\n{content}" else: @@ -310,6 +319,8 @@ class Tweet(Message): @property def media_attachments(self): + if self.is_retweet: + return [] attachments = [] @@ -366,13 +377,16 @@ class Tweet(Message): except (ConnectionError, NewConnectionError) as e: logger.error(f"{e}") attachment_url = None - raise MoaMediaUploadException("Connection Error fetching attachments") + raise MoaMediaUploadException( + "Connection Error fetching attachments") else: attachment_url = attachment.media_url if attachment_url: - attachments.append({'url': attachment_url, - 'description': attachment.ext_alt_text}) + attachments.append({ + 'url': attachment_url, + 'description': attachment.ext_alt_text + }) return attachments diff --git a/moa/worker.py b/moa/worker.py index 733cb48..9c89934 100644 --- a/moa/worker.py +++ b/moa/worker.py @@ -45,15 +45,22 @@ if c.SENTRY_DSN: from sentry_sdk.integrations.logging import LoggingIntegration sentry_logging = LoggingIntegration( - level=logging.INFO, # Capture info and above as breadcrumbs - event_level=logging.FATAL # Only send fatal errors as events + level=logging.INFO, # Capture info and above as breadcrumbs + event_level=logging.FATAL # Only send fatal errors as events ) - sentry_sdk.init(dsn=c.SENTRY_DSN, integrations=[sentry_logging, - SqlalchemyIntegration(), - FlaskIntegration()]) + sentry_sdk.init(dsn=c.SENTRY_DSN, + integrations=[ + sentry_logging, + SqlalchemyIntegration(), + FlaskIntegration() + ]) parser = argparse.ArgumentParser(description='Moa Worker') -parser.add_argument('--worker', dest='worker', type=int, required=False, default=1) +parser.add_argument('--worker', + dest='worker', + type=int, + required=False, + default=1) args = parser.parse_args() worker_stat = WorkerStat(worker=args.worker) @@ -87,7 +94,7 @@ except exc.SQLAlchemyError as e: session = Session(engine) -lockfile = Path(f'worker_{args.worker}.lock') +lockfile = Path(f'/tmp/moa_worker_{args.worker}.lock') def check_worker_stop(): @@ -131,11 +138,13 @@ with lockfile.open('wt') as f: if not c.SEND: l.warning("SENDING IS NOT ENABLED") -bridges = session.query(Bridge).filter_by(enabled=True).filter(BridgeMetadata.worker_id == args.worker).filter(BridgeMetadata.id == Bridge.metadata_id) +bridges = session.query(Bridge).filter_by(enabled=True).filter( + BridgeMetadata.worker_id == args.worker).filter( + BridgeMetadata.id == Bridge.metadata_id) l.info(f"Working on {bridges.count()} bridges") -if 'sqlite' not in c.SQLALCHEMY_DATABASE_URI and not c.DEVELOPMENT: +if 'sqlite' not in c.SQLALCHEMY_DATABASE_URI and 'postgresql' not in c.SQLALCHEMY_DATABASE_URI and not c.DEVELOPMENT: bridges = bridges.order_by(func.rand()) bridge_count = 0 @@ -173,32 +182,38 @@ for bridge in bridges: mastodon_last_id = bridge.mastodon_last_id mastodonhost = bridge.mastodon_host - if mastodonhost.defer_until and mastodonhost.defer_until > datetime.now(): + if mastodonhost.defer_until and mastodonhost.defer_until > datetime.now( + ): l.warning(f"Deferring connections to {mastodonhost.hostname}") continue - mast_api = Mastodon( - client_id=mastodonhost.client_id, - client_secret=mastodonhost.client_secret, - api_base_url=f"https://{mastodonhost.hostname}", - access_token=bridge.mastodon_access_code, - debug_requests=False, - request_timeout=15, - ratelimit_method='throw' - ) + mast_api = Mastodon(client_id=mastodonhost.client_id, + client_secret=mastodonhost.client_secret, + api_base_url=f"https://{mastodonhost.hostname}", + access_token=bridge.mastodon_access_code, + debug_requests=False, + request_timeout=15, + ratelimit_method='throw') try: new_toots = mast_api.account_statuses( - bridge.mastodon_account_id, - since_id=bridge.mastodon_last_id - ) + bridge.mastodon_account_id, since_id=bridge.mastodon_last_id) except (MastodonAPIError, MastodonNetworkError) as e: msg = f"{bridge.mastodon_user}@{mastodonhost.hostname} MastodonAPIError: {e}" l.error(msg) - if any(x in repr(e) for x in ['revoked', 'invalid', 'not found', 'Forbidden', 'Unauthorized', 'Bad Request', - 'Name or service not known',]): - l.warning(f"Disabling bridge for user {bridge.mastodon_user}@{mastodonhost.hostname}") + if any(x in repr(e) for x in [ + 'revoked', + 'invalid', + 'not found', + 'Forbidden', + 'Unauthorized', + 'Bad Request', + 'Name or service not known', + ]): + l.warning( + f"Disabling bridge for user {bridge.mastodon_user}@{mastodonhost.hostname}" + ) bridge.mastodon_access_code = None bridge.enabled = False else: @@ -273,9 +288,11 @@ for bridge in bridges: except MastodonRatelimitError as e: l.error(f"{bridge.mastodon_user}@{mastodonhost.hostname}: {e}") - if len(new_toots) > c.MAX_MESSAGES_PER_RUN: - l.error(f"{bridge.mastodon_user}@{mastodonhost.hostname}: Limiting to {c.MAX_MESSAGES_PER_RUN} messages") - new_toots = new_toots[-c.MAX_MESSAGES_PER_RUN:] + #if len(new_toots) > c.MAX_MESSAGES_PER_RUN: + # l.error( + # f"{bridge.mastodon_user}@{mastodonhost.hostname}: Limiting to {c.MAX_MESSAGES_PER_RUN} messages" + # ) + # new_toots = new_toots[-c.MAX_MESSAGES_PER_RUN:] if c.SEND and len(new_toots) != 0: try: @@ -296,18 +313,18 @@ for bridge in bridges: twitter_last_id = bridge.twitter_last_id twitter_api = twitter.Api( - consumer_key=c.TWITTER_CONSUMER_KEY, - consumer_secret=c.TWITTER_CONSUMER_SECRET, - access_token_key=bridge.twitter_oauth_token, - access_token_secret=bridge.twitter_oauth_secret, - tweet_mode='extended' # Allow tweets longer than 140 raw characters + consumer_key=c.TWITTER_CONSUMER_KEY, + consumer_secret=c.TWITTER_CONSUMER_SECRET, + access_token_key=bridge.twitter_oauth_token, + access_token_secret=bridge.twitter_oauth_secret, + tweet_mode='extended' # Allow tweets longer than 140 raw characters ) try: new_tweets = twitter_api.GetUserTimeline( - since_id=bridge.twitter_last_id, - include_rts=True, - exclude_replies=False) + since_id=bridge.twitter_last_id, + include_rts=True, + exclude_replies=False) except TwitterError as e: l.error(f"@{bridge.twitter_handle}: {e}") @@ -320,7 +337,9 @@ for bridge in bridges: elif isinstance(e.message, list) and len(e.message) > 0: if e.message[0]['code'] in [89, 326]: - l.warning(f"Disabling bridge for Twitter user {bridge.twitter_handle}") + l.warning( + f"Disabling bridge for Twitter user {bridge.twitter_handle}" + ) bridge.twitter_oauth_token = None bridge.twitter_oauth_secret = None bridge.enabled = False @@ -331,7 +350,9 @@ for bridge in bridges: continue if len(new_tweets) > c.MAX_MESSAGES_PER_RUN: - l.error(f"@{bridge.twitter_handle}: Limiting to {c.MAX_MESSAGES_PER_RUN} messages") + l.error( + f"@{bridge.twitter_handle}: Limiting to {c.MAX_MESSAGES_PER_RUN} messages" + ) new_tweets = new_tweets[-c.MAX_MESSAGES_PER_RUN:] if c.SEND and len(new_tweets) != 0: @@ -399,7 +420,9 @@ for bridge in bridges: tweet_poster = TweetPoster(c.SEND, session, twitter_api, bridge) if bridge.mastodon_access_code: - l.info(f"{bridge.id}: M - {bridge.mastodon_user}@{mastodonhost.hostname}") + l.info( + f"{bridge.id}: M - {bridge.mastodon_user}@{mastodonhost.hostname}" + ) tweet_poster = TweetPoster(c.SEND, session, twitter_api, bridge) @@ -458,7 +481,6 @@ for bridge in bridges: bridge.md.last_tweet = tweet.created_at session.commit() - # # Post Instagram # @@ -489,7 +511,8 @@ for bridge in bridges: stat_recorded = True if not insta.should_skip_twitter and bridge.twitter_oauth_token: - tweet_poster = TweetPoster(c.SEND, session, twitter_api, bridge) + tweet_poster = TweetPoster(c.SEND, session, twitter_api, + bridge) try: result = tweet_poster.post(insta) @@ -524,7 +547,9 @@ if len(c.HEALTHCHECKS) >= args.worker: except Exception: pass -l.info(f"-- All done -> Total time: {worker_stat.formatted_time} / {worker_stat.items} items / {bridge_count} Bridges") +l.info( + f"-- All done -> Total time: {worker_stat.formatted_time} / {worker_stat.items} items / {bridge_count} Bridges" +) session.add(worker_stat)