fix(playout): missing live show events (#2087)
This commit is contained in:
parent
be14fb8096
commit
ef1de34111
|
@ -26,11 +26,13 @@ class Show(models.Model):
|
||||||
)
|
)
|
||||||
|
|
||||||
live_auth_registered = models.BooleanField(
|
live_auth_registered = models.BooleanField(
|
||||||
|
default=False,
|
||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
db_column="live_stream_using_airtime_auth",
|
db_column="live_stream_using_airtime_auth",
|
||||||
)
|
)
|
||||||
live_auth_custom = models.BooleanField(
|
live_auth_custom = models.BooleanField(
|
||||||
|
default=False,
|
||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
db_column="live_stream_using_custom_auth",
|
db_column="live_stream_using_custom_auth",
|
||||||
|
@ -48,6 +50,10 @@ class Show(models.Model):
|
||||||
db_column="live_stream_pass",
|
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 is linkable if it has never been linked before. Once
|
||||||
# a show becomes unlinked it can not be linked again.
|
# a show becomes unlinked it can not be linked again.
|
||||||
linked = models.BooleanField()
|
linked = models.BooleanField()
|
||||||
|
|
|
@ -15,6 +15,7 @@ class ShowSerializer(serializers.ModelSerializer):
|
||||||
"image",
|
"image",
|
||||||
"foreground_color",
|
"foreground_color",
|
||||||
"background_color",
|
"background_color",
|
||||||
|
"live_enabled",
|
||||||
"linked",
|
"linked",
|
||||||
"linkable",
|
"linkable",
|
||||||
"auto_playlist",
|
"auto_playlist",
|
||||||
|
|
|
@ -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
|
|
@ -6379,6 +6379,9 @@ components:
|
||||||
type: string
|
type: string
|
||||||
nullable: true
|
nullable: true
|
||||||
maxLength: 6
|
maxLength: 6
|
||||||
|
live_enabled:
|
||||||
|
type: boolean
|
||||||
|
readOnly: true
|
||||||
linked:
|
linked:
|
||||||
type: boolean
|
type: boolean
|
||||||
linkable:
|
linkable:
|
||||||
|
@ -7185,6 +7188,9 @@ components:
|
||||||
type: string
|
type: string
|
||||||
nullable: true
|
nullable: true
|
||||||
maxLength: 6
|
maxLength: 6
|
||||||
|
live_enabled:
|
||||||
|
type: boolean
|
||||||
|
readOnly: true
|
||||||
linked:
|
linked:
|
||||||
type: boolean
|
type: boolean
|
||||||
linkable:
|
linkable:
|
||||||
|
@ -7202,6 +7208,7 @@ components:
|
||||||
- id
|
- id
|
||||||
- linkable
|
- linkable
|
||||||
- linked
|
- linked
|
||||||
|
- live_enabled
|
||||||
- name
|
- name
|
||||||
ShowDays:
|
ShowDays:
|
||||||
type: object
|
type: object
|
||||||
|
|
|
@ -10,6 +10,7 @@ from libretime_shared.datetime import (
|
||||||
time_in_seconds,
|
time_in_seconds,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from ..liquidsoap.models import StreamPreferences
|
||||||
from .events import EventKind
|
from .events import EventKind
|
||||||
|
|
||||||
EVENT_KEY_FORMAT = "%Y-%m-%d-%H-%M-%S"
|
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):
|
def get_schedule(api_client: ApiClient):
|
||||||
|
stream_preferences = StreamPreferences(**api_client.get_stream_preferences().json())
|
||||||
|
|
||||||
current_time = datetime.utcnow()
|
current_time = datetime.utcnow()
|
||||||
end_time = current_time + timedelta(days=1)
|
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_instance = api_client.get_show_instance(item["instance"]).json()
|
||||||
show = api_client.get_show(show_instance["show"]).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"]:
|
if item["file"]:
|
||||||
file = api_client.get_file(item["file"]).json()
|
file = api_client.get_file(item["file"]).json()
|
||||||
generate_file_events(events, item, file, show)
|
generate_file_events(events, item, file, show)
|
||||||
|
@ -70,6 +82,36 @@ def get_schedule(api_client: ApiClient):
|
||||||
return {"media": dict(sorted(events.items()))}
|
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(
|
def generate_file_events(
|
||||||
events: dict,
|
events: dict,
|
||||||
schedule: dict,
|
schedule: dict,
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import random
|
import random
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from dateutil.parser import isoparse
|
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.events import EventKind
|
||||||
from libretime_playout.player.schedule import (
|
from libretime_playout.player.schedule import (
|
||||||
generate_file_events,
|
generate_file_events,
|
||||||
|
generate_live_events,
|
||||||
generate_webstream_events,
|
generate_webstream_events,
|
||||||
get_schedule,
|
get_schedule,
|
||||||
)
|
)
|
||||||
|
@ -18,10 +20,10 @@ def _api_client_fixture():
|
||||||
return ApiClient(base_url=base_url, api_key="test_key")
|
return ApiClient(base_url=base_url, api_key="test_key")
|
||||||
|
|
||||||
|
|
||||||
SHOW_1 = {"id": 1, "name": "Show 1"}
|
SHOW_1 = {"id": 1, "name": "Show 1", "live_enabled": False}
|
||||||
SHOW_2 = {"id": 2, "name": "Show 2"}
|
SHOW_2 = {"id": 2, "name": "Show 2", "live_enabled": False}
|
||||||
SHOW_3 = {"id": 3, "name": "Show 3"}
|
SHOW_3 = {"id": 3, "name": "Show 3", "live_enabled": True}
|
||||||
SHOW_4 = {"id": 4, "name": "Show 4"}
|
SHOW_4 = {"id": 4, "name": "Show 4", "live_enabled": False}
|
||||||
|
|
||||||
SHOW_INSTANCE_1 = {
|
SHOW_INSTANCE_1 = {
|
||||||
"id": 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():
|
def test_generate_file_events():
|
||||||
schedule_1 = SCHEDULE_1.copy()
|
schedule_1 = SCHEDULE_1.copy()
|
||||||
schedule_1["starts_at"] = isoparse(schedule_1["starts_at"])
|
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):
|
def test_get_schedule(schedule, requests_mock, api_client: ApiClient):
|
||||||
base_url = api_client.base_url
|
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/schedule", json=schedule)
|
||||||
|
|
||||||
requests_mock.get(f"{base_url}/api/v2/shows/1", json=SHOW_1)
|
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",
|
"replay_gain": "4.52",
|
||||||
"filesize": 10000,
|
"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": {
|
"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,
|
"type": EventKind.FILE,
|
||||||
"row_id": 9,
|
"row_id": 9,
|
||||||
"start": "2022-09-05-13-00-00",
|
"start": "2022-09-05-13-00-00",
|
||||||
|
|
Loading…
Reference in New Issue