From 4e98db48681acf6528c96f2bea8e97b57d791017 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lo=C3=AFc=20Bonavent?= <56730254+LoicBonavent@users.noreply.github.com> Date: Fri, 1 Mar 2024 15:05:15 +0100 Subject: [PATCH] [DONE] Feature import video from BBB platform with token (#1065) * Add search_internal_recording function * Manage the links of recordings (video and presentation playback) to take into account the single-use token * Add recording_with_token route * Add functions to manage recordings used by an infrastructure that require a token * Add the possibility to upload a recording from BBB ESR infrastructure * Translations added for this new feature * Manage translations in search_internal_recording function * Manage translations in some functions * Translations added for this new feature --- pod/import_video/models.py | 54 ++++++++ .../templates/import_video/list.html | 63 ++++++++- pod/import_video/urls.py | 5 + pod/import_video/utils.py | 122 +++++++++++++++-- pod/import_video/views.py | 126 +++++++++++++++--- pod/locale/fr/LC_MESSAGES/django.po | 53 +++++--- pod/locale/fr/LC_MESSAGES/djangojs.po | 2 +- pod/locale/nl/LC_MESSAGES/django.po | 38 ++++-- pod/locale/nl/LC_MESSAGES/djangojs.po | 2 +- pod/meeting/models.py | 16 +-- 10 files changed, 406 insertions(+), 75 deletions(-) diff --git a/pod/import_video/models.py b/pod/import_video/models.py index a724b77a0a..c6e4ec6228 100644 --- a/pod/import_video/models.py +++ b/pod/import_video/models.py @@ -1,4 +1,8 @@ """Models for the Import_video module.""" +import requests + +from urllib.parse import urlencode +import xml.etree.ElementTree as et from django.conf import settings from django.contrib.sites.models import Site @@ -9,7 +13,15 @@ from django.utils import timezone from django.utils.translation import ugettext_lazy as _ +from pod.meeting.utils import ( + api_call, + parseXmlToJson, + slash_join, +) + SITE_ID = getattr(settings, "SITE_ID", 1) +USE_MEETING = getattr(settings, "USE_MEETING", False) +BBB_API_URL = getattr(settings, "BBB_API_URL", "") class ExternalRecording(models.Model): @@ -121,6 +133,48 @@ class ExternalRecording(models.Model): help_text=_("User who uploaded to Pod the video file"), ) + def search_internal_recording(self, recording_id): + """Search for an internal recording that corresponds to recording_id. + + This function checks whether an external recording has been made on the BBB + environment used by internal recordings and the meetings module. + Typically, this function recovers the single-use token of a BBB session + performed on the ESR infrastructure. + """ + if USE_MEETING: + action = "getRecordings" + parameters = {} + parameters["recordID"] = recording_id + query = urlencode(parameters) + hashed = api_call(query, action) + if BBB_API_URL == "": + msg = {} + msg["error"] = _("Unable to call BBB server.") + msg["message"] = _("Parameter %s needs to be defined.") % "BBB_API_URL" + raise ValueError(msg) + url = slash_join(BBB_API_URL, action, "?%s" % hashed) + response = requests.get(url) + if response.status_code != 200: + msg = {} + msg["error"] = _("Unable to call BBB server.") + msg["returncode"] = response.status_code + msg["message"] = response.content.decode("utf-8") + raise ValueError(msg) + result = response.content.decode("utf-8") + xmldoc = et.fromstring(result) + recording_json = parseXmlToJson(xmldoc) + if recording_json.get("returncode", "") != "SUCCESS": + msg = {} + msg["error"] = _("Unable to get recording!") + msg["returncode"] = recording_json.get("returncode", "") + msg["messageKey"] = recording_json.get("messageKey", "") + msg["message"] = recording_json.get("message", "") + raise ValueError(msg) + else: + return recording_json + else: + return ValueError(_("Method not allowed")) + def __unicode__(self): return "%s - %s" % (self.id, self.name) diff --git a/pod/import_video/templates/import_video/list.html b/pod/import_video/templates/import_video/list.html index 2906f44c75..14d5434ed1 100644 --- a/pod/import_video/templates/import_video/list.html +++ b/pod/import_video/templates/import_video/list.html @@ -57,14 +57,28 @@ {{ record.type }} {% if record.presentationUrl != "" %} - - - + {% if record.recordingId == "" %} + + + + {% else %} + + + + {% endif %} {% endif %} {% if record.videoUrl != "" and record.videoUrl != record.presentationUrl %} - - - + {% if record.recordingId == "" %} + + + + {% else %} + + + + {% endif %} {% endif %} @@ -144,5 +158,42 @@ } return answer; } + + /** + * Function that allow to manage the link of recordings (video and presentation playback) to take into account the single-use token. + **/ + function manageTokenLink(type, element) { + // Get important information from element + let recordId = element.dataset.podRecordingId; + // AJAX Pod server to re-request the BBB server + urlGetRecording = "/import_video/recording_with_token/" + recordId + "/"; + + // Get infos from the recording (new token appears here) + fetch(urlGetRecording, { + method: "GET", + headers: { + "X-CSRFToken": "{{ csrf_token }}", + "X-Requested-With": "XMLHttpRequest", + }, + dataType: "html", + cache: "no-store", + }) + .then((response) => response.text()) + .then((data) => { + // Get new playback URL + let jsonData = JSON.parse(data); + var urlNew; + if (type == "presentation") { + urlNew = jsonData.presentationUrl; + } else { + urlNew = jsonData.videoUrl; + } + // Open the good URL in a new tab + window.open(urlNew, '_blank'); + }) + .catch((error) => { + console.debug(error); + }); + } {% endblock more_script %} diff --git a/pod/import_video/urls.py b/pod/import_video/urls.py index b53de34fe3..f101470917 100644 --- a/pod/import_video/urls.py +++ b/pod/import_video/urls.py @@ -28,4 +28,9 @@ views.delete_external_recording, name="delete_external_recording", ), + path( + "recording_with_token//", + views.recording_with_token, + name="recording_with_token", + ), ] diff --git a/pod/import_video/utils.py b/pod/import_video/utils.py index fe87b5a85f..28e335a3c1 100644 --- a/pod/import_video/utils.py +++ b/pod/import_video/utils.py @@ -11,6 +11,7 @@ from django.utils.html import mark_safe from django.utils.translation import gettext_lazy as _ from html.parser import HTMLParser +from pod.meeting.utils import slash_join from pod.video.models import Video from pod.video.models import Type from urllib.parse import parse_qs, urlparse @@ -48,6 +49,11 @@ VIDEOS_DIR = getattr(settings, "VIDEOS_DIR", "videos") +# The meeting application is activated? +USE_MEETING = getattr(settings, "USE_MEETING", False) +# BBB server API +BBB_API_URL = getattr(settings, "BBB_API_URL", "") + def secure_request_for_upload(request): """Check that the request is correct for uploading a recording. @@ -85,13 +91,8 @@ def parse_remote_file(session: Session, source_html_url: str): try: response = session.get(source_html_url) if response.status_code != 200: - msg = {} - msg["error"] = _( - "The HTML file for this recording was not found on the server." - ) - # If we want to display the 404/500... page to the user - # msg["message"] = response.content.decode("utf-8") - msg["message"] = _("Error number: %s") % response.status_code + # No more informations needed for common error + msg = '' raise ValueError(msg) # Parse the BBB video HTML file @@ -160,7 +161,7 @@ def manage_recording_url(source_url: str, video_file_add: str) -> str: if url.query: query = parse_qs(url.query, keep_blank_values=True) if query["token"][0]: - # 1st case (ex: ESR URL), URL likes (ex for ESR URL:) + # 1st case (ex: ESR URL), URL likes # https://_site_/recording/_recording_id/video?token=_token_ # Get recording unique identifier recording_id = url.path.split("/")[2] @@ -236,7 +237,7 @@ def download_video_file(session: Session, source_video_url: str, dest_file: str) # print(session.cookies.get_dict()) if response.status_code != 200: raise ValueError( - _("The video file for this recording " "was not found on the server.") + _("The video file for this recording was not found on the server.") ) with open(dest_file, "wb+") as file: @@ -361,7 +362,8 @@ def check_source_url(source_url: str): # noqa: C901 Platforms managed : - Mediacad platform (Médiathèque académique) : rewrite source URL if required and manage JSON API. - - Old BigBlueButton : source URL for old BBB presentation playback + - BBB ESR infrastructure : rewrite source URL if required + - Old BigBlueButton presentation : source URL for old BBB presentation playback """ base_url = "" media_id = "" @@ -428,6 +430,27 @@ def check_source_url(source_url: str): # noqa: C901 media_id, ) platform = "Mediacad" + elif source_url.find("bbb.numerique-esr.fr/video/") != -1: + # BBB ESR video link + # https://univ.scalelite.bbb.numerique-esr.fr/video/#id#/ + source_video_url = source_url + format = "m4v" + platform = "BBB_ESR" + elif source_url.find("bbb.numerique-esr.fr/playback/presentation/2.3/") != -1: + # BBB ESR presentation link : rewrite for video source URL + # https://univ.scalelite.bbb.numerique-esr.fr/playback/presentation/2.3/#id# + source_video_url = source_url.replace( + "/playback/presentation/2.3/", "/video/" + ) + "/" + format = "m4v" + platform = "BBB_ESR" + elif source_url.find("bbb.numerique-esr.fr/recording/") != -1: + # BBB ESR video or presentation link : rewrite for video source URL if pres + # https://univ.scalelite.bbb.numerique-esr.fr/recording/#id#/presentation? + # https://univ.scalelite.bbb.numerique-esr.fr/recording/#id#/video? + source_video_url = source_url.replace("/presentation", "/video") + format = "m4v" + platform = "BBB_ESR" elif source_url.find("/playback/presentation/2.0/playback.html?") != -1: # Old BBB 2.x (<2.3) presentation link # Conversion from @@ -454,8 +477,8 @@ def check_source_url(source_url: str): # noqa: C901 platform_type = TypeSourceURL( platform, source_video_url, format, url_api_video ) - if platform == "BBB_Presentation": - # Platform type: older BBB, format presentation + # Platform type: BBB ESR or older BBB, format presentation + if platform == "BBB_ESR" or platform == "BBB_Presentation": platform_type = TypeSourceURL(platform, source_video_url, format, "") return platform_type @@ -507,6 +530,79 @@ def move_file(source: str, destination: str): print(f"Error moving the file: {e}") +def check_url_format_presentation(source_url: str) -> bool: + """ Check if the URL looks like a BBB presentation.""" + presentation = False + # Management for old presentation URLs with BBB or Scalelite server + if source_url.find("/playback/presentation/2.0/playback.html?") != -1: + presentation = True + # Management for standard presentation URLs with BBB or Scalelite server + elif source_url.find("/playback/presentation/2.3/") != -1: + presentation = True + return presentation + + +def check_url_need_token(source_url: str) -> str: + """ Check if the URL is used by an infrastructure that need a token. + + Useful for generating the single-use token required to access video file. + 2 conditions : + - URL was created on the same BBB infrastructure as the meeting module, + - At present, only ESR's BBB infrastructure uses single-use tokens. + If both conditions are met, then returns the recording_id. + """ + if USE_MEETING: + # Calculate base URL for BBB meeting + meeting_base_url = slash_join(BBB_API_URL.replace("bigbluebutton/api", "")) + # If the URL was created on the same BBB ESR infrastructure as the meeting module + # Ex: + # https://univ.scalelite.bbb.numerique-esr.fr/video/##id##/ + # https://univ.scalelite.bbb.numerique-esr.fr/presentation/##id##/ + # https://univ.scalelite.bbb.numerique-esr.fr/recording/##id##/video?xxx + # https://univ.scalelite.bbb.numerique-esr.fr/recording/##id##/presentation?xxx + # https://univ.scalelite.bbb.numerique-esr.fr/playback/presentation/2.3/##id## + if ( + source_url.find(meeting_base_url) != -1 + and source_url.find("bbb.numerique-esr.fr/") != -1 + ): + # Get recording id + array_url = source_url.split("/") + recording_id = array_url[-2] + # Specific case + if recording_id == "2.3": + recording_id = array_url[-1] + return recording_id + else: + return "" + else: + return "" + + +def get_playbacks_urls_with_token(recording): + """ Get presentation and video source URL, with token, for a recording. + + Check if the recording was made by an infrastructure that need a token. + In such a case, request the BBB infrastructure to get playbacks with token. + """ + # Default values + presentation_url = "" + video_url = "" + # Check if the URL is used by an infrastructure that need a token + recording_id = check_url_need_token(recording.source_url) + if recording_id != "": + # Request the BBB infrastructure to get playbacks with token + recordings = recording.search_internal_recording(recording_id) + if type(recordings.get("recordings")) is dict: + for data in recordings["recordings"].values(): + for playback in data["playback"]: + if data["playback"][playback]["type"] == "video": + video_url = data["playback"][playback]["url"] + if data["playback"][playback]["type"] == "presentation": + presentation_url = data["playback"][playback]["url"] + # Return the 2 usefuls playbacks + return presentation_url, video_url + + class TypeSourceURL: """Manage external recording source URL. @@ -599,6 +695,8 @@ class StatelessRecording: presentationUrl = "" # Video playback URL, used as the source URL for the video file videoUrl = "" + # Recording id (BBB format), when created on the same BBB infra as meeting module + recordingId = "" def __init__(self, id, name, state): """Initiliaze.""" diff --git a/pod/import_video/views.py b/pod/import_video/views.py index 157ba5362b..c5f6d80df1 100644 --- a/pod/import_video/views.py +++ b/pod/import_video/views.py @@ -14,12 +14,13 @@ from .models import ExternalRecording from .forms import ExternalRecordingForm -from .utils import StatelessRecording, check_url_exists, download_video_file +from .utils import StatelessRecording, download_video_file from .utils import manage_recording_url, check_source_url, parse_remote_file from .utils import save_video, secure_request_for_upload from .utils import check_video_size, verify_video_exists_and_size from .utils import define_dest_file_and_path, check_file_exists, move_file -from .utils import TypeSourceURL +from .utils import check_url_format_presentation, check_url_need_token +from .utils import get_playbacks_urls_with_token, TypeSourceURL from datetime import datetime from django.conf import settings from django.contrib.auth.decorators import login_required @@ -27,6 +28,7 @@ from django.contrib.sites.shortcuts import get_current_site from django.core.exceptions import PermissionDenied from django.contrib import messages +from django.http import HttpResponse, HttpResponseBadRequest from django.shortcuts import render, redirect from django.shortcuts import get_object_or_404 from django.urls import reverse @@ -212,14 +214,12 @@ def external_recordings(request): owner__id=request.user.id, site=site ) # | request.user.owners_external_recordings.all().filter(site=site) - external_recordings = external_recordings.order_by("-id").distinct() + external_recordings = external_recordings.order_by("-start_at").distinct() recordings = [] for data in external_recordings: recording = get_stateless_recording(request, data) - recordings.append(recording) - return render( request, "import_video/list.html", @@ -467,6 +467,13 @@ def upload_video_recording_to_pod(request, record_id: int): # noqa: C901 return True else: return False + elif type_source_url is not None and type_source_url.type == "BBB_ESR": + # BBB ESR platform + # For BBB ESR infrastructure, there is always a video playback + return upload_bbb_esr_video_recording_to_pod( + record_id, + type_source_url.url + ) else: # Video file (or BBB video file, same process) source URL return upload_standard_video_recording_to_pod(record_id) @@ -507,8 +514,8 @@ def upload_standard_video_recording_to_pod(record_id: int) -> bool: recording = ExternalRecording.objects.get(id=record_id) # Step 1: Download and parse the remote HTML file if necessary - # Check if extension is a video extension source_url = recording.source_url + # Check if extension is a video extension extension = source_url.split(".")[-1].lower() # Name of the video file to add to the URL (if necessary) @@ -548,6 +555,59 @@ def upload_standard_video_recording_to_pod(record_id: int) -> bool: manage_standard_exception(exc) +def upload_bbb_esr_video_recording_to_pod(record_id: int, source_url: str) -> bool: + """Upload a BBB - from BBB ESR infrastructure - video file recording to Pod. + + Useful for generating the single-use token required to access the video file. + This is only possible for recordings made on the same BBB infrastructure used + by the meetings module. + """ + try: + # Session useful to achieve requests (and keep cookies between) + session = requests.Session() + + recording = ExternalRecording.objects.get(id=record_id) + + # Check if the URL is used by an infrastructure that need a token + # In such a case, request the BBB infrastructure to get playbacks with token + presentation_url, video_url = get_playbacks_urls_with_token(recording) + if video_url != "": + # Define the new source URL, with token + recording.source_url = video_url + source_url = video_url + + # Download and parse the remote HTML file (BBB specific) + video_file_add = parse_remote_file(session, source_url) + extension = video_file_add.split(".")[-1].lower() + + # Impossible to check simply that video exists + # and not oversized for BBB ESR infrastructure. + # This seems to be a technical constraint. + + # Step 2: Define destination source file + dest_file, dest_path = define_dest_file_and_path( + recording.owner, recording.id, extension + ) + + # Step 3: Download the video file + manage_download(session, source_url, video_file_add, dest_file) + + # Step 4: Save informations about the recording + save_external_recording(recording.owner, record_id) + + # Step 5: Save and encode Pod video + description = _( + 'This video has been uploaded to Pod and its source is the virtual ' + 'classroom solution of the French Ministry of Higher Education and Research.' + ) + + save_video(recording.owner, dest_path, recording.name, description) + + return True + except Exception as exc: + manage_standard_exception(exc) + + def upload_local_video_recording_to_pod(record_id: id, dest_file: str, dest_path: str): """Upload a local (typically in Pod filesystem) video file recording to Pod. Useful for video files that have been encoded following @@ -1078,32 +1138,33 @@ def get_stateless_recording(request, data: ExternalRecording): recording.state = get_status_recording(data) # Management of the external recording type - if data.type == "bigbluebutton": + if data.type == "bigbluebutton" or data.type == "video": # Manage BBB recording URL - # Management for old presentation URLs with BBB or Scalelite server - if data.source_url.find("/playback/presentation/2.0/playback.html?") != -1: - recording.presentationUrl = data.source_url - # Management for standard presentation URLs with BBB or Scalelite server - elif data.source_url.find("/playback/presentation/2.3/") != -1: + if check_url_format_presentation(data.source_url): + # Presentation format recording.presentationUrl = data.source_url - # Management of other situations else: + # Management of other situations recording.videoUrl = data.source_url + # Check if the URL is used by an infrastructure that need a token + recording.recordingId = check_url_need_token(data.source_url) + # If possible to encode BBB presentation # Useful for old BBB or BBB 2.6+ without video playback if USE_IMPORT_VIDEO_BBB_RECORDER and recording.presentationUrl != "": recording.canUpload = True recording.videoUrl = recording.presentationUrl - # In all case - if recording.videoUrl == "" or check_url_exists(recording.videoUrl) is False: + + # To optimize, do not check that url exists + if recording.videoUrl == "": recording.state = _( "No video file found. Upload to Pod as a video is not possible." ) recording.canUpload = False recording.videoUrl = "" else: - # For PeerTube, Video file, Youtube + # For PeerTube, Youtube recording.videoUrl = data.source_url # Display type label @@ -1116,8 +1177,39 @@ def get_status_recording(data: ExternalRecording) -> str: """Get the status of an external recording.""" if data.uploaded_to_pod_by is None and data.state is None: state = _("Video file not uploaded to Pod") - elif data.uploaded_to_pod_by is not None and data.state is None: + elif data.uploaded_to_pod_by is not None: state = _("Video file already uploaded to Pod") else: state = data.state return state + + +@csrf_protect +@ensure_csrf_cookie +@login_required(redirect_field_name="referrer") +def recording_with_token(request, id): + """Get for specific recording (recording created on BBB infrastructure that need a + token and used by meeting module), the presentation and video source URL, in JSON. + + Args: + request (Request): HTTP request + record_id (int): record id + + Raises: + PermissionDenied: if user not allowed + + Returns: + HTTPResponse: internal recording (JSON format) + """ + recording = ExternalRecording.objects.get(id=id) + + # Check if the URL is used by an infrastructure that need a token + # In such a case, request the BBB infrastructure to get playbacks with token + presentation_url, video_url = get_playbacks_urls_with_token(recording) + + # JSON format + data = '{"presentationUrl": "%s", "videoUrl": "%s"}' % (presentation_url, video_url) + if request.is_ajax(): + return HttpResponse(data, content_type="application/json") + else: + return HttpResponseBadRequest() diff --git a/pod/locale/fr/LC_MESSAGES/django.po b/pod/locale/fr/LC_MESSAGES/django.po index 4ee0860109..a4e9c88e11 100644 --- a/pod/locale/fr/LC_MESSAGES/django.po +++ b/pod/locale/fr/LC_MESSAGES/django.po @@ -5,7 +5,7 @@ msgid "" msgstr "" "Project-Id-Version: Pod\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2024-02-29 10:01+0000\n" +"POT-Creation-Date: 2024-03-01 13:49+0100\n" "PO-Revision-Date: \n" "Last-Translator: obado \n" "Language-Team: Pod Team cotech-esup-pod@esup-portail.org\n" @@ -2545,6 +2545,23 @@ msgstr "Etat de l’enregistrement" msgid "User who uploaded to Pod the video file" msgstr "Utilisateur qui a téléversé le fichier vidéo sur Pod" +#: pod/import_video/models.py pod/meeting/models.py +msgid "Unable to call BBB server." +msgstr "Impossible de joindre le serveur BBB." + +#: pod/import_video/models.py +#, python-format +msgid "Parameter %s needs to be defined." +msgstr "Le paramètre %s doit être défini." + +#: pod/import_video/models.py pod/meeting/models.py +msgid "Unable to get recording!" +msgstr "Impossible d’obtenir l’enregistrement !" + +#: pod/import_video/models.py +msgid "Method not allowed" +msgstr "Méthode non autorisée" + #: pod/import_video/models.py msgid "External recordings" msgstr "Enregistrements externes" @@ -2855,15 +2872,6 @@ msgstr "" "Essayez de modifier le type d’enregistrement ou l’adresse de cet " "enregistrement" -#: pod/import_video/utils.py -msgid "The HTML file for this recording was not found on the server." -msgstr "Le fichier HTML de cet enregistrement est introuvable sur le serveur." - -#: pod/import_video/utils.py -#, python-format -msgid "Error number: %s" -msgstr "Numéro d’erreur : %s" - #: pod/import_video/utils.py msgid "The video file for this recording was not found in the HTML file." msgstr "" @@ -2947,8 +2955,17 @@ msgid "" "This video was uploaded to Pod; its origin is %(type)s: %(url)s" msgstr "" -"Cette vidéo a été téléversée sur Pod ; son origine est %(type)s : %(url)s" +"Cette vidéo a été téléversée sur Pod ; son origine est %(type)s : %(url)s" + +#: pod/import_video/views.py +msgid "" +"This video has been uploaded to Pod and its source is the virtual classroom " +"solution of the French Ministry of Higher Education and Research." +msgstr "" +"Cette vidéo a été téléversée sur Pod et sa source est la solution de classe " +"virtuelle du Ministère Français de l'Enseignement Supérieur et de la " +"Recherche." #: pod/import_video/views.py #, python-format @@ -5594,10 +5611,6 @@ msgstr "Impossible de terminer la réunion !" msgid "Unable to get meeting recordings!" msgstr "Impossible d’obtenir les enregistrements de la réunion !" -#: pod/meeting/models.py -msgid "Unable to get recording!" -msgstr "Impossible d’obtenir l’enregistrement !" - #: pod/meeting/models.py msgid "Unable to delete recording!" msgstr "Impossible de supprimer l’enregistrement !" @@ -9265,6 +9278,14 @@ msgstr "Résultats de la recherche" msgid "Esup-Pod xAPI" msgstr "xAPI Esup-Pod" +#~ msgid "The HTML file for this recording was not found on the server." +#~ msgstr "" +#~ "Le fichier HTML de cet enregistrement est introuvable sur le serveur." + +#, python-format +#~ msgid "Error number: %s" +#~ msgstr "Numéro d’erreur : %s" + #~ msgid "Playlist image" #~ msgstr "Image de la playlist" diff --git a/pod/locale/fr/LC_MESSAGES/djangojs.po b/pod/locale/fr/LC_MESSAGES/djangojs.po index ffae9cd7d0..11a9ebb8a1 100644 --- a/pod/locale/fr/LC_MESSAGES/djangojs.po +++ b/pod/locale/fr/LC_MESSAGES/djangojs.po @@ -5,7 +5,7 @@ msgid "" msgstr "" "Project-Id-Version: Esup-Pod\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2024-02-29 10:01+0000\n" +"POT-Creation-Date: 2024-03-01 13:49+0100\n" "PO-Revision-Date: \n" "Last-Translator: obado \n" "Language-Team: \n" diff --git a/pod/locale/nl/LC_MESSAGES/django.po b/pod/locale/nl/LC_MESSAGES/django.po index 7acb5d6772..a2d63bde3c 100644 --- a/pod/locale/nl/LC_MESSAGES/django.po +++ b/pod/locale/nl/LC_MESSAGES/django.po @@ -5,7 +5,7 @@ msgid "" msgstr "" "Project-Id-Version: Pod\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2024-02-29 10:01+0000\n" +"POT-Creation-Date: 2024-03-01 13:49+0100\n" "PO-Revision-Date: 2023-06-08 14:37+0200\n" "Last-Translator: obado \n" "Language-Team: \n" @@ -2421,6 +2421,23 @@ msgstr "" msgid "User who uploaded to Pod the video file" msgstr "" +#: pod/import_video/models.py pod/meeting/models.py +msgid "Unable to call BBB server." +msgstr "" + +#: pod/import_video/models.py +#, python-format +msgid "Parameter %s needs to be defined." +msgstr "" + +#: pod/import_video/models.py pod/meeting/models.py +msgid "Unable to get recording!" +msgstr "" + +#: pod/import_video/models.py +msgid "Method not allowed" +msgstr "" + #: pod/import_video/models.py msgid "External recordings" msgstr "" @@ -2684,15 +2701,6 @@ msgstr "" msgid "Try changing the record type or address for this recording" msgstr "" -#: pod/import_video/utils.py -msgid "The HTML file for this recording was not found on the server." -msgstr "" - -#: pod/import_video/utils.py -#, python-format -msgid "Error number: %s" -msgstr "" - #: pod/import_video/utils.py msgid "The video file for this recording was not found in the HTML file." msgstr "" @@ -2770,6 +2778,12 @@ msgid "" "target=\"_blank\">%(url)s" msgstr "" +#: pod/import_video/views.py +msgid "" +"This video has been uploaded to Pod and its source is the virtual classroom " +"solution of the French Ministry of Higher Education and Research." +msgstr "" + #: pod/import_video/views.py #, python-format msgid "" @@ -5299,10 +5313,6 @@ msgstr "" msgid "Unable to get meeting recordings!" msgstr "" -#: pod/meeting/models.py -msgid "Unable to get recording!" -msgstr "" - #: pod/meeting/models.py msgid "Unable to delete recording!" msgstr "" diff --git a/pod/locale/nl/LC_MESSAGES/djangojs.po b/pod/locale/nl/LC_MESSAGES/djangojs.po index ca7c586d33..af89d1a839 100644 --- a/pod/locale/nl/LC_MESSAGES/djangojs.po +++ b/pod/locale/nl/LC_MESSAGES/djangojs.po @@ -5,7 +5,7 @@ msgid "" msgstr "" "Project-Id-Version: Esup-Pod\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2024-02-29 10:01+0000\n" +"POT-Creation-Date: 2024-03-01 13:49+0100\n" "PO-Revision-Date: 2023-02-08 15:22+0100\n" "Last-Translator: obado \n" "Language-Team: \n" diff --git a/pod/meeting/models.py b/pod/meeting/models.py index bb45dfc512..7ee879af89 100644 --- a/pod/meeting/models.py +++ b/pod/meeting/models.py @@ -635,7 +635,7 @@ def create(self, request=None): response = self.get_create_response(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -774,7 +774,7 @@ def get_is_meeting_running(self): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -810,7 +810,7 @@ def get_meeting_info(self): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -838,7 +838,7 @@ def end(self): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -868,7 +868,7 @@ def get_recordings(self): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -897,7 +897,7 @@ def get_recording(self, record_id): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -925,7 +925,7 @@ def delete_recording(self, record_id): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg) @@ -955,7 +955,7 @@ def get_all_meetings(): response = requests.get(url) if response.status_code != 200: msg = {} - msg["error"] = "Unable to call BBB server." + msg["error"] = _("Unable to call BBB server.") msg["returncode"] = response.status_code msg["message"] = response.content.decode("utf-8") raise ValueError(msg)