655 lines
15 KiB
C
655 lines
15 KiB
C
/*
|
|
* tvheadend, Automatic recordings
|
|
* Copyright (C) 2010 Andreas Öman
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <pthread.h>
|
|
#include <ctype.h>
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
#include <errno.h>
|
|
#include <math.h>
|
|
|
|
#include "tvheadend.h"
|
|
#include "settings.h"
|
|
#include "dvr.h"
|
|
#include "notify.h"
|
|
#include "dtable.h"
|
|
#include "epg.h"
|
|
|
|
dtable_t *autorec_dt;
|
|
|
|
TAILQ_HEAD(dvr_autorec_entry_queue, dvr_autorec_entry);
|
|
|
|
static int dvr_autorec_in_init = 0;
|
|
|
|
struct dvr_autorec_entry_queue autorec_entries;
|
|
|
|
static void dvr_autorec_changed(dvr_autorec_entry_t *dae, int purge);
|
|
|
|
/**
|
|
* Unlink - and remove any unstarted
|
|
*/
|
|
static void
|
|
dvr_autorec_purge_spawns(dvr_autorec_entry_t *dae)
|
|
{
|
|
dvr_entry_t *de;
|
|
|
|
while((de = LIST_FIRST(&dae->dae_spawns)) != NULL) {
|
|
LIST_REMOVE(de, de_autorec_link);
|
|
de->de_autorec = NULL;
|
|
if (de->de_sched_state == DVR_SCHEDULED)
|
|
dvr_entry_cancel(de);
|
|
else
|
|
dvr_entry_save(de);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* return 1 if the event 'e' is matched by the autorec rule 'dae'
|
|
*/
|
|
static int
|
|
autorec_cmp(dvr_autorec_entry_t *dae, epg_broadcast_t *e)
|
|
{
|
|
channel_tag_mapping_t *ctm;
|
|
dvr_config_t *cfg;
|
|
|
|
if (!e->channel) return 0;
|
|
if (!e->episode) return 0;
|
|
if(dae->dae_enabled == 0 || dae->dae_weekdays == 0)
|
|
return 0;
|
|
|
|
if(dae->dae_channel == NULL &&
|
|
dae->dae_channel_tag == NULL &&
|
|
dae->dae_content_type.code == 0 &&
|
|
(dae->dae_title == NULL ||
|
|
dae->dae_title[0] == '\0') &&
|
|
dae->dae_brand == NULL &&
|
|
dae->dae_season == NULL &&
|
|
dae->dae_serieslink == NULL)
|
|
return 0; // Avoid super wildcard match
|
|
|
|
// Note: we always test season first, though it will only be set
|
|
// if configured
|
|
if(dae->dae_serieslink) {
|
|
if (!e->serieslink || dae->dae_serieslink != e->serieslink) return 0;
|
|
return 1;
|
|
}
|
|
if(dae->dae_season)
|
|
if (!e->episode->season || dae->dae_season != e->episode->season) return 0;
|
|
if(dae->dae_brand)
|
|
if (!e->episode->brand || dae->dae_brand != e->episode->brand) return 0;
|
|
|
|
if(dae->dae_title != NULL && dae->dae_title[0] != '\0') {
|
|
lang_str_ele_t *ls;
|
|
if(!e->episode->title) return 0;
|
|
RB_FOREACH(ls, e->episode->title, link)
|
|
if (!regexec(&dae->dae_title_preg, ls->str, 0, NULL, 0)) break;
|
|
if (!ls) return 0;
|
|
}
|
|
|
|
// Note: ignore channel test if we allow quality unlocking
|
|
cfg = dvr_config_find_by_name_default(dae->dae_config_name);
|
|
if (cfg->dvr_sl_quality_lock)
|
|
if(dae->dae_channel != NULL &&
|
|
dae->dae_channel != e->channel)
|
|
return 0;
|
|
|
|
if(dae->dae_channel_tag != NULL) {
|
|
LIST_FOREACH(ctm, &dae->dae_channel_tag->ct_ctms, ctm_tag_link)
|
|
if(ctm->ctm_channel == e->channel)
|
|
break;
|
|
if(ctm == NULL)
|
|
return 0;
|
|
}
|
|
|
|
if(dae->dae_content_type.code != 0) {
|
|
if (!epg_genre_list_contains(&e->episode->genre, &dae->dae_content_type, 1))
|
|
return 0;
|
|
}
|
|
|
|
if(dae->dae_approx_time != 0) {
|
|
struct tm a_time;
|
|
struct tm ev_time;
|
|
localtime_r(&e->start, &a_time);
|
|
localtime_r(&e->start, &ev_time);
|
|
a_time.tm_min = dae->dae_approx_time % 60;
|
|
a_time.tm_hour = dae->dae_approx_time / 60;
|
|
if(abs(mktime(&a_time) - mktime(&ev_time)) > 900)
|
|
return 0;
|
|
}
|
|
|
|
if(dae->dae_weekdays != 0x7f) {
|
|
struct tm tm;
|
|
localtime_r(&e->start, &tm);
|
|
if(!((1 << ((tm.tm_wday ?: 7) - 1)) & dae->dae_weekdays))
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
dvr_autorec_entry_t *
|
|
autorec_entry_find(const char *id, int create)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
char buf[20];
|
|
static int tally;
|
|
|
|
if(id != NULL) {
|
|
TAILQ_FOREACH(dae, &autorec_entries, dae_link)
|
|
if(!strcmp(dae->dae_id, id))
|
|
return dae;
|
|
}
|
|
|
|
if(create == 0)
|
|
return NULL;
|
|
|
|
dae = calloc(1, sizeof(dvr_autorec_entry_t));
|
|
if(id == NULL) {
|
|
tally++;
|
|
snprintf(buf, sizeof(buf), "%d", tally);
|
|
id = buf;
|
|
} else {
|
|
tally = MAX(atoi(id), tally);
|
|
}
|
|
dae->dae_weekdays = 0x7f;
|
|
dae->dae_pri = DVR_PRIO_NORMAL;
|
|
|
|
dae->dae_id = strdup(id);
|
|
TAILQ_INSERT_TAIL(&autorec_entries, dae, dae_link);
|
|
return dae;
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static void
|
|
autorec_entry_destroy(dvr_autorec_entry_t *dae)
|
|
{
|
|
dvr_autorec_purge_spawns(dae);
|
|
|
|
free(dae->dae_id);
|
|
|
|
free(dae->dae_config_name);
|
|
free(dae->dae_creator);
|
|
free(dae->dae_comment);
|
|
|
|
if(dae->dae_title != NULL) {
|
|
free(dae->dae_title);
|
|
regfree(&dae->dae_title_preg);
|
|
}
|
|
|
|
if(dae->dae_channel != NULL)
|
|
LIST_REMOVE(dae, dae_channel_link);
|
|
|
|
if(dae->dae_channel_tag != NULL)
|
|
LIST_REMOVE(dae, dae_channel_tag_link);
|
|
|
|
if(dae->dae_brand)
|
|
dae->dae_brand->putref(dae->dae_brand);
|
|
if(dae->dae_season)
|
|
dae->dae_season->putref(dae->dae_season);
|
|
if(dae->dae_serieslink)
|
|
dae->dae_serieslink->putref(dae->dae_serieslink);
|
|
|
|
|
|
TAILQ_REMOVE(&autorec_entries, dae, dae_link);
|
|
free(dae);
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static void
|
|
build_weekday_tags(htsmsg_t *l, int mask)
|
|
{
|
|
int i;
|
|
for(i = 0; i < 7; i++) {
|
|
if(mask & (1 << i))
|
|
htsmsg_add_u32(l, NULL, i+1);
|
|
}
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static int
|
|
build_weekday_mask(htsmsg_t *l)
|
|
{
|
|
int r = 0;
|
|
uint32_t u32;
|
|
htsmsg_field_t *f;
|
|
HTSMSG_FOREACH(f, l)
|
|
if (!htsmsg_field_get_u32(f, &u32))
|
|
r |= 1 << (u32 - 1);
|
|
return r;
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static htsmsg_t *
|
|
autorec_record_build(dvr_autorec_entry_t *dae)
|
|
{
|
|
htsmsg_t *e = htsmsg_create_map();
|
|
htsmsg_t *l = htsmsg_create_list();
|
|
|
|
htsmsg_add_str(e, "id", dae->dae_id);
|
|
htsmsg_add_u32(e, "enabled", !!dae->dae_enabled);
|
|
|
|
if (dae->dae_config_name != NULL)
|
|
htsmsg_add_str(e, "config_name", dae->dae_config_name);
|
|
if(dae->dae_creator != NULL)
|
|
htsmsg_add_str(e, "creator", dae->dae_creator);
|
|
if(dae->dae_comment != NULL)
|
|
htsmsg_add_str(e, "comment", dae->dae_comment);
|
|
|
|
if(dae->dae_channel != NULL)
|
|
htsmsg_add_str(e, "channel", channel_get_uuid(dae->dae_channel));
|
|
|
|
if(dae->dae_channel_tag != NULL)
|
|
htsmsg_add_str(e, "tag", dae->dae_channel_tag->ct_name);
|
|
|
|
htsmsg_add_u32(e, "contenttype",dae->dae_content_type.code);
|
|
|
|
htsmsg_add_str(e, "title", dae->dae_title ?: "");
|
|
|
|
htsmsg_add_u32(e, "approx_time", dae->dae_approx_time);
|
|
|
|
build_weekday_tags(l, dae->dae_weekdays);
|
|
htsmsg_add_msg(e, "weekdays", l);
|
|
|
|
htsmsg_add_str(e, "pri", dvr_val2pri(dae->dae_pri));
|
|
|
|
if (dae->dae_brand)
|
|
htsmsg_add_str(e, "brand", dae->dae_brand->uri);
|
|
if (dae->dae_season)
|
|
htsmsg_add_str(e, "season", dae->dae_season->uri);
|
|
if (dae->dae_serieslink)
|
|
htsmsg_add_str(e, "serieslink", dae->dae_serieslink->uri);
|
|
|
|
return e;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static htsmsg_t *
|
|
autorec_record_get_all(void *opaque)
|
|
{
|
|
htsmsg_t *r = htsmsg_create_list();
|
|
dvr_autorec_entry_t *dae;
|
|
|
|
TAILQ_FOREACH(dae, &autorec_entries, dae_link)
|
|
htsmsg_add_msg(r, NULL, autorec_record_build(dae));
|
|
|
|
return r;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static htsmsg_t *
|
|
autorec_record_get(void *opaque, const char *id)
|
|
{
|
|
dvr_autorec_entry_t *ae;
|
|
|
|
if((ae = autorec_entry_find(id, 0)) == NULL)
|
|
return NULL;
|
|
return autorec_record_build(ae);
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static htsmsg_t *
|
|
autorec_record_create(void *opaque)
|
|
{
|
|
return autorec_record_build(autorec_entry_find(NULL, 1));
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static htsmsg_t *
|
|
autorec_record_update(void *opaque, const char *id, htsmsg_t *values,
|
|
int maycreate)
|
|
{
|
|
int save;
|
|
dvr_autorec_entry_t *dae;
|
|
const char *s;
|
|
channel_t *ch;
|
|
channel_tag_t *ct;
|
|
uint32_t u32;
|
|
htsmsg_t *l;
|
|
|
|
if((dae = autorec_entry_find(id, maycreate)) == NULL)
|
|
return NULL;
|
|
|
|
tvh_str_update(&dae->dae_config_name, htsmsg_get_str(values, "config_name"));
|
|
tvh_str_update(&dae->dae_creator, htsmsg_get_str(values, "creator"));
|
|
tvh_str_update(&dae->dae_comment, htsmsg_get_str(values, "comment"));
|
|
|
|
if((s = htsmsg_get_str(values, "channel")) != NULL) {
|
|
if(dae->dae_channel != NULL) {
|
|
LIST_REMOVE(dae, dae_channel_link);
|
|
dae->dae_channel = NULL;
|
|
}
|
|
ch = channel_find(s);
|
|
if (!ch) ch = channel_find_by_name(s);
|
|
if (ch) {
|
|
LIST_INSERT_HEAD(&ch->ch_autorecs, dae, dae_channel_link);
|
|
dae->dae_channel = ch;
|
|
}
|
|
}
|
|
|
|
if((s = htsmsg_get_str(values, "title")) != NULL) {
|
|
if(dae->dae_title != NULL) {
|
|
free(dae->dae_title);
|
|
dae->dae_title = NULL;
|
|
regfree(&dae->dae_title_preg);
|
|
}
|
|
|
|
if(!regcomp(&dae->dae_title_preg, s,
|
|
REG_ICASE | REG_EXTENDED | REG_NOSUB)) {
|
|
dae->dae_title = strdup(s);
|
|
}
|
|
}
|
|
|
|
if((s = htsmsg_get_str(values, "tag")) != NULL) {
|
|
if(dae->dae_channel_tag != NULL) {
|
|
LIST_REMOVE(dae, dae_channel_tag_link);
|
|
dae->dae_channel_tag = NULL;
|
|
}
|
|
if((ct = channel_tag_find_by_name(s, 0)) != NULL) {
|
|
LIST_INSERT_HEAD(&ct->ct_autorecs, dae, dae_channel_tag_link);
|
|
dae->dae_channel_tag = ct;
|
|
}
|
|
}
|
|
|
|
if (!htsmsg_get_u32(values, "contenttype", &u32))
|
|
dae->dae_content_type.code = u32;
|
|
|
|
if((s = htsmsg_get_str(values, "approx_time")) != NULL) {
|
|
if(strchr(s, ':') != NULL) {
|
|
// formatted time string - convert
|
|
dae->dae_approx_time = (atoi(s) * 60) + atoi(s + 3);
|
|
} else if(strlen(s) == 0) {
|
|
dae->dae_approx_time = 0;
|
|
} else {
|
|
dae->dae_approx_time = atoi(s);
|
|
}
|
|
}
|
|
|
|
if((l = htsmsg_get_list(values, "weekdays")) != NULL)
|
|
dae->dae_weekdays = build_weekday_mask(l);
|
|
|
|
if(!htsmsg_get_u32(values, "enabled", &u32))
|
|
dae->dae_enabled = u32;
|
|
|
|
if((s = htsmsg_get_str(values, "pri")) != NULL)
|
|
dae->dae_pri = dvr_pri2val(s);
|
|
|
|
if((s = htsmsg_get_str(values, "brand")) != NULL) {
|
|
dae->dae_brand = epg_brand_find_by_uri(s, 1, &save);
|
|
if (dae->dae_brand)
|
|
dae->dae_brand->getref((epg_object_t*)dae->dae_brand);
|
|
}
|
|
if((s = htsmsg_get_str(values, "season")) != NULL) {
|
|
dae->dae_season = epg_season_find_by_uri(s, 1, &save);
|
|
if (dae->dae_season)
|
|
dae->dae_season->getref((epg_object_t*)dae->dae_season);
|
|
}
|
|
if((s = htsmsg_get_str(values, "serieslink")) != NULL) {
|
|
dae->dae_serieslink = epg_serieslink_find_by_uri(s, 1, &save);
|
|
if (dae->dae_serieslink)
|
|
dae->dae_serieslink->getref(dae->dae_serieslink);
|
|
}
|
|
if (!dvr_autorec_in_init)
|
|
dvr_autorec_changed(dae, 1);
|
|
|
|
return autorec_record_build(dae);
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static int
|
|
autorec_record_delete(void *opaque, const char *id)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
|
|
if((dae = autorec_entry_find(id, 0)) == NULL)
|
|
return -1;
|
|
autorec_entry_destroy(dae);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static const dtable_class_t autorec_dtc = {
|
|
.dtc_record_get = autorec_record_get,
|
|
.dtc_record_get_all = autorec_record_get_all,
|
|
.dtc_record_create = autorec_record_create,
|
|
.dtc_record_update = autorec_record_update,
|
|
.dtc_record_delete = autorec_record_delete,
|
|
.dtc_read_access = ACCESS_RECORDER,
|
|
.dtc_write_access = ACCESS_RECORDER,
|
|
.dtc_mutex = &global_lock,
|
|
};
|
|
|
|
/**
|
|
*
|
|
*/
|
|
void
|
|
dvr_autorec_init(void)
|
|
{
|
|
TAILQ_INIT(&autorec_entries);
|
|
autorec_dt = dtable_create(&autorec_dtc, "autorec", NULL);
|
|
dvr_autorec_in_init = 1;
|
|
dtable_load(autorec_dt);
|
|
dvr_autorec_in_init = 0;
|
|
}
|
|
|
|
void
|
|
dvr_autorec_update(void)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
TAILQ_FOREACH(dae, &autorec_entries, dae_link) {
|
|
dvr_autorec_changed(dae, 0);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_dvr_autorec_add(const char *config_name,
|
|
const char *title, channel_t *ch,
|
|
const char *tag, epg_genre_t *content_type,
|
|
epg_brand_t *brand, epg_season_t *season,
|
|
epg_serieslink_t *serieslink,
|
|
int approx_time, epg_episode_num_t *epnum,
|
|
const char *creator, const char *comment)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
htsmsg_t *m;
|
|
channel_tag_t *ct;
|
|
|
|
if((dae = autorec_entry_find(NULL, 1)) == NULL)
|
|
return;
|
|
|
|
tvh_str_set(&dae->dae_config_name, config_name);
|
|
tvh_str_set(&dae->dae_creator, creator);
|
|
tvh_str_set(&dae->dae_comment, comment);
|
|
|
|
if(ch) {
|
|
LIST_INSERT_HEAD(&ch->ch_autorecs, dae, dae_channel_link);
|
|
dae->dae_channel = ch;
|
|
}
|
|
|
|
if(title != NULL &&
|
|
!regcomp(&dae->dae_title_preg, title,
|
|
REG_ICASE | REG_EXTENDED | REG_NOSUB)) {
|
|
dae->dae_title = strdup(title);
|
|
}
|
|
|
|
if(tag != NULL && (ct = channel_tag_find_by_name(tag, 0)) != NULL) {
|
|
LIST_INSERT_HEAD(&ct->ct_autorecs, dae, dae_channel_tag_link);
|
|
dae->dae_channel_tag = ct;
|
|
}
|
|
|
|
dae->dae_enabled = 1;
|
|
if (content_type)
|
|
dae->dae_content_type.code = content_type->code;
|
|
|
|
if(serieslink) {
|
|
serieslink->getref(serieslink);
|
|
dae->dae_serieslink = serieslink;
|
|
}
|
|
|
|
dae->dae_approx_time = approx_time;
|
|
|
|
m = autorec_record_build(dae);
|
|
hts_settings_save(m, "%s/%s", "autorec", dae->dae_id);
|
|
htsmsg_destroy(m);
|
|
|
|
/* Notify web clients that we have messed with the tables */
|
|
|
|
notify_reload("autorec");
|
|
|
|
dvr_autorec_changed(dae, 1);
|
|
}
|
|
|
|
void
|
|
dvr_autorec_add(const char *config_name,
|
|
const char *title, const char *channel,
|
|
const char *tag, epg_genre_t *content_type,
|
|
const char *creator, const char *comment)
|
|
{
|
|
channel_t *ch = NULL;
|
|
if(channel != NULL) ch = channel_find(channel);
|
|
_dvr_autorec_add(config_name, title, ch, tag, content_type,
|
|
NULL, NULL, NULL, 0, NULL, creator, comment);
|
|
}
|
|
|
|
void dvr_autorec_add_series_link
|
|
( const char *dvr_config_name, epg_broadcast_t *event,
|
|
const char *creator, const char *comment )
|
|
{
|
|
char *title;
|
|
if (!event || !event->episode) return;
|
|
title = regexp_escape(epg_broadcast_get_title(event, NULL));
|
|
_dvr_autorec_add(dvr_config_name,
|
|
title,
|
|
event->channel,
|
|
NULL, 0, // tag/content type
|
|
NULL,
|
|
NULL,
|
|
event->serieslink,
|
|
0, NULL,
|
|
creator, comment);
|
|
if (title)
|
|
free(title);
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
void
|
|
dvr_autorec_check_event(epg_broadcast_t *e)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
|
|
TAILQ_FOREACH(dae, &autorec_entries, dae_link)
|
|
if(autorec_cmp(dae, e))
|
|
dvr_entry_create_by_autorec(e, dae);
|
|
// Note: no longer updating event here as it will be done from EPG
|
|
// anyway
|
|
}
|
|
|
|
void dvr_autorec_check_brand(epg_brand_t *b)
|
|
{
|
|
// Note: for the most part this will only be relevant should an episode
|
|
// to which a broadcast is linked suddenly get added to a new brand
|
|
// this is pretty damn unlikely!
|
|
}
|
|
|
|
void dvr_autorec_check_season(epg_season_t *s)
|
|
{
|
|
// Note: I guess new episodes might have been added, but again its likely
|
|
// this will already have been picked up by the check_event call
|
|
}
|
|
|
|
void dvr_autorec_check_serieslink(epg_serieslink_t *s)
|
|
{
|
|
// TODO: need to implement this
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static void
|
|
dvr_autorec_changed(dvr_autorec_entry_t *dae, int purge)
|
|
{
|
|
channel_t *ch;
|
|
epg_broadcast_t *e;
|
|
|
|
if (purge)
|
|
dvr_autorec_purge_spawns(dae);
|
|
|
|
CHANNEL_FOREACH(ch) {
|
|
RB_FOREACH(e, &ch->ch_epg_schedule, sched_link) {
|
|
if(autorec_cmp(dae, e))
|
|
dvr_entry_create_by_autorec(e, dae);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
void
|
|
autorec_destroy_by_channel(channel_t *ch)
|
|
{
|
|
dvr_autorec_entry_t *dae;
|
|
htsmsg_t *m;
|
|
|
|
while((dae = LIST_FIRST(&ch->ch_autorecs)) != NULL) {
|
|
dtable_record_erase(autorec_dt, dae->dae_id);
|
|
autorec_entry_destroy(dae);
|
|
}
|
|
|
|
/* Notify web clients that we have messed with the tables */
|
|
m = htsmsg_create_map();
|
|
htsmsg_add_u32(m, "reload", 1);
|
|
notify_by_msg("autorec", m);
|
|
}
|