diff --git a/beat/web/attestations/management/commands/send_attestation_cleanup_warning.py b/beat/web/attestations/management/commands/send_attestation_cleanup_warning.py index 230f681e0c87f5fcfe6a1a43828697e33ae7b2ae..932174edd3588c4986cdef40e3dbbd0c5a71983c 100644 --- a/beat/web/attestations/management/commands/send_attestation_cleanup_warning.py +++ b/beat/web/attestations/management/commands/send_attestation_cleanup_warning.py @@ -27,6 +27,7 @@ ############################################################################### +from django.core.management import call_command from django.core.management.base import BaseCommand, CommandError from django.core.mail import send_mail from django.template.loader import render_to_string @@ -45,26 +46,29 @@ class Command(BaseCommand): help = 'Send email warning for attestations about to expire' def handle(self, *args, **options): - expiration_date = date.today() + timedelta(days=7) - attestations_about_to_expire = Attestation.objects.filter(locked=True, expiration_date__range=(datetime.combine(expiration_date, time.min), - datetime.combine(expiration_date, time.max))) - if attestations_about_to_expire: - current_site = Site.objects.get_current() - template_path = 'attestations/attestation_about_to_expire_email.txt' - for attestation in attestations_about_to_expire: - subject = "Attestation for experiment %s is about to expire" % \ - attestation.experiment.fullname() + for expiration_reminder in settings.EXPIRATION_REMINDERS: + expiration_date = date.today() + timedelta(days=expiration_reminder) + attestations_about_to_expire = Attestation.objects.filter(locked=True, expiration_date__range=(datetime.combine(expiration_date, time.min), + datetime.combine(expiration_date, time.max))) + if attestations_about_to_expire: + current_site = Site.objects.get_current() + template_path = 'attestations/attestation_about_to_expire_email.txt' + for attestation in attestations_about_to_expire: + subject = "Attestation for experiment %s is about to expire" % \ + attestation.experiment.fullname() - send_mail(subject, - render_to_string(template_path, - { - 'attestation': attestation, - 'beat_version': __version__, - 'site': current_site, - } - ), - settings.DEFAULT_FROM_EMAIL, - [attestation.experiment.author.email]) - self.stdout.write('{} attestation(s) about to expire'.format(attestations_about_to_expire.count())) - else: - self.stdout.write('No attestation(s) about to expire') + send_mail(subject, + render_to_string(template_path, + { + 'attestation': attestation, + 'beat_version': __version__, + 'site': current_site, + } + ), + settings.DEFAULT_FROM_EMAIL, + [attestation.experiment.author.email]) + self.stdout.write('{} attestation(s) about to expire'.format(attestations_about_to_expire.count())) + else: + self.stdout.write('No attestation(s) about to expire in {} day(s)'.format(expiration_reminder)) + + call_command('clean_attestations', '--noinput') diff --git a/beat/web/attestations/tests.py b/beat/web/attestations/tests.py index 28367c3c56c8cd4ce32a34bf5310ab666b8572ec..28e0e2cb0d9b104a4934219a4617fb0278a6629b 100644 --- a/beat/web/attestations/tests.py +++ b/beat/web/attestations/tests.py @@ -590,7 +590,7 @@ class CleanAttstationManagementCommandTestCase(AttestationsAPIBase): def test_outdated_attestation(self): experiment = Experiment.objects.all()[0] attestation = Attestation.objects.create_attestation(experiment) - attestation.expiration_date = attestation.expiration_date - timedelta(days=100) + attestation.expiration_date = attestation.expiration_date - timedelta(days=200) attestation.save() command_output = self.run_command() self.assertEqual(command_output, '1 attestation(s) successfully cleaned') diff --git a/beat/web/reports/admin.py b/beat/web/reports/admin.py index 5c4b12c1ff6e2129bc71929b0a1ba7791f8e99f3..705f488cf11c16ae269d03846a6c7ffeab33e166 100644 --- a/beat/web/reports/admin.py +++ b/beat/web/reports/admin.py @@ -84,7 +84,7 @@ class ReportAdmin(admin.ModelAdmin): ('Dates', dict( classes=('collapse',), - fields=('creation_date', 'publication_date',), + fields=('creation_date', 'expiration_date', 'publication_date',), ), ), ('Documentation', @@ -100,7 +100,7 @@ class ReportAdmin(admin.ModelAdmin): ), ) - list_display = ('id', 'name', 'number', 'author', 'creation_date', 'publication_date') + list_display = ('id', 'name', 'number', 'author', 'creation_date', 'expiration_date', 'publication_date') search_fields = [ 'author__username', 'name', diff --git a/beat/web/reports/api.py b/beat/web/reports/api.py index cf912d2e03568d2df41bd98b5dbe2acb818b7236..0cd920858bb71bcb6fa0964c72cae80041065a0b 100644 --- a/beat/web/reports/api.py +++ b/beat/web/reports/api.py @@ -58,7 +58,7 @@ from ..common.exceptions import ShareError from ..common.mixins import CommonContextMixin from itertools import chain -from datetime import datetime +from datetime import datetime, timedelta from .permissions import IsAuthor, IsAuthorOrPublished, IsAuthorOrAccessible, IsAccessibleOutside @@ -333,6 +333,7 @@ class LockReportView(BaseReportActionView): return ForbiddenResponse('Report is empty') report.status = Report.LOCKED + report.expiration_date = datetime.now() + timedelta(days=settings.EXPIRATION_DELTA) report.save() return Response(status=status.HTTP_204_NO_CONTENT) @@ -376,6 +377,7 @@ class PublishReportView(BaseReportActionView): report.status = Report.PUBLISHED report.publication_date = datetime.now() + report.expiration_date = None report.save() return Response(status=status.HTTP_204_NO_CONTENT) diff --git a/beat/web/reports/management/__init__.py b/beat/web/reports/management/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/beat/web/reports/management/commands/__init__.py b/beat/web/reports/management/commands/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/beat/web/reports/management/commands/clean_report.py b/beat/web/reports/management/commands/clean_report.py new file mode 100644 index 0000000000000000000000000000000000000000..7be69d626a1c30e83886a0ff444f3852c78fa9b5 --- /dev/null +++ b/beat/web/reports/management/commands/clean_report.py @@ -0,0 +1,90 @@ +#!/usr/bin/env python +# vim: set fileencoding=utf-8 : +# encoding: utf-8 + +############################################################################### +# # +# Copyright (c) 2016 Idiap Research Institute, http://www.idiap.ch/ # +# Contact: beat.support@idiap.ch # +# # +# This file is part of the beat.web module of the BEAT platform. # +# # +# Commercial License Usage # +# Licensees holding valid commercial BEAT licenses may use this file in # +# accordance with the terms contained in a written agreement between you # +# and Idiap. For further information contact tto@idiap.ch # +# # +# Alternatively, this file may be used under the terms of the GNU Affero # +# Public License version 3 as published by the Free Software and appearing # +# in the file LICENSE.AGPL included in the packaging of this file. # +# The BEAT platform is distributed in the hope that it will be useful, but # +# WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY # +# or FITNESS FOR A PARTICULAR PURPOSE. # +# # +# You should have received a copy of the GNU Affero Public License along # +# with the BEAT platform. If not, see http://www.gnu.org/licenses/. # +# # +############################################################################### + + +from django.core.management.base import BaseCommand, CommandError + +from datetime import date + +from ...models import Report + +import sys +import random + +class Command(BaseCommand): + + help = 'Cleanup outdated locked reports' + + def add_arguments(self, parser): + parser.add_argument('--noinput', action='store_false', dest='interactive', default=True, + help=('Tells Django to NOT prompt the user for input of any kind.')) + + + def handle(self, *args, **options): + clean = True + + if options['interactive']: + try: + answer = self.get_input_data('Make report(s) editable again with new unique id (Y/n)? ', 'y').lower() + except KeyboardInterrupt: + self.stderr.write("\nOperation canceled.") + sys.exit(1) + + if answer != 'y': + self.stdout.write('Cleanup canceled') + sys.exit(1) + + if clean: + expired_reports = Report.objects.filter(status=Report.LOCKED, expiration_date__lte=date.today()) + report_count = expired_reports.count() + for expired_report in expired_reports: + # Generate a unique report number + used_numbers = map(lambda x: x.number, Report.objects.all()) + number = 0 + while (number == 0) or number in used_numbers: + number = random.randint(100000, 2**31) + + expired_report.status = Report.EDITABLE + expired_report.number = number + expired_report.expiration_date = None + expired_report.save() + + self.stdout.write('{} locked report(s) successfully cleaned'.format(report_count)) + + + def get_input_data(self, message, default=None): + """ + Override this method if you want to customize data inputs or + validation exceptions. + """ + raw_value = raw_input(message) + + if default and raw_value == '': + raw_value = default + + return raw_value diff --git a/beat/web/reports/management/commands/send_report_cleanup_warning_and_cleanup.py b/beat/web/reports/management/commands/send_report_cleanup_warning_and_cleanup.py new file mode 100644 index 0000000000000000000000000000000000000000..03a0b863a27940e3a33d2e316f1dd1d2f359167a --- /dev/null +++ b/beat/web/reports/management/commands/send_report_cleanup_warning_and_cleanup.py @@ -0,0 +1,75 @@ +#!/usr/bin/env python +# vim: set fileencoding=utf-8 : +# encoding: utf-8 + +############################################################################### +# # +# Copyright (c) 2016 Idiap Research Institute, http://www.idiap.ch/ # +# Contact: beat.support@idiap.ch # +# # +# This file is part of the beat.web module of the BEAT platform. # +# # +# Commercial License Usage # +# Licensees holding valid commercial BEAT licenses may use this file in # +# accordance with the terms contained in a written agreement between you # +# and Idiap. For further information contact tto@idiap.ch # +# # +# Alternatively, this file may be used under the terms of the GNU Affero # +# Public License version 3 as published by the Free Software and appearing # +# in the file LICENSE.AGPL included in the packaging of this file. # +# The BEAT platform is distributed in the hope that it will be useful, but # +# WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY # +# or FITNESS FOR A PARTICULAR PURPOSE. # +# # +# You should have received a copy of the GNU Affero Public License along # +# with the BEAT platform. If not, see http://www.gnu.org/licenses/. # +# # +############################################################################### + + +from django.core.management import call_command +from django.core.management.base import BaseCommand, CommandError +from django.core.mail import send_mail +from django.template.loader import render_to_string +from django.conf import settings +from django.contrib.sites.models import Site + +from datetime import datetime, time, date, timedelta + +from ...models import Report +from .... import __version__ + +import sys + +class Command(BaseCommand): + + help = 'Send email warning for reports about to expire and cleanup old reports' + + def handle(self, *args, **options): + for expiration_reminder in settings.EXPIRATION_REMINDERS: + expiration_date = date.today() + timedelta(days=expiration_reminder) + + reports_about_to_expire = Report.objects.filter(status=Report.LOCKED, expiration_date__range=(datetime.combine(expiration_date, time.min), + datetime.combine(expiration_date, time.max))) + if reports_about_to_expire: + current_site = Site.objects.get_current() + template_path = 'reports/report_about_to_expire_email.txt' + for report in reports_about_to_expire: + subject = "Report %s is about to expire" % \ + report.name + + send_mail(subject, + render_to_string(template_path, + { + 'report': report, + 'beat_version': __version__, + 'site': current_site, + } + ), + settings.DEFAULT_FROM_EMAIL, + [report.author.email]) + self.stdout.write('{} report(s) about to expire'.format(reports_about_to_expire.count())) + else: + self.stdout.write('No report(s) about to expire in {} day(s)'.format(expiration_reminder)) + + call_command('clean_report', '--noinput') diff --git a/beat/web/reports/migrations/0002_report_expiration_date.py b/beat/web/reports/migrations/0002_report_expiration_date.py new file mode 100644 index 0000000000000000000000000000000000000000..cbfe719022be1cf7f9fb3d50cb6b15909b84160b --- /dev/null +++ b/beat/web/reports/migrations/0002_report_expiration_date.py @@ -0,0 +1,35 @@ +# -*- coding: utf-8 -*- +from __future__ import unicode_literals + +from django.db import migrations, models + +from datetime import datetime, timedelta +from django.conf import settings + +def add_timeout_to_existing_locked_report(apps, schema_editor): + report_model = apps.get_model("reports", "report") + + for single_report in report_model.objects.all(): + if single_report.status == 'L': + single_report.expiration_date = datetime.now() + timedelta(days=settings.EXPIRATION_DELTA) + single_report.save() + +def backward_dummy(apps, schema_editor): + pass + +class Migration(migrations.Migration): + + dependencies = [ + ('reports', '0001_initial'), + ] + + operations = [ + migrations.AddField( + model_name='report', + name='expiration_date', + field=models.DateTimeField(null=True, blank=True), + ), + + migrations.RunPython(add_timeout_to_existing_locked_report, + backward_dummy) + ] diff --git a/beat/web/reports/models.py b/beat/web/reports/models.py index 673bca2fb803fb737ab36dd0f3a29a7374762ba2..4a2c87c78aaa2e8c9234519510f5a8e9769556ec 100644 --- a/beat/web/reports/models.py +++ b/beat/web/reports/models.py @@ -60,6 +60,7 @@ class ReportManager(models.Manager): report.content = content report.creation_date = datetime.now() report.publication_date = None + report.expiration_date = None report.status = self.model.EDITABLE report.save() @@ -107,6 +108,7 @@ class Report(models.Model): author = models.ForeignKey(User, related_name='%(class)ss') experiments = models.ManyToManyField(Experiment, related_name='reports', blank=True) creation_date = models.DateTimeField() + expiration_date = models.DateTimeField(null=True, blank=True) publication_date = models.DateTimeField(null=True, blank=True) short_description = models.CharField(max_length=100, default='', blank=True, help_text=Messages['short_description']) description = models.TextField(default='', blank=True) diff --git a/beat/web/reports/templates/reports/report.html b/beat/web/reports/templates/reports/report.html index 0d07fc25d5dc835534e6781ac49a78030c90828a..0d4e922cc09abfcc7df7908598aa3df4fef80b88 100644 --- a/beat/web/reports/templates/reports/report.html +++ b/beat/web/reports/templates/reports/report.html @@ -134,6 +134,7 @@ {% endif %} {% elif status == 'Locked' %} <i class="fa fa-warning"></i> This report is <strong class="text-warning">{{ status }}</strong> (not yet published)<br/> +<i class="fa fa-calendar-o"></i> Expires in <strong>{{ report.expiration_date|naturaltime }}</strong>, on {{ report.expiration_date }} (publish it to make it permanent)<br/> {% endif %} </p> diff --git a/beat/web/reports/templates/reports/report_about_to_expire_email.txt b/beat/web/reports/templates/reports/report_about_to_expire_email.txt new file mode 100644 index 0000000000000000000000000000000000000000..1b7cac307ae20fe19fa541f0e980e1f80391707e --- /dev/null +++ b/beat/web/reports/templates/reports/report_about_to_expire_email.txt @@ -0,0 +1,17 @@ +Hello, + +We'd like you to know that your report {{ report.name }} +is still locked and about to expire. + +More details: http://{{ site.domain }}{{ report.get_author_absolute_url }} + +Notice that, if you don't take any action to set it to public, this +report will be set to editable mode on {{ report.expiration_date }}. + +We would like to inform you that the unique report id {{ report.get_absolute_url }} +(accessible via this url http://{{ site.domain }}{{ report.get_absolute_url }}) +will be modified too if your report is not made public very soon. + +-- +https://groups.google.com/forum/#!forum/beat-devel +BEAT version {{ beat_version }} diff --git a/beat/web/settings/settings.py b/beat/web/settings/settings.py index 63d886b82cdbca7fa6bd53e8a6388e07441045f4..2f53e5f11fa03376d9a63ac5886ef233a9ef62c8 100644 --- a/beat/web/settings/settings.py +++ b/beat/web/settings/settings.py @@ -170,7 +170,7 @@ ACCOUNTS_TO_EXCLUDE_FROM_TEAMS = [SYSTEM_ACCOUNT, PLOT_ACCOUNT, SCHEDULER_ACCOUN # ########################################################################### -DEFAULT_FROM_EMAIL = 'BEAT Development Platform <support@beat-eu.org>' +DEFAULT_FROM_EMAIL = 'BEAT Development Platform <beat.support@idiap.ch>' SERVER_EMAIL = DEFAULT_FROM_EMAIL EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend' @@ -351,7 +351,8 @@ REST_FRAMEWORK = { ############################################################################## #In days -EXPIRATION_DELTA = 90 +EXPIRATION_DELTA = 180 +EXPIRATION_REMINDERS = [1, 7, 30] ############################################################################## # diff --git a/beat/web/utils/management/commands/daily_cron_actions.py b/beat/web/utils/management/commands/daily_cron_actions.py new file mode 100644 index 0000000000000000000000000000000000000000..d92e736856a9b31dd9ed3f8b5775e80172cab478 --- /dev/null +++ b/beat/web/utils/management/commands/daily_cron_actions.py @@ -0,0 +1,54 @@ +#!/usr/bin/env python +# vim: set fileencoding=utf-8 : +# encoding: utf-8 + +############################################################################### +# # +# Copyright (c) 2016 Idiap Research Institute, http://www.idiap.ch/ # +# Contact: beat.support@idiap.ch # +# # +# This file is part of the beat.web module of the BEAT platform. # +# # +# Commercial License Usage # +# Licensees holding valid commercial BEAT licenses may use this file in # +# accordance with the terms contained in a written agreement between you # +# and Idiap. For further information contact tto@idiap.ch # +# # +# Alternatively, this file may be used under the terms of the GNU Affero # +# Public License version 3 as published by the Free Software and appearing # +# in the file LICENSE.AGPL included in the packaging of this file. # +# The BEAT platform is distributed in the hope that it will be useful, but # +# WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY # +# or FITNESS FOR A PARTICULAR PURPOSE. # +# # +# You should have received a copy of the GNU Affero Public License along # +# with the BEAT platform. If not, see http://www.gnu.org/licenses/. # +# # +############################################################################### + + +from django.core.management import call_command +from django.core.management.base import BaseCommand, CommandError +from django.core.mail import send_mail +from django.template.loader import render_to_string +from django.conf import settings +from django.contrib.sites.models import Site + +from datetime import datetime, time, date, timedelta + +from ....reports.models import Report +from .... import __version__ + +import sys + +class Command(BaseCommand): + + help = 'Daily CRON actions' + + def handle(self, *args, **options): + + # Send attestations cleanup warnings and cleanup attestations + call_command('send_attestation_cleanup_warning') + + # Send report cleanup warnings and cleanup reports + call_command('send_report_cleanup_warning_and_cleanup')