pythonhtmlpython-3.xdjangohttp-status-code-405

405 HTML: Method Not Allowed


I have created a calendar that is embedded in my website. There is an option for admins to create events and when I try to create an event, I get faced with a 405 Error. Could someone please point me in the right direction into why this isn't working, when in previous versions of Python Django, it was working?

My code is below for convenience:

urls.py


from django.urls import path

from main_website import views
from main_website.views import (AboutView, ArticleCreateView,
                                ArticleDeleteView, ArticleUpdateView,
                                CalendarView, ContactUsView, DonateView,
                                EditEventView, EventView, HelpUsView,
                                ImageAddView, ImageCategoryAddView,
                                ImageDetailView, ImageGalleryView, IndexView,
                                SearchView, TaggedView)

urlpatterns = [
    path('', IndexView.as_view(), name='main_website_home'),
    path('about', AboutView.as_view(), name='main_website_about'),
    path('search/', SearchView.as_view(), name='main_website_search'),
    path('article_create', ArticleCreateView.as_view(), name='main_website_article_create'),
    path('article/<slug:slug>', views.article_detail, name='main_website_article_detail'),
    path('article/update/<slug:slug>/', ArticleUpdateView.as_view(), name='main_website_article_update'),
    path('article/delete/<slug:slug>/', ArticleDeleteView.as_view(), name='main_website_article_delete'),
    path('tags/<slug:slug>/articles/', TaggedView.as_view(), name="main_website_article_tags"),
    path('calendar/', CalendarView.as_view(), name='main_website_calendar'),
    path('calendar/event/new/', EventView.as_view(), name='main_website_calendar_new_event'),
    path('calendar/event/edit/<event_id>/', EditEventView.as_view(), name='main_website_calendar_edit_event'),
    path('waiting_list/register', views.waiting_list_register, name='main_website_waiting_list_register'),
    path('gallery', ImageGalleryView.as_view(), name='main_website_gallery'),
    path('gallery/upload', ImageAddView.as_view(), name='main_website_gallery_upload'),
    path('gallery/category/add/', ImageCategoryAddView.as_view(), name='main_website_gallery_add_category'),
    path('gallery/image/<str:pk>/', ImageDetailView.as_view(), name='main_website_gallery_image_detail'),
    path('help_us', HelpUsView.as_view(), name='main_website_help_us'),
    path('ways_to_donate', DonateView.as_view(), name='main_website_ways_to_donate'),
    path('contact_us', ContactUsView.as_view(), name='main_website_contact_us')
]

utils.py

import calendar
from calendar import HTMLCalendar
from datetime import date, datetime, timedelta

from main_website.models import Event


class Calendar(HTMLCalendar):
    def __init__(self, year=None, month=None):
        self.year = year
        self.month = month
        super(Calendar, self).__init__()

    # formats a day as a td
    # filter events by day
    def formatday(self, day, events):
        events_per_day = events.filter(start_time__day=day)
        d = ''
        for event in events_per_day:
            d += f'<li> {event.get_html_url} </li>'

        if day != 0:
            return f"<td><span class='date'>{day}</span><ul> {d} </ul></td>"
        return '<td></td>'

    # formats a week as a tr
    def formatweek(self, theweek, events):
        week = ''
        for d, weekday in theweek:
            week += self.formatday(d, events)
        return f'<tr> {week} </tr>'

    # formats a month as a table
    # filter events by year and month
    def formatmonth(self, withyear=True):
        events = Event.objects.filter(start_time__year=self.year, start_time__month=self.month)

        cal = f'<table border="0" cellpadding="0" cellspacing="0" class="calendar">\n'
        cal += f'{self.formatmonthname(self.year, self.month, withyear=withyear)}\n'
        cal += f'{self.formatweekheader()}\n'
        for week in self.monthdays2calendar(self.year, self.month):
            cal += f'{self.formatweek(week, events)}\n'
        return cal

def get_date(req_month):
    if req_month:
        year, month = (int(x) for x in req_month.split('-'))
        return date(year, month, day=1)
    return datetime.today()

def prev_month(d):
    first = d.replace(day=1)
    prev_month = first - timedelta(days=1)
    month = 'month=' + str(prev_month.year) + '-' + str(prev_month.month)
    return month

def next_month(d):
    days_in_month = calendar.monthrange(d.year, d.month)[1]
    last = d.replace(day=days_in_month)
    next_month = last + timedelta(days=1)
    month = 'month=' + str(next_month.year) + '-' + str(next_month.month)
    return month

def get_filename(filename):
    return filename.upper()

views.py (everything calendar related)

import calendar
from datetime import date, datetime, timedelta

from django.conf import settings
from django.contrib import messages
from django.contrib.auth.decorators import login_required
from django.contrib.auth.mixins import LoginRequiredMixin, UserPassesTestMixin
from django.db.models import Q
from django.shortcuts import get_object_or_404, redirect, render
from django.utils.safestring import mark_safe
from django.views import generic
from django.views.generic import CreateView, DeleteView, ListView, UpdateView
from taggit.models import Tag

from main_website.forms import (AddImageCategoryForm, ArticleForm, EventForm,
                                UploadImageForm, WaitingListForm)
from main_website.models import (Article, Event, ImageGallery,
                                 ImageGalleryCategory)
from main_website.utils import Calendar, get_date, next_month, prev_month

class CalendarView(generic.ListView):
    model = Event
    template_name = 'calendar/calendar.html'

    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        d = get_date(self.request.GET.get('month', None))
        cal = Calendar(d.year, d.month)
        html_cal = cal.formatmonth(withyear=True)
        context['calendar'] = mark_safe(html_cal)
        context['prev_month'] = prev_month(d)
        context['next_month'] = next_month(d)
        context['title'] = 'Calendar'
        return context

class EventView(LoginRequiredMixin, generic.View):
    def get(self, request, event_id=None):
        instance = Event()
        if event_id:
            instance = get_object_or_404(Event, pk=event_id)
        else:
            instance = Event()

        form = EventForm()

        context = { 
            'form': form,
            'instance': instance,
            'title': 'New Event',
        }
        return render(request, 'calendar/event.html', context)

class EditEventView(LoginRequiredMixin, generic.View):
    def get(self, request, event_id=None):
        instance = Event()
        if event_id:
            instance = get_object_or_404(Event, pk=event_id)
        else:
            instance = Event()

        form = EventForm()

        context = { 
            'form': form,
            'instance': instance,
            'title': 'Edit Event',
        }
        return render(request, 'calendar/edit_event.html', context)

    def post(self, request):
        instance = Event()
        form = EventForm(request.POST or None, instance=instance)
        if form.is_valid():
            form.save()
            return redirect('main_website_calendar')

forms.py (calendar related only)

from django import forms
from django.forms import DateInput
from taggit.forms import TagField
from taggit_labels.widgets import LabelWidget

from accounts.models import User
from main_website.models import (Article, Event, ImageGallery,
                                 ImageGalleryCategory, WaitingList)

class EventForm(forms.ModelForm):
    class Meta:
        model = Event
        # datetime-local is a HTML5 input type, format to make date time show on fields
        widgets = {
            'start_time': DateInput(attrs={'type': 'datetime-local'}, format='%Y-%m-%dT%H:%M'),
            'end_time': DateInput(attrs={'type': 'datetime-local'}, format='%Y-%m-%dT%H:%M'),
        }
        fields = '__all__'

    def __init__(self, *args, **kwargs):
        super(EventForm, self).__init__(*args, **kwargs)
        # input_formats parses HTML5 datetime-local input to datetime field
        self.fields['start_time'].input_formats = ('%Y-%m-%dT%H:%M',)
        self.fields['end_time'].input_formats = ('%Y-%m-%dT%H:%M',)

event.html

{% extends 'calendar/base.html' %}
{% load crispy_forms_tags %}

{% block content %}
<div class="calendar-section">
    <form method="POST">
        {% csrf_token %}
        <legend class="border-bottom mb-4">Create New Event</legend>
        {{ form | crispy }}
        <button type="submit" class="btn btn-custom-purple">
            <i class="fa-duotone fa-calendar-plus"></i> 
            Submit
        </button>
    </form>
    <hr>
    <button class="btn btn-custom-purple">
        <i class="fa-duotone fa-angles-left"></i>
        <a href="{% url 'main_website_calendar' %}"> Back to Calendar</a>
    </button>
</div>
{% endblock %}

Solution

  • Your calendar/event/new/ route points to EventView, which only has a get method defined. Posting to a view without a post method will produce a 405 error.

    Also, your EditEventView doesn't edit an event, but instead always creates a new event.

    To fix this, you need to do this:

    EventEditView post method:

    def post(self, request, event_id):
        instance = get_object_or_404(Event, pk=event_id)
        form = EventForm(request.POST or None, instance=instance)
        if form.is_valid():
            form.save()
            return redirect('main_website_calendar')
    

    You should also implement some error handling in these post requests, but I'll leave that to you.

    EDIT:

    EventView post method:

    def post(self, request):
            instance = Event()
            form = EventForm(request.POST or None, instance=instance)
            if form.is_valid():
                form.save()
                return redirect('main_website_calendar')