diff --git a/apps/api/schedule.py b/apps/api/schedule.py index cf26c7b77..4e405c500 100644 --- a/apps/api/schedule.py +++ b/apps/api/schedule.py @@ -216,6 +216,20 @@ def post(self): ) proposal.c3voc_url = None + if payload["voctoweb"]["thumb_path"]: + path = payload["voctoweb"]["thumb_path"] + if path.startswith("/static.media.ccc.de"): + path = "https://static.media.ccc.de/media" + path[len("/static.media.ccc.de"):] + if not path.startswith("https://"): + abort(406, message="voctoweb thumb_path must start with https:// or /static.media.ccc.de") + app.logger.info(f"C3VOC webhook set thumbnail_url for {proposal.id=} to {path}") + proposal.thumbnail_url = path + else: + app.logger.warning( + f"C3VOC webhook cleared thumbnail_url for {proposal.id=}, was {proposal.thumbnail_url}" + ) + proposal.thumbnail_url = None + if payload["youtube"]["enabled"]: if payload["youtube"]["urls"]: # Please do not overwrite existing youtube urls diff --git a/tests/test_api_c3voc.py b/tests/test_api_c3voc.py index 3ab335352..3a2e99e52 100644 --- a/tests/test_api_c3voc.py +++ b/tests/test_api_c3voc.py @@ -17,10 +17,11 @@ def proposal(db, user): return proposal -def clean_proposal(db, proposal, c3voc_url=None, youtube_url=None, video_recording_lost=True): +def clean_proposal(db, proposal, c3voc_url=None, youtube_url=None, thumbnail_url=None, video_recording_lost=True): proposal.c3voc_url = c3voc_url - proposal.youtube_url = youtube_url + proposal.thumbnail_url = thumbnail_url proposal.video_recording_lost = video_recording_lost + proposal.youtube_url = youtube_url db.session.add(proposal) db.session.commit() @@ -109,7 +110,7 @@ def test_denies_request_wrong_year(client, app, proposal): assert rv.status_code == 422 -def test_request_none_update_none(client, app, db, proposal): +def test_request_none_unchanged(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -144,7 +145,7 @@ def test_request_none_update_none(client, app, db, proposal): assert proposal.c3voc_url is None -def test_request_voctoweb_update_voctoweb_correct_url(client, app, db, proposal): +def test_update_voctoweb_with_correct_url(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -167,6 +168,7 @@ def test_request_voctoweb_update_voctoweb_correct_url(client, app, db, proposal) "voctoweb": { "enabled": True, "frontend_url": "https://media.ccc.de/", + "thumb_path": "", }, "youtube": { "enabled": False, @@ -181,7 +183,7 @@ def test_request_voctoweb_update_voctoweb_correct_url(client, app, db, proposal) assert proposal.youtube_url is None -def test_request_voctoweb_update_voctoweb_wrong_url(client, app, db, proposal): +def test_denies_voctoweb_with_wrong_url(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -204,6 +206,7 @@ def test_request_voctoweb_update_voctoweb_wrong_url(client, app, db, proposal): "voctoweb": { "enabled": True, "frontend_url": "https://example.org", + "thumb_path": "", }, "youtube": { "enabled": False, @@ -218,7 +221,7 @@ def test_request_voctoweb_update_voctoweb_wrong_url(client, app, db, proposal): assert proposal.c3voc_url == "https://example.com" -def test_request_voctoweb_clears_voctoweb(client, app, db, proposal): +def test_clears_voctoweb(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -241,6 +244,81 @@ def test_request_voctoweb_clears_voctoweb(client, app, db, proposal): "voctoweb": { "enabled": True, "frontend_url": "", + "thumb_path": "", + }, + "youtube": { + "enabled": False, + }, + }, + ) + assert rv.status_code == 204 + + proposal = Proposal.query.get(proposal.id) + assert proposal.c3voc_url is None + + +def test_update_thumbnail_with_path(client, app, db, proposal): + app.config.update( + { + "VIDEO_API_KEY": "api-key", + } + ) + + clean_proposal(db, proposal) + + rv = client.post( + f"/api/proposal/c3voc-publishing-webhook", + headers={ + "Authorization": "Bearer api-key", + }, + json={ + "is_master": True, + "fahrplan": { + "conference": f"emf{event_year()}", + "id": proposal.id, + }, + "voctoweb": { + "enabled": True, + "frontend_url": "", + "thumb_path": "/static.media.ccc.de/thumb.jpg", + }, + "youtube": { + "enabled": False, + }, + }, + ) + assert rv.status_code == 204 + + proposal = Proposal.query.get(proposal.id) + assert proposal.thumbnail_url == "https://static.media.ccc.de/media/thumb.jpg" + assert proposal.c3voc_url is None + assert proposal.youtube_url is None + + +def test_update_thumbnail_with_url(client, app, db, proposal): + app.config.update( + { + "VIDEO_API_KEY": "api-key", + } + ) + + clean_proposal(db, proposal) + + rv = client.post( + f"/api/proposal/c3voc-publishing-webhook", + headers={ + "Authorization": "Bearer api-key", + }, + json={ + "is_master": True, + "fahrplan": { + "conference": f"emf{event_year()}", + "id": proposal.id, + }, + "voctoweb": { + "enabled": True, + "frontend_url": "", + "thumb_path": "https://example.com/thumb.jpg", }, "youtube": { "enabled": False, @@ -250,10 +328,84 @@ def test_request_voctoweb_clears_voctoweb(client, app, db, proposal): assert rv.status_code == 204 proposal = Proposal.query.get(proposal.id) + assert proposal.thumbnail_url == "https://example.com/thumb.jpg" assert proposal.c3voc_url is None + assert proposal.youtube_url is None + + +def test_denies_thumbnail_not_url(client, app, db, proposal): + app.config.update( + { + "VIDEO_API_KEY": "api-key", + } + ) + + clean_proposal(db, proposal, thumbnail_url="https://example.com/thumb.jpg") + + rv = client.post( + f"/api/proposal/c3voc-publishing-webhook", + headers={ + "Authorization": "Bearer api-key", + }, + json={ + "is_master": True, + "fahrplan": { + "conference": f"emf{event_year()}", + "id": proposal.id, + }, + "voctoweb": { + "enabled": True, + "frontend_url": "", + "thumb_path": "/example.com/thumb.jpg", + }, + "youtube": { + "enabled": False, + }, + }, + ) + assert rv.status_code == 406 + + proposal = Proposal.query.get(proposal.id) + assert proposal.thumbnail_url == "https://example.com/thumb.jpg" + + +def test_clears_thumbnail(client, app, db, proposal): + app.config.update( + { + "VIDEO_API_KEY": "api-key", + } + ) + + clean_proposal(db, proposal, thumbnail_url="https://example.com/thumb.jpg") + + rv = client.post( + f"/api/proposal/c3voc-publishing-webhook", + headers={ + "Authorization": "Bearer api-key", + }, + json={ + "is_master": True, + "fahrplan": { + "conference": f"emf{event_year()}", + "id": proposal.id, + }, + "voctoweb": { + "enabled": True, + "frontend_url": "", + "thumb_path": "", + }, + "youtube": { + "enabled": False, + }, + }, + ) + assert rv.status_code == 204 + + proposal = Proposal.query.get(proposal.id) + assert proposal.thumbnail_url is None -def test_request_youtube_update_youtube_correct_url(client, app, db, proposal): +def test_update_youtube_with_correct_url(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -292,7 +444,7 @@ def test_request_youtube_update_youtube_correct_url(client, app, db, proposal): assert proposal.youtube_url == "https://www.youtube.com/watch" -def test_request_youtube_update_youtube_correct_url_but_existing_url(client, app, db, proposal): +def test_denies_youtube_update_with_exisiting_url(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -331,7 +483,7 @@ def test_request_youtube_update_youtube_correct_url_but_existing_url(client, app assert proposal.youtube_url == "https://example.com" -def test_request_youtube_update_youtube_wrong_url(client, app, db, proposal): +def test_denies_youtube_update_with_wrong_url(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key", @@ -370,7 +522,7 @@ def test_request_youtube_update_youtube_wrong_url(client, app, db, proposal): assert proposal.youtube_url == "https://example.com" -def test_request_youtube_clears_youtube(client, app, db, proposal): +def test_clears_youtube(client, app, db, proposal): app.config.update( { "VIDEO_API_KEY": "api-key",