fossasia/open-event-orga-server

View on GitHub
app/api/speakers_calls.py

Summary

Maintainability
A
3 hrs
Test Coverage
from flask_rest_jsonapi import ResourceDetail, ResourceList, ResourceRelationship
from flask_rest_jsonapi.exceptions import ObjectNotFound
from sqlalchemy.orm.exc import NoResultFound

from app.api.bootstrap import api
from app.api.helpers.errors import ConflictError, ForbiddenError
from app.api.helpers.permission_manager import has_access
from app.api.helpers.utilities import require_relationship
from app.api.schema.speakers_calls import SpeakersCallSchema
from app.models import db
from app.models.event import Event
from app.models.speakers_call import SpeakersCall


class SpeakersCallList(ResourceList):
    """
    create Speakers Call
    """

    def before_post(self, args, kwargs, data):
        """
        before post method to check for required relationship and proper permission
        :param args:
        :param kwargs:
        :param data:
        :return:
        """
        require_relationship(['event'], data)
        if not has_access('is_coorganizer', event_id=data['event']):
            raise ForbiddenError({'source': ''}, 'Co-organizer access is required.')

    def before_create_object(self, data, view_kwargs):
        """
        method to check if speaker calls object already exists for an event
        :param data:
        :param view_kwargs:
        :return:
        """
        try:
            speakers_call = (
                self.session.query(SpeakersCall)
                .filter_by(event_id=data['event'], deleted_at=None)
                .one()
            )
            event = speakers_call.event
            if (
                speakers_call.starts_at > event.starts_at
                or speakers_call.ends_at > event.starts_at
            ):
                raise ForbiddenError(
                    {'source': ''},
                    "Speakers call date can\'t be after the event start date",
                )
        except NoResultFound:
            pass
        else:
            raise ConflictError(
                {'pointer': '/data/relationships/event'},
                "Speakers Call already exists for this event",
            )

    schema = SpeakersCallSchema
    methods = [
        'POST',
    ]
    data_layer = {
        'session': db.session,
        'model': SpeakersCall,
        'methods': {'before_create_object': before_create_object},
    }


class SpeakersCallDetail(ResourceDetail):
    """
    speakers call detail by id
    """

    def before_patch(self, args, kwargs, data):
        """
        before patch method to check for existing speakers-call
        :param args:
        :param kwargs:
        :param data:
        :return:
        """
        if kwargs.get('event_id'):
            try:
                speakers_call = SpeakersCall.query.filter_by(
                    event_id=kwargs['event_id']
                ).one()
                event = speakers_call.event
                if (
                    speakers_call.starts_at > event.starts_at
                    or speakers_call.ends_at > event.starts_at
                ):
                    raise ForbiddenError(
                        {'source': ''},
                        "Speakers call date can\'t be after the event start date",
                    )
            except NoResultFound:
                raise ObjectNotFound({'source': ''}, "Object: not found")
            kwargs['id'] = speakers_call.id

    def before_get_object(self, view_kwargs):
        """
        before get method to get the resource id for fetching details
        :param view_kwargs:
        :return:
        """
        if view_kwargs.get('event_identifier'):
            try:
                event = (
                    self.session.query(Event)
                    .filter_by(identifier=view_kwargs['event_identifier'])
                    .one()
                )
            except NoResultFound:
                raise ObjectNotFound(
                    {'parameter': 'event_identifier'},
                    "Event: {} not found".format(view_kwargs['event_identifier']),
                )
            else:
                view_kwargs['event_id'] = event.id

        if view_kwargs.get('event_id'):
            try:
                speakers_call = (
                    self.session.query(SpeakersCall)
                    .filter_by(event_id=view_kwargs['event_id'])
                    .one()
                )
            except NoResultFound:
                raise ObjectNotFound(
                    {'parameter': 'event_identifier'}, "Object: not found"
                )
            view_kwargs['id'] = speakers_call.id

    decorators = (
        api.has_permission(
            'is_coorganizer',
            fetch="event_id",
            model=SpeakersCall,
            methods="PATCH,DELETE",
        ),
    )
    schema = SpeakersCallSchema
    data_layer = {
        'session': db.session,
        'model': SpeakersCall,
        'methods': {'before_get_object': before_get_object},
    }


class SpeakersCallRelationship(ResourceRelationship):
    """
    speakers call Relationship
    """

    decorators = (
        api.has_permission(
            'is_coorganizer',
            fetch="event_id",
            model=SpeakersCall,
            methods="PATCH,DELETE",
        ),
    )
    schema = SpeakersCallSchema
    methods = ['GET', 'PATCH']
    data_layer = {'session': db.session, 'model': SpeakersCall}