This commit is contained in:
		
							
								
								
									
										79
									
								
								birb_scheduler_teller/app.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								birb_scheduler_teller/app.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,79 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
import sentry_sdk
 | 
			
		||||
from sentry_sdk.integrations.flask import FlaskIntegration
 | 
			
		||||
from flask import Flask
 | 
			
		||||
from werkzeug.middleware.proxy_fix import ProxyFix
 | 
			
		||||
 | 
			
		||||
# import stuff
 | 
			
		||||
from utils import register_all_error_handlers, register_health_checks, redis_client
 | 
			
		||||
 | 
			
		||||
# import views
 | 
			
		||||
from views import AssignmentView
 | 
			
		||||
 | 
			
		||||
from config import Config
 | 
			
		||||
 | 
			
		||||
# Tracing stuffs
 | 
			
		||||
import jaeger_client
 | 
			
		||||
import opentracing
 | 
			
		||||
from flask_opentracing import FlaskTracing
 | 
			
		||||
 | 
			
		||||
# Setup sentry
 | 
			
		||||
if Config.SENTRY_DSN:
 | 
			
		||||
    sentry_sdk.init(
 | 
			
		||||
        dsn=Config.SENTRY_DSN,
 | 
			
		||||
        integrations=[FlaskIntegration()],
 | 
			
		||||
        traces_sample_rate=0,
 | 
			
		||||
        send_default_pii=True,
 | 
			
		||||
        release=Config.RELEASE_ID,
 | 
			
		||||
        environment=Config.RELEASEMODE
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
# create flask app
 | 
			
		||||
app = Flask(__name__)
 | 
			
		||||
app.config.from_object(Config)
 | 
			
		||||
app.wsgi_app = ProxyFix(app.wsgi_app, x_proto=1)
 | 
			
		||||
 | 
			
		||||
# init stuff
 | 
			
		||||
redis_client.init_app(app)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# register error handlers
 | 
			
		||||
register_all_error_handlers(app)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# Setup tracing
 | 
			
		||||
def initialize_tracer():
 | 
			
		||||
    app.logger.info("Initializing jaeger...")
 | 
			
		||||
    jaeger_cfg = jaeger_client.Config(config={}, service_name='birb-scheduler-teller', validate=True)
 | 
			
		||||
    tracer = jaeger_cfg.initialize_tracer()
 | 
			
		||||
    return tracer
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
tracing = FlaskTracing(initialize_tracer, True, app)
 | 
			
		||||
 | 
			
		||||
# register views
 | 
			
		||||
for view in [AssignmentView]:
 | 
			
		||||
    view.register(app, trailing_slash=False)
 | 
			
		||||
 | 
			
		||||
register_health_checks(app)
 | 
			
		||||
 | 
			
		||||
# start debugging if needed
 | 
			
		||||
if __name__ == "__main__":
 | 
			
		||||
    app.run(debug=True)
 | 
			
		||||
else:
 | 
			
		||||
    import os
 | 
			
		||||
 | 
			
		||||
    if "gunicorn" in os.environ.get("SERVER_SOFTWARE", ""):
 | 
			
		||||
        import logging
 | 
			
		||||
 | 
			
		||||
        gunicorn_logger = logging.getLogger('gunicorn.error')
 | 
			
		||||
        app.logger.handlers = gunicorn_logger.handlers
 | 
			
		||||
        app.logger.setLevel(gunicorn_logger.level)
 | 
			
		||||
 | 
			
		||||
        jaeger_logger = logging.getLogger('jaeger_tracing')
 | 
			
		||||
        jaeger_logger.handlers = gunicorn_logger.handlers
 | 
			
		||||
        jaeger_logger.setLevel(gunicorn_logger.level)
 | 
			
		||||
 | 
			
		||||
        app.logger.info("Gunicorn environment detected!")
 | 
			
		||||
    else:
 | 
			
		||||
        app.logger.info("Not gunicorn")
 | 
			
		||||
							
								
								
									
										26
									
								
								birb_scheduler_teller/config.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								birb_scheduler_teller/config.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
			
		||||
import os
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def parse_default_while_not_defined():
 | 
			
		||||
    envvar = os.environ.get("DEFAULT_WHILE_NOT_DEFINED")
 | 
			
		||||
 | 
			
		||||
    if not envvar:
 | 
			
		||||
        return None, None
 | 
			
		||||
 | 
			
		||||
    envvar = envvar.split(';', 1)[2:]
 | 
			
		||||
 | 
			
		||||
    return envvar
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Config:
 | 
			
		||||
    SECRET_KEY = os.environ.get('SECRET_KEY', os.urandom(12))
 | 
			
		||||
 | 
			
		||||
    SENTRY_DSN = os.environ.get("SENTRY_DSN")
 | 
			
		||||
    RELEASE_ID = os.environ.get("RELEASE_ID", "test")
 | 
			
		||||
    RELEASEMODE = os.environ.get("RELEASEMODE", "dev")
 | 
			
		||||
 | 
			
		||||
    REDIS_URL = os.environ.get("REDIS_URL", "redis://localhost:6379/0")
 | 
			
		||||
 | 
			
		||||
    DEFAULT_WHILE_NOT_DEFINED = parse_default_while_not_defined()
 | 
			
		||||
 | 
			
		||||
    DEVICE_TIMEOUT = int(os.environ.get("DEVICE_TIMEOUT", 120))
 | 
			
		||||
							
								
								
									
										1
									
								
								birb_scheduler_teller/schemas/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								birb_scheduler_teller/schemas/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
							
								
								
									
										1
									
								
								birb_scheduler_teller/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								birb_scheduler_teller/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
import os
 | 
			
		||||
							
								
								
									
										5
									
								
								birb_scheduler_teller/utils/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								birb_scheduler_teller/utils/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from .require_decorators import json_required
 | 
			
		||||
from .error_handlers import register_all_error_handlers
 | 
			
		||||
from .healthchecks import register_health_checks
 | 
			
		||||
from .redis_client import redis_client
 | 
			
		||||
							
								
								
									
										20
									
								
								birb_scheduler_teller/utils/error_handlers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								birb_scheduler_teller/utils/error_handlers.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,20 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from flask import jsonify
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def get_standard_error_handler(code: int):
 | 
			
		||||
    def error_handler(err):
 | 
			
		||||
        return jsonify({"msg": err.description, "status": code}), code
 | 
			
		||||
 | 
			
		||||
    return error_handler
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def register_all_error_handlers(app):
 | 
			
		||||
    """
 | 
			
		||||
    function to register all handlers
 | 
			
		||||
    """
 | 
			
		||||
 | 
			
		||||
    error_codes_to_override = [404, 403, 401, 405, 400, 409, 422]
 | 
			
		||||
 | 
			
		||||
    for code in error_codes_to_override:
 | 
			
		||||
        app.register_error_handler(code, get_standard_error_handler(code))
 | 
			
		||||
							
								
								
									
										13
									
								
								birb_scheduler_teller/utils/healthchecks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								birb_scheduler_teller/utils/healthchecks.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from healthcheck import HealthCheck
 | 
			
		||||
from flask import Flask
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def health_database_status():
 | 
			
		||||
    return True, f'db is ok'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def register_health_checks(app: Flask):
 | 
			
		||||
    health = HealthCheck()
 | 
			
		||||
    health.add_check(health_database_status)
 | 
			
		||||
    app.add_url_rule("/healthz", "healthcheck", view_func=lambda: health.run())
 | 
			
		||||
							
								
								
									
										3
									
								
								birb_scheduler_teller/utils/redis_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								birb_scheduler_teller/utils/redis_client.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,3 @@
 | 
			
		||||
from flask_redis import FlaskRedis
 | 
			
		||||
 | 
			
		||||
redis_client = FlaskRedis()
 | 
			
		||||
							
								
								
									
										16
									
								
								birb_scheduler_teller/utils/require_decorators.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								birb_scheduler_teller/utils/require_decorators.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,16 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from flask import request, abort
 | 
			
		||||
 | 
			
		||||
from functools import wraps
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def json_required(f):
 | 
			
		||||
    @wraps(f)
 | 
			
		||||
    def call(*args, **kwargs):
 | 
			
		||||
 | 
			
		||||
        if request.is_json:
 | 
			
		||||
            return f(*args, **kwargs)
 | 
			
		||||
        else:
 | 
			
		||||
            abort(400, "JSON required")
 | 
			
		||||
 | 
			
		||||
    return call
 | 
			
		||||
							
								
								
									
										2
									
								
								birb_scheduler_teller/views/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								birb_scheduler_teller/views/__init__.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from .assignment_view import AssignmentView
 | 
			
		||||
							
								
								
									
										71
									
								
								birb_scheduler_teller/views/assignment_view.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								birb_scheduler_teller/views/assignment_view.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,71 @@
 | 
			
		||||
#!/usr/bin/env python3
 | 
			
		||||
from typing import Optional
 | 
			
		||||
from flask import jsonify, request, abort, current_app
 | 
			
		||||
from utils import json_required, redis_client
 | 
			
		||||
from flask_classful import FlaskView
 | 
			
		||||
 | 
			
		||||
import json
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class AssignmentView(FlaskView):
 | 
			
		||||
 | 
			
		||||
    @staticmethod
 | 
			
		||||
    def _resolve_response(site: Optional[str]) -> dict:
 | 
			
		||||
        site_url_map = redis_client.get("SITEURLMAP")
 | 
			
		||||
 | 
			
		||||
        if site_url_map:
 | 
			
		||||
            site_url_map = json.loads(site_url_map.decode('utf-8'))
 | 
			
		||||
 | 
			
		||||
        if not site_url_map:
 | 
			
		||||
            return {
 | 
			
		||||
                "site": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][0],
 | 
			
		||||
                "url": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][1],
 | 
			
		||||
                "hard_default": True
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        if not site:
 | 
			
		||||
            return {
 | 
			
		||||
                "site": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][0],
 | 
			
		||||
                "url": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][1],
 | 
			
		||||
                "hard_default": True
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        if site not in site_url_map.keys():
 | 
			
		||||
            # This should be an internal server error instead
 | 
			
		||||
            return {
 | 
			
		||||
                "site": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][0],
 | 
			
		||||
                "url": current_app.config['DEFAULT_WHILE_NOT_DEFINED'][1],
 | 
			
		||||
                "hard_default": True
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            "site": site,
 | 
			
		||||
            "url": site_url_map[site],
 | 
			
		||||
            "hard_default": False
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    @json_required
 | 
			
		||||
    def post(self):
 | 
			
		||||
        device_id = request.json['device_id']
 | 
			
		||||
 | 
			
		||||
        schedule_store_key = f"SCHEDULED:{device_id}"
 | 
			
		||||
 | 
			
		||||
        target_site = redis_client.get(schedule_store_key)
 | 
			
		||||
 | 
			
		||||
        if target_site:
 | 
			
		||||
            target_site = target_site.decode('utf-8')
 | 
			
		||||
 | 
			
		||||
        if not target_site:
 | 
			
		||||
            default_target = redis_client.get('DEFAULT:SCHEDULED')
 | 
			
		||||
 | 
			
		||||
            if default_target:
 | 
			
		||||
                target_site = default_target.decode('utf-8')
 | 
			
		||||
                redis_client.set(schedule_store_key, default_target)
 | 
			
		||||
 | 
			
		||||
        # Update (or set) TTL
 | 
			
		||||
        redis_client.expire(
 | 
			
		||||
            schedule_store_key,
 | 
			
		||||
            current_app.config['DEVICE_TIMEOUT']
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
        return jsonify(self._resolve_response(target_site))
 | 
			
		||||
		Reference in New Issue
	
	Block a user