fediseer/overseer/apis/v1/base.py

181 lines
10 KiB
Python
Raw Normal View History

2023-06-22 00:04:45 +00:00
import os
2023-06-20 17:47:56 +00:00
from flask import request
from flask_restx import Namespace, Resource, reqparse
2023-06-22 00:04:45 +00:00
from overseer.flask import cache, db
2023-06-20 17:47:56 +00:00
from overseer.observer import retrieve_suspicious_instances
from loguru import logger
2023-06-21 17:37:34 +00:00
from overseer.classes.instance import Instance
from overseer.database import functions as database
2023-06-22 00:04:45 +00:00
from overseer import exceptions as e
from overseer.utils import hash_api_key
from overseer.lemmy import pm_new_api_key
from pythorhead import Lemmy
2023-06-20 17:47:56 +00:00
api = Namespace('v1', 'API Version 1' )
from overseer.apis.models.v1 import Models
models = Models(api)
2023-06-22 00:04:45 +00:00
handle_bad_request = api.errorhandler(e.BadRequest)(e.handle_bad_requests)
handle_forbidden = api.errorhandler(e.Forbidden)(e.handle_bad_requests)
2023-06-22 00:32:08 +00:00
handle_unauthorized = api.errorhandler(e.Unauthorized)(e.handle_bad_requests)
handle_not_found = api.errorhandler(e.NotFound)(e.handle_bad_requests)
2023-06-22 00:04:45 +00:00
2023-06-20 17:47:56 +00:00
# Used to for the flask limiter, to limit requests per url paths
def get_request_path():
# logger.info(dir(request))
return f"{request.remote_addr}@{request.method}@{request.path}"
2023-06-21 17:37:34 +00:00
class Suspicions(Resource):
2023-06-20 17:47:56 +00:00
get_parser = reqparse.RequestParser()
get_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
2023-06-20 23:57:30 +00:00
get_parser.add_argument("activity_suspicion", required=False, default=20, type=int, help="How many users per local post+comment to consider suspicious", location="args")
2023-06-20 22:41:13 +00:00
get_parser.add_argument("csv", required=False, type=bool, help="Set to true to return just the domains as a csv. Mutually exclusive with domains", location="args")
get_parser.add_argument("domains", required=False, type=bool, help="Set to true to return just the domains as a list. Mutually exclusive with csv", location="args")
2023-06-20 17:47:56 +00:00
@api.expect(get_parser)
@logger.catch(reraise=True)
@cache.cached(timeout=10, query_string=True)
2023-06-21 17:37:34 +00:00
@api.marshal_with(models.response_model_model_Suspicions_get, code=200, description='Suspicious Instances', skip_none=True)
2023-06-20 17:47:56 +00:00
def get(self):
'''A List with the details of all suspicious instances
'''
self.args = self.get_parser.parse_args()
2023-06-20 23:57:30 +00:00
sus_instances = retrieve_suspicious_instances(self.args.activity_suspicion)
2023-06-20 22:33:37 +00:00
if self.args.csv:
return {"csv": ",".join([instance["domain"] for instance in sus_instances])},200
if self.args.domains:
return {"domains": [instance["domain"] for instance in sus_instances]},200
return {"instances": sus_instances},200
2023-06-21 17:37:34 +00:00
2023-06-21 17:44:33 +00:00
class Whitelist(Resource):
2023-06-21 17:37:34 +00:00
get_parser = reqparse.RequestParser()
get_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
2023-06-22 09:30:51 +00:00
get_parser.add_argument("endorsements", required=False, default=0, type=int, help="Limit to this amount of endorsements of more", location="args")
get_parser.add_argument("guarantors", required=False, default=1, type=int, help="Limit to this amount of guarantors of more", location="args")
2023-06-21 17:37:34 +00:00
get_parser.add_argument("csv", required=False, type=bool, help="Set to true to return just the domains as a csv. Mutually exclusive with domains", location="args")
get_parser.add_argument("domains", required=False, type=bool, help="Set to true to return just the domains as a list. Mutually exclusive with csv", location="args")
@api.expect(get_parser)
@cache.cached(timeout=10, query_string=True)
2023-06-22 00:04:45 +00:00
@api.marshal_with(models.response_model_model_Whitelist_get, code=200, description='Instances', skip_none=True)
2023-06-21 17:37:34 +00:00
def get(self):
'''A List with the details of all instances and their endorsements
'''
self.args = self.get_parser.parse_args()
instance_details = []
2023-06-22 09:30:51 +00:00
for instance in database.get_all_instances(self.args.endorsements,self.args.guarantors):
2023-06-21 17:37:34 +00:00
instance_details.append(instance.get_details())
if self.args.csv:
return {"csv": ",".join([instance["domain"] for instance in instance_details])},200
if self.args.domains:
return {"domains": [instance["domain"] for instance in instance_details]},200
return {"instances": instance_details},200
2023-06-22 00:04:45 +00:00
put_parser = reqparse.RequestParser()
put_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
put_parser.add_argument("domain", required=False, type=str, help="The instance domain. It MUST be alredy registered in https://overctrl.dbzer0.com", location="json")
put_parser.add_argument("guarantor", required=False, type=str, help="(Optiona) The domain of the guaranteeing instance. They will receive a PM to validate you", location="json")
@api.expect(put_parser)
@api.marshal_with(models.response_model_instances, code=200, description='Instances')
2023-06-22 09:30:51 +00:00
@api.response(400, 'Bad Request', models.response_model_error)
2023-06-22 00:04:45 +00:00
def put(self):
2023-06-22 00:32:08 +00:00
'''Register a new instance to the overseer
An instance account has to exist in the overseer lemmy instance
That account will recieve the new API key via PM
2023-06-22 00:04:45 +00:00
'''
self.args = self.put_parser.parse_args()
existing_instance = Instance.query.filter_by(domain=self.args.domain).first()
if existing_instance:
return existing_instance.get_details(),200
requested_lemmy = Lemmy(f"https://{self.args.domain}")
site = requested_lemmy.site.get()
api_key = pm_new_api_key(self.args.domain)
if not api_key:
raise e.BadRequest("Failed to generate API Key")
new_instance = Instance(
domain=self.args.domain,
api_key=hash_api_key(api_key),
open_registrations=site["site_view"]["local_site"]["registration_mode"] == "open",
email_verify=site["site_view"]["local_site"]["require_email_verification"],
software=requested_lemmy.nodeinfo['software']['name'],
)
new_instance.create()
return new_instance.get_details(),200
patch_parser = reqparse.RequestParser()
2023-06-22 00:32:08 +00:00
patch_parser.add_argument("apikey", type=str, required=True, help="The sending instance's API key.", location='headers')
2023-06-22 00:04:45 +00:00
patch_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
patch_parser.add_argument("domain", required=False, type=str, help="The instance domain. It MUST be alredy registered in https://overctrl.dbzer0.com", location="json")
2023-06-22 00:32:08 +00:00
patch_parser.add_argument("regenerate_key", required=False, type=bool, help="If True, will PM a new api key to this instance", location="json")
2023-06-22 00:04:45 +00:00
2023-06-22 00:32:08 +00:00
@api.expect(patch_parser)
2023-06-22 00:04:45 +00:00
@api.marshal_with(models.response_model_instances, code=200, description='Instances', skip_none=True)
2023-06-22 09:30:51 +00:00
@api.response(401, 'Invalid API Key', models.response_model_error)
@api.response(403, 'Instance Not Registered', models.response_model_error)
2023-06-22 00:04:45 +00:00
def patch(self):
2023-06-22 00:32:08 +00:00
'''Regenerate API key for instance
2023-06-22 00:04:45 +00:00
'''
self.args = self.patch_parser.parse_args()
2023-06-22 00:32:08 +00:00
if not self.args.apikey:
raise e.Unauthorized("You must provide the API key that was PM'd to your overctrl.dbzer0.com account")
2023-06-22 09:30:51 +00:00
instance = database.find_instance_by_api_key(self.args.apikey)
2023-06-22 00:32:08 +00:00
if not instance:
2023-06-22 09:30:51 +00:00
raise e.Forbidden(f"No Instance found matching provided API key and domain. Have you remembered to register it?")
2023-06-22 00:32:08 +00:00
if self.args.regenerate_key:
new_key = pm_new_api_key(self.args.domain)
instance.api_key = hash_api_key(new_key)
db.session.commit()
return instance.get_details(),200
2023-06-22 00:37:25 +00:00
delete_parser = reqparse.RequestParser()
delete_parser.add_argument("apikey", type=str, required=True, help="The sending instance's API key.", location='headers')
delete_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
delete_parser.add_argument("domain", required=False, type=str, help="The instance domain. It MUST be alredy registered in https://overctrl.dbzer0.com", location="json")
@api.expect(delete_parser)
@api.marshal_with(models.response_model_simple_response, code=200, description='Instances', skip_none=True)
2023-06-22 09:30:51 +00:00
@api.response(400, 'Bad Request', models.response_model_error)
@api.response(401, 'Invalid API Key', models.response_model_error)
@api.response(403, 'Forbidden', models.response_model_error)
2023-06-22 00:37:25 +00:00
def delete(self):
'''Delete instance from overseer
'''
self.args = self.patch_parser.parse_args()
if not self.args.apikey:
raise e.Unauthorized("You must provide the API key that was PM'd to your overctrl.dbzer0.com account")
instance = database.find_authenticated_instance(self.args.domain, self.args.apikey)
if not instance:
raise e.BadRequest(f"No Instance found matching provided API key and domain. Have you remembered to register it?")
if self.args.domain == os.getenv('OVERSEER_LEMMY_DOMAIN'):
raise e.Forbidden("Cannot delete overseer control instance")
db.session.delete(instance)
db.session.commit()
logger.warning(f"{self.args.domain} deleted")
2023-06-22 09:30:51 +00:00
return {"message":'Changed'}, 200
class WhitelistDomain(Resource):
get_parser = reqparse.RequestParser()
get_parser.add_argument("Client-Agent", default="unknown:0:unknown", type=str, required=False, help="The client name and version.", location="headers")
@api.expect(get_parser)
@cache.cached(timeout=10, query_string=True)
@api.marshal_with(models.response_model_instances, code=200, description='Instances')
def get(self, domain):
'''Display info about a specific instance
'''
self.args = self.get_parser.parse_args()
instance = database.find_instance_by_domain(domain)
if not instance:
raise e.NotFound(f"No Instance found matching provided domain. Have you remembered to register it?")
return instance.get_details(),200