2018-10-29 16:52:09 +00:00
|
|
|
from crispy_forms.helper import FormHelper
|
2018-10-20 22:20:31 +00:00
|
|
|
from crispy_forms.layout import Layout, Field, HTML
|
|
|
|
from django import forms
|
2018-10-27 00:33:45 +00:00
|
|
|
from django.contrib.auth.decorators import login_required
|
|
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
2018-10-20 22:20:31 +00:00
|
|
|
from django.db.models import Q
|
2018-10-13 20:01:45 +00:00
|
|
|
from django.http import HttpRequest, HttpResponseBadRequest, JsonResponse
|
|
|
|
from django.shortcuts import render
|
2018-11-03 12:43:23 +00:00
|
|
|
from django.views.generic import CreateView, UpdateView, DeleteView, FormView
|
2018-10-17 21:38:40 +00:00
|
|
|
from django.views.generic.edit import FormMixin
|
2018-10-20 22:20:31 +00:00
|
|
|
|
2018-10-13 20:01:45 +00:00
|
|
|
from YtManagerApp.management.videos import get_videos
|
2018-10-27 00:33:45 +00:00
|
|
|
from YtManagerApp.models import Subscription, SubscriptionFolder, VIDEO_ORDER_CHOICES, VIDEO_ORDER_MAPPING
|
2018-11-03 12:43:23 +00:00
|
|
|
from YtManagerApp.utils import youtube, subscription_file_parser
|
2018-10-20 22:20:31 +00:00
|
|
|
from YtManagerApp.views.controls.modal import ModalMixin
|
|
|
|
|
2018-11-03 12:43:23 +00:00
|
|
|
import logging
|
|
|
|
|
2018-10-13 20:01:45 +00:00
|
|
|
|
|
|
|
class VideoFilterForm(forms.Form):
|
|
|
|
CHOICES_SHOW_WATCHED = (
|
|
|
|
('y', 'Watched'),
|
|
|
|
('n', 'Not watched'),
|
|
|
|
('all', '(All)')
|
|
|
|
)
|
|
|
|
|
|
|
|
CHOICES_SHOW_DOWNLOADED = (
|
|
|
|
('y', 'Downloaded'),
|
|
|
|
('n', 'Not downloaded'),
|
|
|
|
('all', '(All)')
|
|
|
|
)
|
|
|
|
|
|
|
|
MAPPING_SHOW = {
|
|
|
|
'y': True,
|
|
|
|
'n': False,
|
|
|
|
'all': None
|
|
|
|
}
|
|
|
|
|
|
|
|
query = forms.CharField(label='', required=False)
|
2018-10-27 00:33:45 +00:00
|
|
|
sort = forms.ChoiceField(label='Sort:', choices=VIDEO_ORDER_CHOICES, initial='newest')
|
2018-10-13 20:01:45 +00:00
|
|
|
show_watched = forms.ChoiceField(label='Show only: ', choices=CHOICES_SHOW_WATCHED, initial='all')
|
|
|
|
show_downloaded = forms.ChoiceField(label='', choices=CHOICES_SHOW_DOWNLOADED, initial='all')
|
|
|
|
subscription_id = forms.IntegerField(
|
|
|
|
required=False,
|
|
|
|
widget=forms.HiddenInput()
|
|
|
|
)
|
|
|
|
folder_id = forms.IntegerField(
|
|
|
|
required=False,
|
|
|
|
widget=forms.HiddenInput()
|
|
|
|
)
|
|
|
|
|
|
|
|
def __init__(self, data=None):
|
|
|
|
super().__init__(data, auto_id='form_video_filter_%s')
|
|
|
|
self.helper = FormHelper()
|
|
|
|
self.helper.form_id = 'form_video_filter'
|
|
|
|
self.helper.form_class = 'form-inline'
|
|
|
|
self.helper.form_method = 'POST'
|
2018-10-17 21:38:40 +00:00
|
|
|
self.helper.form_action = 'ajax_get_videos'
|
2018-10-13 20:01:45 +00:00
|
|
|
self.helper.field_class = 'mr-1'
|
|
|
|
self.helper.label_class = 'ml-2 mr-1 no-asterisk'
|
|
|
|
|
|
|
|
self.helper.layout = Layout(
|
|
|
|
Field('query', placeholder='Search'),
|
|
|
|
'sort',
|
|
|
|
'show_watched',
|
|
|
|
'show_downloaded',
|
|
|
|
'subscription_id',
|
|
|
|
'folder_id'
|
|
|
|
)
|
|
|
|
|
|
|
|
def clean_sort(self):
|
|
|
|
data = self.cleaned_data['sort']
|
2018-10-27 00:33:45 +00:00
|
|
|
return VIDEO_ORDER_MAPPING[data]
|
2018-10-13 20:01:45 +00:00
|
|
|
|
|
|
|
def clean_show_downloaded(self):
|
|
|
|
data = self.cleaned_data['show_downloaded']
|
|
|
|
return VideoFilterForm.MAPPING_SHOW[data]
|
|
|
|
|
|
|
|
def clean_show_watched(self):
|
|
|
|
data = self.cleaned_data['show_watched']
|
|
|
|
return VideoFilterForm.MAPPING_SHOW[data]
|
|
|
|
|
|
|
|
|
|
|
|
def __tree_folder_id(fd_id):
|
|
|
|
if fd_id is None:
|
|
|
|
return '#'
|
|
|
|
return 'folder' + str(fd_id)
|
|
|
|
|
|
|
|
|
|
|
|
def __tree_sub_id(sub_id):
|
|
|
|
if sub_id is None:
|
|
|
|
return '#'
|
2018-10-17 21:38:40 +00:00
|
|
|
return 'sub' + str(sub_id)
|
2018-10-13 20:01:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def index(request: HttpRequest):
|
|
|
|
if request.user.is_authenticated:
|
|
|
|
context = {
|
|
|
|
'filter_form': VideoFilterForm()
|
|
|
|
}
|
|
|
|
return render(request, 'YtManagerApp/index.html', context)
|
|
|
|
else:
|
|
|
|
return render(request, 'YtManagerApp/index_unauthenticated.html')
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
@login_required
|
2018-10-13 20:01:45 +00:00
|
|
|
def ajax_get_tree(request: HttpRequest):
|
|
|
|
|
|
|
|
def visit(node):
|
|
|
|
if isinstance(node, SubscriptionFolder):
|
|
|
|
return {
|
|
|
|
"id": __tree_folder_id(node.id),
|
|
|
|
"text": node.name,
|
|
|
|
"type": "folder",
|
|
|
|
"state": {"opened": True},
|
|
|
|
"parent": __tree_folder_id(node.parent_id)
|
|
|
|
}
|
|
|
|
elif isinstance(node, Subscription):
|
|
|
|
return {
|
|
|
|
"id": __tree_sub_id(node.id),
|
|
|
|
"type": "sub",
|
|
|
|
"text": node.name,
|
|
|
|
"icon": node.icon_default,
|
|
|
|
"parent": __tree_folder_id(node.parent_folder_id)
|
|
|
|
}
|
|
|
|
|
2018-10-17 21:38:40 +00:00
|
|
|
result = SubscriptionFolder.traverse(None, request.user, visit)
|
2018-10-13 20:01:45 +00:00
|
|
|
return JsonResponse(result, safe=False)
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
@login_required
|
2018-10-13 20:01:45 +00:00
|
|
|
def ajax_get_videos(request: HttpRequest):
|
|
|
|
if request.method == 'POST':
|
|
|
|
form = VideoFilterForm(request.POST)
|
|
|
|
if form.is_valid():
|
|
|
|
videos = get_videos(
|
|
|
|
user=request.user,
|
|
|
|
sort_order=form.cleaned_data['sort'],
|
|
|
|
query=form.cleaned_data['query'],
|
|
|
|
subscription_id=form.cleaned_data['subscription_id'],
|
|
|
|
folder_id=form.cleaned_data['folder_id'],
|
|
|
|
only_watched=form.cleaned_data['show_watched'],
|
|
|
|
only_downloaded=form.cleaned_data['show_downloaded']
|
|
|
|
)
|
|
|
|
|
|
|
|
context = {
|
|
|
|
'videos': videos
|
|
|
|
}
|
|
|
|
|
|
|
|
return render(request, 'YtManagerApp/index_videos.html', context)
|
|
|
|
|
|
|
|
return HttpResponseBadRequest()
|
|
|
|
|
|
|
|
|
2018-10-14 21:45:08 +00:00
|
|
|
class SubscriptionFolderForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = SubscriptionFolder
|
|
|
|
fields = ['name', 'parent']
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.helper = FormHelper()
|
|
|
|
self.helper.form_tag = False
|
|
|
|
|
|
|
|
def clean_name(self):
|
|
|
|
name = self.cleaned_data['name']
|
|
|
|
return name.strip()
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
name = cleaned_data.get('name')
|
|
|
|
parent = cleaned_data.get('parent')
|
|
|
|
|
|
|
|
# Check name is unique in parent folder
|
|
|
|
args_id = []
|
|
|
|
if self.instance is not None:
|
|
|
|
args_id.append(~Q(id=self.instance.id))
|
|
|
|
|
|
|
|
if SubscriptionFolder.objects.filter(parent=parent, name__iexact=name, *args_id).count() > 0:
|
2018-10-29 16:52:09 +00:00
|
|
|
raise forms.ValidationError(
|
|
|
|
'A folder with the same name already exists in the given parent directory!', code='already_exists')
|
2018-10-14 21:45:08 +00:00
|
|
|
|
|
|
|
# Check for cycles
|
|
|
|
if self.instance is not None:
|
|
|
|
self.__test_cycles(parent)
|
|
|
|
|
|
|
|
def __test_cycles(self, new_parent):
|
|
|
|
visited = [self.instance.id]
|
|
|
|
current = new_parent
|
|
|
|
while current is not None:
|
|
|
|
if current.id in visited:
|
|
|
|
raise forms.ValidationError('Selected parent would create a parenting cycle!', code='parenting_cycle')
|
|
|
|
visited.append(current.id)
|
|
|
|
current = current.parent
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class CreateFolderModal(LoginRequiredMixin, ModalMixin, CreateView):
|
2018-10-14 21:45:08 +00:00
|
|
|
template_name = 'YtManagerApp/controls/folder_create_modal.html'
|
|
|
|
form_class = SubscriptionFolderForm
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
form.instance.user = self.request.user
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class UpdateFolderModal(LoginRequiredMixin, ModalMixin, UpdateView):
|
2018-10-14 21:45:08 +00:00
|
|
|
template_name = 'YtManagerApp/controls/folder_update_modal.html'
|
2018-10-13 20:01:45 +00:00
|
|
|
model = SubscriptionFolder
|
2018-10-14 21:45:08 +00:00
|
|
|
form_class = SubscriptionFolderForm
|
|
|
|
|
2018-10-13 20:01:45 +00:00
|
|
|
|
2018-10-17 21:38:40 +00:00
|
|
|
class DeleteFolderForm(forms.Form):
|
|
|
|
keep_subscriptions = forms.BooleanField(required=False, initial=False, label="Keep subscriptions")
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class DeleteFolderModal(LoginRequiredMixin, ModalMixin, FormMixin, DeleteView):
|
2018-10-14 21:45:08 +00:00
|
|
|
template_name = 'YtManagerApp/controls/folder_delete_modal.html'
|
|
|
|
model = SubscriptionFolder
|
2018-10-17 21:38:40 +00:00
|
|
|
form_class = DeleteFolderForm
|
|
|
|
|
|
|
|
def delete(self, request, *args, **kwargs):
|
|
|
|
self.object = self.get_object()
|
|
|
|
form = self.get_form()
|
|
|
|
if form.is_valid():
|
|
|
|
return self.form_valid(form)
|
|
|
|
else:
|
|
|
|
return self.form_invalid(form)
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
self.object.delete_folder(keep_subscriptions=form.cleaned_data['keep_subscriptions'])
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
|
|
|
|
class CreateSubscriptionForm(forms.ModelForm):
|
|
|
|
playlist_url = forms.URLField(label='Playlist/Channel URL')
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Subscription
|
2018-10-27 00:33:45 +00:00
|
|
|
fields = ['parent_folder', 'auto_download',
|
|
|
|
'download_limit', 'download_order', 'delete_after_watched']
|
2018-10-17 21:38:40 +00:00
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
2018-10-29 16:52:09 +00:00
|
|
|
self.yt_api = youtube.YoutubeAPI.build_public()
|
2018-10-17 21:38:40 +00:00
|
|
|
self.helper = FormHelper()
|
|
|
|
self.helper.form_tag = False
|
|
|
|
self.helper.layout = Layout(
|
|
|
|
'playlist_url',
|
2018-10-27 00:33:45 +00:00
|
|
|
'parent_folder',
|
|
|
|
HTML('<hr>'),
|
|
|
|
HTML('<h5>Download configuration overloads</h5>'),
|
|
|
|
'auto_download',
|
|
|
|
'download_limit',
|
|
|
|
'download_order',
|
|
|
|
'delete_after_watched'
|
2018-10-17 21:38:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def clean_playlist_url(self):
|
2018-10-29 16:52:09 +00:00
|
|
|
playlist_url: str = self.cleaned_data['playlist_url']
|
2018-10-17 21:38:40 +00:00
|
|
|
try:
|
2018-10-29 16:52:09 +00:00
|
|
|
parsed_url = self.yt_api.parse_url(playlist_url)
|
|
|
|
except youtube.InvalidURL as e:
|
|
|
|
raise forms.ValidationError(str(e))
|
|
|
|
|
|
|
|
is_playlist = 'playlist' in parsed_url
|
|
|
|
is_channel = parsed_url['type'] in ('channel', 'user', 'channel_custom')
|
|
|
|
|
|
|
|
if not is_channel and not is_playlist:
|
|
|
|
raise forms.ValidationError('The given URL must link to a channel or a playlist!')
|
|
|
|
|
2018-10-17 21:38:40 +00:00
|
|
|
return playlist_url
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class CreateSubscriptionModal(LoginRequiredMixin, ModalMixin, CreateView):
|
2018-10-17 21:38:40 +00:00
|
|
|
template_name = 'YtManagerApp/controls/subscription_create_modal.html'
|
|
|
|
form_class = CreateSubscriptionForm
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
form.instance.user = self.request.user
|
|
|
|
api = youtube.YoutubeAPI.build_public()
|
|
|
|
try:
|
|
|
|
form.instance.fetch_from_url(form.cleaned_data['playlist_url'], api)
|
2018-10-29 16:52:09 +00:00
|
|
|
except youtube.InvalidURL as e:
|
|
|
|
return self.modal_response(form, False, str(e))
|
|
|
|
except ValueError as e:
|
|
|
|
return self.modal_response(form, False, str(e))
|
|
|
|
# except youtube.YoutubeUserNotFoundException:
|
|
|
|
# return self.modal_response(
|
|
|
|
# form, False, 'Could not find an user based on the given URL. Please verify that the URL is correct.')
|
|
|
|
# except youtube.YoutubePlaylistNotFoundException:
|
|
|
|
# return self.modal_response(
|
|
|
|
# form, False, 'Could not find a playlist based on the given URL. Please verify that the URL is correct.')
|
|
|
|
# except youtube.YoutubeException as e:
|
|
|
|
# return self.modal_response(
|
|
|
|
# form, False, str(e))
|
|
|
|
# except youtube.APIError as e:
|
|
|
|
# return self.modal_response(
|
|
|
|
# form, False, 'An error occurred while communicating with the YouTube API: ' + str(e))
|
2018-10-17 21:38:40 +00:00
|
|
|
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
|
|
|
|
class UpdateSubscriptionForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
|
|
model = Subscription
|
2018-10-20 22:20:31 +00:00
|
|
|
fields = ['name', 'parent_folder', 'auto_download',
|
2018-10-27 00:33:45 +00:00
|
|
|
'download_limit', 'download_order', 'delete_after_watched']
|
2018-10-17 21:38:40 +00:00
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.helper = FormHelper()
|
|
|
|
self.helper.form_tag = False
|
|
|
|
self.helper.layout = Layout(
|
|
|
|
'name',
|
|
|
|
'parent_folder',
|
|
|
|
HTML('<hr>'),
|
|
|
|
HTML('<h5>Download configuration overloads</h5>'),
|
|
|
|
'auto_download',
|
|
|
|
'download_limit',
|
|
|
|
'download_order',
|
2018-10-27 00:33:45 +00:00
|
|
|
'delete_after_watched'
|
2018-10-17 21:38:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class UpdateSubscriptionModal(LoginRequiredMixin, ModalMixin, UpdateView):
|
2018-10-17 21:38:40 +00:00
|
|
|
template_name = 'YtManagerApp/controls/subscription_update_modal.html'
|
|
|
|
model = Subscription
|
|
|
|
form_class = UpdateSubscriptionForm
|
|
|
|
|
|
|
|
|
|
|
|
class DeleteSubscriptionForm(forms.Form):
|
|
|
|
keep_downloaded_videos = forms.BooleanField(required=False, initial=False, label="Keep downloaded videos")
|
|
|
|
|
|
|
|
|
2018-10-27 00:33:45 +00:00
|
|
|
class DeleteSubscriptionModal(LoginRequiredMixin, ModalMixin, FormMixin, DeleteView):
|
2018-10-17 21:38:40 +00:00
|
|
|
template_name = 'YtManagerApp/controls/subscription_delete_modal.html'
|
|
|
|
model = Subscription
|
|
|
|
form_class = DeleteSubscriptionForm
|
|
|
|
|
|
|
|
def delete(self, request, *args, **kwargs):
|
|
|
|
self.object = self.get_object()
|
|
|
|
form = self.get_form()
|
|
|
|
if form.is_valid():
|
|
|
|
return self.form_valid(form)
|
|
|
|
else:
|
|
|
|
return self.form_invalid(form)
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
self.object.delete_subscription(keep_downloaded_videos=form.cleaned_data['keep_downloaded_videos'])
|
|
|
|
return super().form_valid(form)
|
2018-11-03 12:43:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ImportSubscriptionsForm(forms.Form):
|
|
|
|
TRUE_FALSE_CHOICES = (
|
|
|
|
(None, '(default)'),
|
|
|
|
(True, 'Yes'),
|
|
|
|
(False, 'No')
|
|
|
|
)
|
|
|
|
|
|
|
|
VIDEO_ORDER_CHOICES_WITH_EMPTY = (
|
|
|
|
('', '(default)'),
|
|
|
|
*VIDEO_ORDER_CHOICES,
|
|
|
|
)
|
|
|
|
|
|
|
|
file = forms.FileField(label='File to import',
|
|
|
|
help_text='Supported file types: OPML, subscription list')
|
|
|
|
parent_folder = forms.ModelChoiceField(SubscriptionFolder.objects, required=False)
|
|
|
|
auto_download = forms.ChoiceField(choices=TRUE_FALSE_CHOICES, required=False)
|
|
|
|
download_limit = forms.IntegerField(required=False)
|
|
|
|
download_order = forms.ChoiceField(choices=VIDEO_ORDER_CHOICES_WITH_EMPTY, required=False)
|
|
|
|
delete_after_watched = forms.ChoiceField(choices=TRUE_FALSE_CHOICES, required=False)
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.yt_api = youtube.YoutubeAPI.build_public()
|
|
|
|
self.helper = FormHelper()
|
|
|
|
self.helper.form_tag = False
|
|
|
|
self.helper.layout = Layout(
|
|
|
|
'file',
|
|
|
|
'parent_folder',
|
|
|
|
HTML('<hr>'),
|
|
|
|
HTML('<h5>Download configuration overloads</h5>'),
|
|
|
|
'auto_download',
|
|
|
|
'download_limit',
|
|
|
|
'download_order',
|
|
|
|
'delete_after_watched'
|
|
|
|
)
|
|
|
|
|
|
|
|
def __clean_empty_none(self, name: str):
|
|
|
|
data = self.cleaned_data[name]
|
|
|
|
if isinstance(data, str) and len(data) == 0:
|
|
|
|
return None
|
|
|
|
return data
|
|
|
|
|
|
|
|
def __clean_boolean(self, name: str):
|
|
|
|
data = self.cleaned_data[name]
|
|
|
|
if isinstance(data, str) and len(data) == 0:
|
|
|
|
return None
|
|
|
|
if isinstance(data, str):
|
|
|
|
return data == 'True'
|
|
|
|
return data
|
|
|
|
|
|
|
|
def clean_auto_download(self):
|
|
|
|
return self.__clean_boolean('auto_download')
|
|
|
|
|
|
|
|
def clean_delete_after_watched(self):
|
|
|
|
return self.__clean_boolean('delete_after_watched')
|
|
|
|
|
|
|
|
def clean_download_order(self):
|
|
|
|
return self.__clean_empty_none('download_order')
|
|
|
|
|
|
|
|
|
|
|
|
class ImportSubscriptionsModal(LoginRequiredMixin, ModalMixin, FormView):
|
|
|
|
template_name = 'YtManagerApp/controls/subscriptions_import_modal.html'
|
|
|
|
form_class = ImportSubscriptionsForm
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
file = form.cleaned_data['file']
|
|
|
|
|
|
|
|
# Parse file
|
|
|
|
try:
|
|
|
|
url_list = list(subscription_file_parser.parse(file))
|
|
|
|
except subscription_file_parser.FormatNotSupportedError:
|
|
|
|
return super().modal_response(form, success=False,
|
|
|
|
error_msg="The file could not be parsed! "
|
|
|
|
"Possible problems: format not supported, file is malformed.")
|
|
|
|
|
|
|
|
print(form.cleaned_data)
|
|
|
|
|
|
|
|
# Create subscriptions
|
|
|
|
api = youtube.YoutubeAPI.build_public()
|
|
|
|
for url in url_list:
|
|
|
|
sub = Subscription()
|
|
|
|
sub.user = self.request.user
|
|
|
|
sub.parent_folder = form.cleaned_data['parent_folder']
|
|
|
|
sub.auto_download = form.cleaned_data['auto_download']
|
|
|
|
sub.download_limit = form.cleaned_data['download_limit']
|
|
|
|
sub.download_order = form.cleaned_data['download_order']
|
|
|
|
sub.delete_after_watched = form.cleaned_data['delete_after_watched']
|
|
|
|
try:
|
|
|
|
sub.fetch_from_url(url, api)
|
|
|
|
except Exception as e:
|
|
|
|
logging.error("Import subscription error - error processing URL %s: %s", url, e)
|
|
|
|
continue
|
|
|
|
|
|
|
|
sub.save()
|
|
|
|
|
|
|
|
return super().form_valid(form)
|