fix(playout): missing live show events (#2087)

This commit is contained in:
Jonas L 2022-09-06 14:09:04 +02:00 committed by GitHub
parent be14fb8096
commit ef1de34111
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 132 additions and 4 deletions

View File

@ -26,11 +26,13 @@ class Show(models.Model):
)
live_auth_registered = models.BooleanField(
default=False,
blank=True,
null=True,
db_column="live_stream_using_airtime_auth",
)
live_auth_custom = models.BooleanField(
default=False,
blank=True,
null=True,
db_column="live_stream_using_custom_auth",
@ -48,6 +50,10 @@ class Show(models.Model):
db_column="live_stream_pass",
)
@property
def live_enabled(self) -> bool:
return any((self.live_auth_registered, self.live_auth_custom))
# A show is linkable if it has never been linked before. Once
# a show becomes unlinked it can not be linked again.
linked = models.BooleanField()

View File

@ -15,6 +15,7 @@ class ShowSerializer(serializers.ModelSerializer):
"image",
"foreground_color",
"background_color",
"live_enabled",
"linked",
"linkable",
"auto_playlist",

View File

@ -0,0 +1,15 @@
from ...models import Show
def test_show_live_enabled():
show = Show(
name="My Test Show",
description="My test show description",
)
assert not show.live_enabled
show.live_auth_registered = True
assert show.live_enabled
show.live_auth_custom = True
assert show.live_enabled

View File

@ -6379,6 +6379,9 @@ components:
type: string
nullable: true
maxLength: 6
live_enabled:
type: boolean
readOnly: true
linked:
type: boolean
linkable:
@ -7185,6 +7188,9 @@ components:
type: string
nullable: true
maxLength: 6
live_enabled:
type: boolean
readOnly: true
linked:
type: boolean
linkable:
@ -7202,6 +7208,7 @@ components:
- id
- linkable
- linked
- live_enabled
- name
ShowDays:
type: object

View File

@ -10,6 +10,7 @@ from libretime_shared.datetime import (
time_in_seconds,
)
from ..liquidsoap.models import StreamPreferences
from .events import EventKind
EVENT_KEY_FORMAT = "%Y-%m-%d-%H-%M-%S"
@ -36,6 +37,8 @@ def insert_event(events: dict, event_key: str, event: dict):
def get_schedule(api_client: ApiClient):
stream_preferences = StreamPreferences(**api_client.get_stream_preferences().json())
current_time = datetime.utcnow()
end_time = current_time + timedelta(days=1)
@ -59,6 +62,15 @@ def get_schedule(api_client: ApiClient):
show_instance = api_client.get_show_instance(item["instance"]).json()
show = api_client.get_show(show_instance["show"]).json()
if show["live_enabled"]:
show_instance["starts_at"] = isoparse(show_instance["starts_at"])
show_instance["ends_at"] = isoparse(show_instance["ends_at"])
generate_live_events(
events,
show_instance,
stream_preferences.input_fade_transition,
)
if item["file"]:
file = api_client.get_file(item["file"]).json()
generate_file_events(events, item, file, show)
@ -70,6 +82,36 @@ def get_schedule(api_client: ApiClient):
return {"media": dict(sorted(events.items()))}
def generate_live_events(
events: dict,
show_instance: dict,
input_fade_transition: float,
):
transition = timedelta(seconds=input_fade_transition)
switch_off_event_key = datetime_to_event_key(show_instance["ends_at"] - transition)
kick_out_event_key = datetime_to_event_key(show_instance["ends_at"])
# If enabled, fade the input source out
if switch_off_event_key != kick_out_event_key:
switch_off_event = {
"type": EventKind.EVENT,
"event_type": "switch_off",
"start": switch_off_event_key,
"end": switch_off_event_key,
}
insert_event(events, switch_off_event_key, switch_off_event)
# Then kick the source out
kick_out_event = {
"type": EventKind.EVENT,
"event_type": "kick_out",
"start": kick_out_event_key,
"end": kick_out_event_key,
}
insert_event(events, kick_out_event_key, kick_out_event)
def generate_file_events(
events: dict,
schedule: dict,

View File

@ -1,4 +1,5 @@
import random
from datetime import timedelta
import pytest
from dateutil.parser import isoparse
@ -7,6 +8,7 @@ from libretime_api_client.v2 import ApiClient
from libretime_playout.player.events import EventKind
from libretime_playout.player.schedule import (
generate_file_events,
generate_live_events,
generate_webstream_events,
get_schedule,
)
@ -18,10 +20,10 @@ def _api_client_fixture():
return ApiClient(base_url=base_url, api_key="test_key")
SHOW_1 = {"id": 1, "name": "Show 1"}
SHOW_2 = {"id": 2, "name": "Show 2"}
SHOW_3 = {"id": 3, "name": "Show 3"}
SHOW_4 = {"id": 4, "name": "Show 4"}
SHOW_1 = {"id": 1, "name": "Show 1", "live_enabled": False}
SHOW_2 = {"id": 2, "name": "Show 2", "live_enabled": False}
SHOW_3 = {"id": 3, "name": "Show 3", "live_enabled": True}
SHOW_4 = {"id": 4, "name": "Show 4", "live_enabled": False}
SHOW_INSTANCE_1 = {
"id": 1,
@ -264,6 +266,40 @@ SCHEDULE = [
]
def test_generate_live_events():
show_instance_3 = SHOW_INSTANCE_3.copy()
show_instance_3["starts_at"] = isoparse(show_instance_3["starts_at"])
show_instance_3["ends_at"] = isoparse(show_instance_3["ends_at"])
result = {}
generate_live_events(result, show_instance_3, 0.0)
assert result == {
"2022-09-05-13-00-00": {
"type": EventKind.EVENT,
"event_type": "kick_out",
"start": "2022-09-05-13-00-00",
"end": "2022-09-05-13-00-00",
}
}
result = {}
generate_live_events(result, show_instance_3, 2.0)
assert result == {
"2022-09-05-12-59-58": {
"type": EventKind.EVENT,
"event_type": "switch_off",
"start": "2022-09-05-12-59-58",
"end": "2022-09-05-12-59-58",
},
"2022-09-05-13-00-00": {
"type": EventKind.EVENT,
"event_type": "kick_out",
"start": "2022-09-05-13-00-00",
"end": "2022-09-05-13-00-00",
},
}
def test_generate_file_events():
schedule_1 = SCHEDULE_1.copy()
schedule_1["starts_at"] = isoparse(schedule_1["starts_at"])
@ -349,6 +385,15 @@ def test_generate_webstream_events():
def test_get_schedule(schedule, requests_mock, api_client: ApiClient):
base_url = api_client.base_url
requests_mock.get(
f"{base_url}/api/v2/stream/preferences",
json={
"input_fade_transition": 2.0,
"message_format": 0,
"message_offline": "",
},
)
requests_mock.get(f"{base_url}/api/v2/schedule", json=schedule)
requests_mock.get(f"{base_url}/api/v2/shows/1", json=SHOW_1)
@ -541,7 +586,19 @@ def test_get_schedule(schedule, requests_mock, api_client: ApiClient):
"replay_gain": "4.52",
"filesize": 10000,
},
"2022-09-05-12-59-58": {
"type": EventKind.EVENT,
"event_type": "switch_off",
"start": "2022-09-05-12-59-58",
"end": "2022-09-05-12-59-58",
},
"2022-09-05-13-00-00": {
"type": EventKind.EVENT,
"event_type": "kick_out",
"start": "2022-09-05-13-00-00",
"end": "2022-09-05-13-00-00",
},
"2022-09-05-13-00-00_0": {
"type": EventKind.FILE,
"row_id": 9,
"start": "2022-09-05-13-00-00",