mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-11 09:45:27 +00:00
Merge pull request #2406 from bookwyrm-social/disable-imports
Allow admins to disable starting imports
This commit is contained in:
commit
9aab14ee96
8 changed files with 192 additions and 74 deletions
18
bookwyrm/migrations/0166_sitesettings_imports_enabled.py
Normal file
18
bookwyrm/migrations/0166_sitesettings_imports_enabled.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
# Generated by Django 3.2.16 on 2022-11-17 21:50
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("bookwyrm", "0165_alter_inviterequest_answer"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="sitesettings",
|
||||||
|
name="imports_enabled",
|
||||||
|
field=models.BooleanField(default=True),
|
||||||
|
),
|
||||||
|
]
|
|
@ -86,6 +86,9 @@ class SiteSettings(SiteModel):
|
||||||
admin_email = models.EmailField(max_length=255, null=True, blank=True)
|
admin_email = models.EmailField(max_length=255, null=True, blank=True)
|
||||||
footer_item = models.TextField(null=True, blank=True)
|
footer_item = models.TextField(null=True, blank=True)
|
||||||
|
|
||||||
|
# controls
|
||||||
|
imports_enabled = models.BooleanField(default=True)
|
||||||
|
|
||||||
field_tracker = FieldTracker(fields=["name", "instance_tagline", "logo"])
|
field_tracker = FieldTracker(fields=["name", "instance_tagline", "logo"])
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
|
|
@ -8,83 +8,94 @@
|
||||||
<div class="block">
|
<div class="block">
|
||||||
<h1 class="title">{% trans "Import Books" %}</h1>
|
<h1 class="title">{% trans "Import Books" %}</h1>
|
||||||
|
|
||||||
{% if recent_avg_hours or recent_avg_minutes %}
|
{% if site.imports_enabled %}
|
||||||
<div class="notification">
|
{% if recent_avg_hours or recent_avg_minutes %}
|
||||||
<p>
|
<div class="notification">
|
||||||
{% if recent_avg_hours %}
|
<p>
|
||||||
{% blocktrans trimmed with hours=recent_avg_hours|floatformat:0|intcomma %}
|
{% if recent_avg_hours %}
|
||||||
On average, recent imports have taken {{ hours }} hours.
|
{% blocktrans trimmed with hours=recent_avg_hours|floatformat:0|intcomma %}
|
||||||
{% endblocktrans %}
|
On average, recent imports have taken {{ hours }} hours.
|
||||||
{% else %}
|
{% endblocktrans %}
|
||||||
{% blocktrans trimmed with minutes=recent_avg_minutes|floatformat:0|intcomma %}
|
{% else %}
|
||||||
On average, recent imports have taken {{ minutes }} minutes.
|
{% blocktrans trimmed with minutes=recent_avg_minutes|floatformat:0|intcomma %}
|
||||||
{% endblocktrans %}
|
On average, recent imports have taken {{ minutes }} minutes.
|
||||||
|
{% endblocktrans %}
|
||||||
|
{% endif %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
<form class="box" name="import" action="/import" method="post" enctype="multipart/form-data">
|
||||||
|
{% csrf_token %}
|
||||||
|
|
||||||
|
<div class="columns">
|
||||||
|
<div class="column is-half">
|
||||||
|
<div class="field">
|
||||||
|
<label class="label" for="source">
|
||||||
|
{% trans "Data source:" %}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<div class="select">
|
||||||
|
<select name="source" id="source" aria-describedby="desc_source">
|
||||||
|
<option value="Goodreads" {% if current == 'Goodreads' %}selected{% endif %}>
|
||||||
|
{% trans "Goodreads (CSV)" %}
|
||||||
|
</option>
|
||||||
|
<option value="Storygraph" {% if current == 'Storygraph' %}selected{% endif %}>
|
||||||
|
{% trans "Storygraph (CSV)" %}
|
||||||
|
</option>
|
||||||
|
<option value="LibraryThing" {% if current == 'LibraryThing' %}selected{% endif %}>
|
||||||
|
{% trans "LibraryThing (TSV)" %}
|
||||||
|
</option>
|
||||||
|
<option value="OpenLibrary" {% if current == 'OpenLibrary' %}selected{% endif %}>
|
||||||
|
{% trans "OpenLibrary (CSV)" %}
|
||||||
|
</option>
|
||||||
|
<option value="Calibre" {% if current == 'Calibre' %}selected{% endif %}>
|
||||||
|
{% trans "Calibre (CSV)" %}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="help" id="desc_source">
|
||||||
|
{% blocktrans trimmed %}
|
||||||
|
You can download your Goodreads data from the
|
||||||
|
<a href="https://www.goodreads.com/review/import" target="_blank" rel="nofollow noopener noreferrer">Import/Export page</a>
|
||||||
|
of your Goodreads account.
|
||||||
|
{% endblocktrans %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="field">
|
||||||
|
<label class="label" for="id_csv_file">{% trans "Data file:" %}</label>
|
||||||
|
{{ import_form.csv_file }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="column is-half">
|
||||||
|
<div class="field">
|
||||||
|
<label class="label">
|
||||||
|
<input type="checkbox" name="include_reviews" checked> {% trans "Include reviews" %}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label class="label" for="privacy_import">
|
||||||
|
{% trans "Privacy setting for imported reviews:" %}
|
||||||
|
</label>
|
||||||
|
{% include 'snippets/privacy_select.html' with no_label=True privacy_uuid="import" %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button class="button is-primary" type="submit">{% trans "Import" %}</button>
|
||||||
|
</form>
|
||||||
|
{% else %}
|
||||||
|
<div class="box notification has-text-centered is-warning m-6 content">
|
||||||
|
<p class="mt-5">
|
||||||
|
<span class="icon icon-warning is-size-2" aria-hidden="true"></span>
|
||||||
|
</p>
|
||||||
|
<p class="mb-5">
|
||||||
|
{% trans "Imports are temporarily disabled; thank you for your patience." %}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<form class="box" name="import" action="/import" method="post" enctype="multipart/form-data">
|
|
||||||
{% csrf_token %}
|
|
||||||
|
|
||||||
<div class="columns">
|
|
||||||
<div class="column is-half">
|
|
||||||
<div class="field">
|
|
||||||
<label class="label" for="source">
|
|
||||||
{% trans "Data source:" %}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<div class="select">
|
|
||||||
<select name="source" id="source" aria-describedby="desc_source">
|
|
||||||
<option value="Goodreads" {% if current == 'Goodreads' %}selected{% endif %}>
|
|
||||||
{% trans "Goodreads (CSV)" %}
|
|
||||||
</option>
|
|
||||||
<option value="Storygraph" {% if current == 'Storygraph' %}selected{% endif %}>
|
|
||||||
{% trans "Storygraph (CSV)" %}
|
|
||||||
</option>
|
|
||||||
<option value="LibraryThing" {% if current == 'LibraryThing' %}selected{% endif %}>
|
|
||||||
{% trans "LibraryThing (TSV)" %}
|
|
||||||
</option>
|
|
||||||
<option value="OpenLibrary" {% if current == 'OpenLibrary' %}selected{% endif %}>
|
|
||||||
{% trans "OpenLibrary (CSV)" %}
|
|
||||||
</option>
|
|
||||||
<option value="Calibre" {% if current == 'Calibre' %}selected{% endif %}>
|
|
||||||
{% trans "Calibre (CSV)" %}
|
|
||||||
</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<p class="help" id="desc_source">
|
|
||||||
{% blocktrans trimmed %}
|
|
||||||
You can download your Goodreads data from the
|
|
||||||
<a href="https://www.goodreads.com/review/import" target="_blank" rel="nofollow noopener noreferrer">Import/Export page</a>
|
|
||||||
of your Goodreads account.
|
|
||||||
{% endblocktrans %}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="field">
|
|
||||||
<label class="label" for="id_csv_file">{% trans "Data file:" %}</label>
|
|
||||||
{{ import_form.csv_file }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="column is-half">
|
|
||||||
<div class="field">
|
|
||||||
<label class="label">
|
|
||||||
<input type="checkbox" name="include_reviews" checked> {% trans "Include reviews" %}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div class="field">
|
|
||||||
<label class="label" for="privacy_import">
|
|
||||||
{% trans "Privacy setting for imported reviews:" %}
|
|
||||||
</label>
|
|
||||||
{% include 'snippets/privacy_select.html' with no_label=True privacy_uuid="import" %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<button class="button is-primary" type="submit">{% trans "Import" %}</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="content block">
|
<div class="content block">
|
||||||
|
|
|
@ -11,6 +11,54 @@
|
||||||
|
|
||||||
{% block panel %}
|
{% block panel %}
|
||||||
|
|
||||||
|
<div class="block">
|
||||||
|
{% if site.imports_enabled %}
|
||||||
|
<details class="details-panel box">
|
||||||
|
<summary>
|
||||||
|
<span role="heading" aria-level="2" class="title is-6">
|
||||||
|
{% trans "Disable starting new imports" %}
|
||||||
|
</span>
|
||||||
|
<span class="details-close icon icon-x" aria-hidden="true"></span>
|
||||||
|
</summary>
|
||||||
|
<form
|
||||||
|
name="disable-imports"
|
||||||
|
id="disable-imports"
|
||||||
|
method="POST"
|
||||||
|
action="{% url 'settings-imports-disable' %}"
|
||||||
|
>
|
||||||
|
<div class="notification">
|
||||||
|
{% trans "This is only intended to be used when things have gone very wrong with imports and you need to pause the feature while addressing issues." %}
|
||||||
|
{% trans "While imports are disabled, users will not be allowed to start new imports, but existing imports will not be effected." %}
|
||||||
|
</div>
|
||||||
|
{% csrf_token %}
|
||||||
|
<div class="control">
|
||||||
|
<button type="submit" class="button is-danger">
|
||||||
|
{% trans "Disable imports" %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</details>
|
||||||
|
{% else %}
|
||||||
|
<form
|
||||||
|
name="enable-imports"
|
||||||
|
id="enable-imports"
|
||||||
|
method="POST"
|
||||||
|
action="{% url 'settings-imports-enable' %}"
|
||||||
|
class="box"
|
||||||
|
>
|
||||||
|
<div class="notification is-danger is-light">
|
||||||
|
{% trans "Users are currently unable to start new imports" %}
|
||||||
|
</div>
|
||||||
|
{% csrf_token %}
|
||||||
|
<div class="control">
|
||||||
|
<button type="submit" class="button is-success">
|
||||||
|
{% trans "Enable imports" %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="block">
|
<div class="block">
|
||||||
<div class="tabs">
|
<div class="tabs">
|
||||||
<ul>
|
<ul>
|
||||||
|
|
|
@ -301,6 +301,16 @@ urlpatterns = [
|
||||||
views.ImportList.as_view(),
|
views.ImportList.as_view(),
|
||||||
name="settings-imports-complete",
|
name="settings-imports-complete",
|
||||||
),
|
),
|
||||||
|
re_path(
|
||||||
|
r"^settings/imports/disable/?$",
|
||||||
|
views.disable_imports,
|
||||||
|
name="settings-imports-disable",
|
||||||
|
),
|
||||||
|
re_path(
|
||||||
|
r"^settings/imports/enable/?$",
|
||||||
|
views.enable_imports,
|
||||||
|
name="settings-imports-enable",
|
||||||
|
),
|
||||||
re_path(
|
re_path(
|
||||||
r"^settings/celery/?$", views.CeleryStatus.as_view(), name="settings-celery"
|
r"^settings/celery/?$", views.CeleryStatus.as_view(), name="settings-celery"
|
||||||
),
|
),
|
||||||
|
|
|
@ -10,7 +10,7 @@ from .admin.federation import Federation, FederatedServer
|
||||||
from .admin.federation import AddFederatedServer, ImportServerBlocklist
|
from .admin.federation import AddFederatedServer, ImportServerBlocklist
|
||||||
from .admin.federation import block_server, unblock_server, refresh_server
|
from .admin.federation import block_server, unblock_server, refresh_server
|
||||||
from .admin.email_blocklist import EmailBlocklist
|
from .admin.email_blocklist import EmailBlocklist
|
||||||
from .admin.imports import ImportList
|
from .admin.imports import ImportList, disable_imports, enable_imports
|
||||||
from .admin.ip_blocklist import IPBlocklist
|
from .admin.ip_blocklist import IPBlocklist
|
||||||
from .admin.invite import ManageInvites, Invite, InviteRequest
|
from .admin.invite import ManageInvites, Invite, InviteRequest
|
||||||
from .admin.invite import ManageInviteRequests, ignore_invite_request
|
from .admin.invite import ManageInviteRequests, ignore_invite_request
|
||||||
|
|
|
@ -5,6 +5,7 @@ from django.shortcuts import get_object_or_404, redirect
|
||||||
from django.template.response import TemplateResponse
|
from django.template.response import TemplateResponse
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.views import View
|
from django.views import View
|
||||||
|
from django.views.decorators.http import require_POST
|
||||||
|
|
||||||
from bookwyrm import models
|
from bookwyrm import models
|
||||||
from bookwyrm.settings import PAGE_LENGTH
|
from bookwyrm.settings import PAGE_LENGTH
|
||||||
|
@ -53,3 +54,25 @@ class ImportList(View):
|
||||||
import_job = get_object_or_404(models.ImportJob, id=import_id)
|
import_job = get_object_or_404(models.ImportJob, id=import_id)
|
||||||
import_job.stop_job()
|
import_job.stop_job()
|
||||||
return redirect("settings-imports")
|
return redirect("settings-imports")
|
||||||
|
|
||||||
|
|
||||||
|
@require_POST
|
||||||
|
@permission_required("bookwyrm.edit_instance_settings", raise_exception=True)
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def disable_imports(request):
|
||||||
|
"""When you just need people to please stop starting imports"""
|
||||||
|
site = models.SiteSettings.objects.get()
|
||||||
|
site.imports_enabled = False
|
||||||
|
site.save(update_fields=["imports_enabled"])
|
||||||
|
return redirect("settings-imports")
|
||||||
|
|
||||||
|
|
||||||
|
@require_POST
|
||||||
|
@permission_required("bookwyrm.edit_instance_settings", raise_exception=True)
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def enable_imports(request):
|
||||||
|
"""When you just need people to please stop starting imports"""
|
||||||
|
site = models.SiteSettings.objects.get()
|
||||||
|
site.imports_enabled = True
|
||||||
|
site.save(update_fields=["imports_enabled"])
|
||||||
|
return redirect("settings-imports")
|
||||||
|
|
|
@ -4,6 +4,7 @@ import datetime
|
||||||
|
|
||||||
from django.contrib.auth.decorators import login_required
|
from django.contrib.auth.decorators import login_required
|
||||||
from django.db.models import Avg, ExpressionWrapper, F, fields
|
from django.db.models import Avg, ExpressionWrapper, F, fields
|
||||||
|
from django.core.exceptions import PermissionDenied
|
||||||
from django.core.paginator import Paginator
|
from django.core.paginator import Paginator
|
||||||
from django.http import HttpResponseBadRequest
|
from django.http import HttpResponseBadRequest
|
||||||
from django.shortcuts import redirect
|
from django.shortcuts import redirect
|
||||||
|
@ -54,6 +55,10 @@ class Import(View):
|
||||||
|
|
||||||
def post(self, request):
|
def post(self, request):
|
||||||
"""ingest a goodreads csv"""
|
"""ingest a goodreads csv"""
|
||||||
|
site = models.SiteSettings.objects.get()
|
||||||
|
if not site.imports_enabled:
|
||||||
|
raise PermissionDenied()
|
||||||
|
|
||||||
form = forms.ImportForm(request.POST, request.FILES)
|
form = forms.ImportForm(request.POST, request.FILES)
|
||||||
if not form.is_valid():
|
if not form.is_valid():
|
||||||
return HttpResponseBadRequest()
|
return HttpResponseBadRequest()
|
||||||
|
|
Loading…
Reference in a new issue