2000-03-11 11:22:37 +01:00
|
|
|
|
/*
|
2000-10-03 10:34:48 +02:00
|
|
|
|
* recording.c: Recording file handling
|
2000-03-11 11:22:37 +01:00
|
|
|
|
*
|
2000-04-24 09:46:05 +02:00
|
|
|
|
* See the main source file 'vdr.c' for copyright information and
|
2000-03-11 11:22:37 +01:00
|
|
|
|
* how to reach the author.
|
|
|
|
|
*
|
2002-04-01 11:04:47 +02:00
|
|
|
|
* $Id: recording.c 1.60 2002/04/01 10:51:23 kls Exp $
|
2000-03-11 11:22:37 +01:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "recording.h"
|
|
|
|
|
#include <errno.h>
|
2000-07-24 16:43:04 +02:00
|
|
|
|
#include <fcntl.h>
|
2000-03-11 11:22:37 +01:00
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
2000-07-28 12:45:18 +02:00
|
|
|
|
#include <sys/stat.h>
|
2000-07-24 16:43:04 +02:00
|
|
|
|
#include <unistd.h>
|
2002-01-26 15:25:37 +01:00
|
|
|
|
#include "i18n.h"
|
2000-03-11 11:22:37 +01:00
|
|
|
|
#include "interface.h"
|
|
|
|
|
#include "tools.h"
|
2000-07-29 15:21:42 +02:00
|
|
|
|
#include "videodir.h"
|
2000-03-11 11:22:37 +01:00
|
|
|
|
|
|
|
|
|
#define RECEXT ".rec"
|
|
|
|
|
#define DELEXT ".del"
|
2001-02-18 16:21:05 +01:00
|
|
|
|
#ifdef VFAT
|
|
|
|
|
#define DATAFORMAT "%4d-%02d-%02d.%02d.%02d.%02d.%02d" RECEXT
|
|
|
|
|
#else
|
2000-04-15 17:38:11 +02:00
|
|
|
|
#define DATAFORMAT "%4d-%02d-%02d.%02d:%02d.%02d.%02d" RECEXT
|
2001-02-18 16:21:05 +01:00
|
|
|
|
#endif
|
2000-03-11 11:22:37 +01:00
|
|
|
|
#define NAMEFORMAT "%s/%s/" DATAFORMAT
|
|
|
|
|
|
2001-02-11 11:04:41 +01:00
|
|
|
|
#define RESUMEFILESUFFIX "/resume.vdr"
|
2000-07-24 16:43:04 +02:00
|
|
|
|
#define SUMMARYFILESUFFIX "/summary.vdr"
|
2000-12-28 12:57:16 +01:00
|
|
|
|
#define MARKSFILESUFFIX "/marks.vdr"
|
2000-07-24 16:43:04 +02:00
|
|
|
|
|
2002-01-27 15:14:45 +01:00
|
|
|
|
#define FINDCMD "find %s -follow -type d -name '%s' 2> /dev/null"
|
|
|
|
|
|
2000-03-11 11:22:37 +01:00
|
|
|
|
#define MINDISKSPACE 1024 // MB
|
|
|
|
|
|
2001-02-04 12:36:32 +01:00
|
|
|
|
#define DELETEDLIFETIME 1 // hours after which a deleted recording will be actually removed
|
|
|
|
|
#define REMOVECHECKDELTA 3600 // seconds between checks for removing deleted files
|
2002-01-26 15:25:37 +01:00
|
|
|
|
#define DISKCHECKDELTA 100 // seconds between checks for free disk space
|
2001-06-02 10:47:40 +02:00
|
|
|
|
#define REMOVELATENCY 10 // seconds to wait until next check after removing a file
|
2001-02-04 12:36:32 +01:00
|
|
|
|
|
2002-02-03 15:55:04 +01:00
|
|
|
|
#define TIMERMACRO_TITLE "TITLE"
|
|
|
|
|
#define TIMERMACRO_EPISODE "EPISODE"
|
|
|
|
|
|
2001-02-04 12:36:32 +01:00
|
|
|
|
void RemoveDeletedRecordings(void)
|
|
|
|
|
{
|
|
|
|
|
static time_t LastRemoveCheck = 0;
|
|
|
|
|
if (time(NULL) - LastRemoveCheck > REMOVECHECKDELTA) {
|
2001-09-30 10:38:06 +02:00
|
|
|
|
// Make sure only one instance of VDR does this:
|
|
|
|
|
cLockFile LockFile(VideoDirectory);
|
|
|
|
|
if (!LockFile.Lock())
|
|
|
|
|
return;
|
2001-02-04 12:36:32 +01:00
|
|
|
|
// Remove the oldest file that has been "deleted":
|
|
|
|
|
cRecordings Recordings;
|
|
|
|
|
if (Recordings.Load(true)) {
|
|
|
|
|
cRecording *r = Recordings.First();
|
|
|
|
|
cRecording *r0 = r;
|
|
|
|
|
while (r) {
|
|
|
|
|
if (r->start < r0->start)
|
|
|
|
|
r0 = r;
|
|
|
|
|
r = Recordings.Next(r);
|
|
|
|
|
}
|
|
|
|
|
if (r0 && time(NULL) - r0->start > DELETEDLIFETIME * 60) {
|
|
|
|
|
r0->Remove();
|
2001-02-11 14:53:44 +01:00
|
|
|
|
RemoveEmptyVideoDirectories();
|
2001-02-04 12:36:32 +01:00
|
|
|
|
LastRemoveCheck += REMOVELATENCY;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
LastRemoveCheck = time(NULL);
|
|
|
|
|
}
|
|
|
|
|
}
|
2000-03-11 11:22:37 +01:00
|
|
|
|
|
2001-06-02 10:47:40 +02:00
|
|
|
|
void AssertFreeDiskSpace(int Priority)
|
2000-03-11 11:22:37 +01:00
|
|
|
|
{
|
|
|
|
|
// With every call to this function we try to actually remove
|
|
|
|
|
// a file, or mark a file for removal ("delete" it), so that
|
|
|
|
|
// it will get removed during the next call.
|
2000-04-15 17:38:11 +02:00
|
|
|
|
static time_t LastFreeDiskCheck = 0;
|
|
|
|
|
if (time(NULL) - LastFreeDiskCheck > DISKCHECKDELTA) {
|
2000-07-29 15:21:42 +02:00
|
|
|
|
if (!VideoFileSpaceAvailable(MINDISKSPACE)) {
|
2001-09-30 10:38:06 +02:00
|
|
|
|
// Make sure only one instance of VDR does this:
|
|
|
|
|
cLockFile LockFile(VideoDirectory);
|
|
|
|
|
if (!LockFile.Lock())
|
|
|
|
|
return;
|
2000-04-15 17:38:11 +02:00
|
|
|
|
// Remove the oldest file that has been "deleted":
|
2002-03-09 10:45:10 +01:00
|
|
|
|
isyslog(LOG_INFO, "low disk space while recording, trying to remove a deleted recording...");
|
2000-04-15 17:38:11 +02:00
|
|
|
|
cRecordings Recordings;
|
|
|
|
|
if (Recordings.Load(true)) {
|
|
|
|
|
cRecording *r = Recordings.First();
|
|
|
|
|
cRecording *r0 = r;
|
|
|
|
|
while (r) {
|
|
|
|
|
if (r->start < r0->start)
|
|
|
|
|
r0 = r;
|
|
|
|
|
r = Recordings.Next(r);
|
|
|
|
|
}
|
2000-11-18 16:26:50 +01:00
|
|
|
|
if (r0 && r0->Remove()) {
|
|
|
|
|
LastFreeDiskCheck += REMOVELATENCY;
|
2000-04-15 17:38:11 +02:00
|
|
|
|
return;
|
2000-11-18 16:26:50 +01:00
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
|
}
|
|
|
|
|
// No "deleted" files to remove, so let's see if we can delete a recording:
|
2002-03-09 10:45:10 +01:00
|
|
|
|
isyslog(LOG_INFO, "...no deleted recording found, trying to delete an old recording...");
|
2000-04-15 17:38:11 +02:00
|
|
|
|
if (Recordings.Load(false)) {
|
|
|
|
|
cRecording *r = Recordings.First();
|
|
|
|
|
cRecording *r0 = NULL;
|
|
|
|
|
while (r) {
|
2001-06-12 15:32:47 +02:00
|
|
|
|
if (r->lifetime < MAXLIFETIME) { // recordings with MAXLIFETIME live forever
|
2002-03-07 17:07:04 +01:00
|
|
|
|
if ((r->lifetime == 0 && Priority > r->priority) || // the recording has no guaranteed lifetime and the new recording has higher priority
|
2001-06-12 15:32:47 +02:00
|
|
|
|
(time(NULL) - r->start) / SECSINDAY > r->lifetime) { // the recording's guaranteed lifetime has expired
|
|
|
|
|
if (r0) {
|
|
|
|
|
if (r->priority < r0->priority || (r->priority == r0->priority && r->start < r0->start))
|
|
|
|
|
r0 = r; // in any case we delete the one with the lowest priority (or the older one in case of equal priorities)
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
r0 = r;
|
2000-04-15 17:38:11 +02:00
|
|
|
|
}
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
|
r = Recordings.Next(r);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
|
if (r0 && r0->Delete())
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// Unable to free disk space, but there's nothing we can do about that...
|
2002-03-09 10:45:10 +01:00
|
|
|
|
isyslog(LOG_INFO, "...no old recording found, giving up");
|
2002-02-15 22:24:30 +01:00
|
|
|
|
Interface->Confirm(tr("Low disk space!"), 30);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
2000-05-13 16:16:56 +02:00
|
|
|
|
LastFreeDiskCheck = time(NULL);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-02-11 11:04:41 +01:00
|
|
|
|
// --- cResumeFile ------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
cResumeFile::cResumeFile(const char *FileName)
|
|
|
|
|
{
|
|
|
|
|
fileName = new char[strlen(FileName) + strlen(RESUMEFILESUFFIX) + 1];
|
|
|
|
|
if (fileName) {
|
|
|
|
|
strcpy(fileName, FileName);
|
|
|
|
|
strcat(fileName, RESUMEFILESUFFIX);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
esyslog(LOG_ERR, "ERROR: can't allocate memory for resume file name");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cResumeFile::~cResumeFile()
|
|
|
|
|
{
|
|
|
|
|
delete fileName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int cResumeFile::Read(void)
|
|
|
|
|
{
|
|
|
|
|
int resume = -1;
|
|
|
|
|
if (fileName) {
|
|
|
|
|
int f = open(fileName, O_RDONLY);
|
|
|
|
|
if (f >= 0) {
|
2001-08-12 15:22:48 +02:00
|
|
|
|
if (safe_read(f, &resume, sizeof(resume)) != sizeof(resume)) {
|
2001-02-11 11:04:41 +01:00
|
|
|
|
resume = -1;
|
|
|
|
|
LOG_ERROR_STR(fileName);
|
|
|
|
|
}
|
|
|
|
|
close(f);
|
|
|
|
|
}
|
|
|
|
|
else if (errno != ENOENT)
|
|
|
|
|
LOG_ERROR_STR(fileName);
|
|
|
|
|
}
|
|
|
|
|
return resume;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cResumeFile::Save(int Index)
|
|
|
|
|
{
|
|
|
|
|
if (fileName) {
|
2001-06-02 10:47:40 +02:00
|
|
|
|
int f = open(fileName, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
|
2001-02-11 11:04:41 +01:00
|
|
|
|
if (f >= 0) {
|
2002-03-23 16:17:39 +01:00
|
|
|
|
if (safe_write(f, &Index, sizeof(Index)) < 0)
|
2001-02-11 11:04:41 +01:00
|
|
|
|
LOG_ERROR_STR(fileName);
|
|
|
|
|
close(f);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cResumeFile::Delete(void)
|
|
|
|
|
{
|
|
|
|
|
if (fileName) {
|
|
|
|
|
if (remove(fileName) < 0 && errno != ENOENT)
|
|
|
|
|
LOG_ERROR_STR(fileName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-11 11:22:37 +01:00
|
|
|
|
// --- cRecording ------------------------------------------------------------
|
|
|
|
|
|
2002-01-20 14:05:28 +01:00
|
|
|
|
#define RESUME_NOT_INITIALIZED (-2)
|
|
|
|
|
|
2001-06-16 10:36:13 +02:00
|
|
|
|
struct tCharExchange { char a; char b; };
|
|
|
|
|
tCharExchange CharExchange[] = {
|
2001-09-02 10:28:20 +02:00
|
|
|
|
{ '~', '/' },
|
2001-06-16 10:36:13 +02:00
|
|
|
|
{ ' ', '_' },
|
|
|
|
|
{ '\'', '\x01' },
|
|
|
|
|
{ '/', '\x02' },
|
|
|
|
|
{ 0, 0 }
|
|
|
|
|
};
|
|
|
|
|
|
2002-02-10 14:21:36 +01:00
|
|
|
|
static char *ExchangeChars(char *s, bool ToFileSystem)
|
2001-06-16 10:36:13 +02:00
|
|
|
|
{
|
2001-09-02 10:28:20 +02:00
|
|
|
|
char *p = s;
|
|
|
|
|
while (*p) {
|
2002-02-10 14:21:36 +01:00
|
|
|
|
#ifdef VFAT
|
|
|
|
|
// The VFAT file system can't handle all characters, so we
|
|
|
|
|
// have to take extra efforts to encode/decode them:
|
|
|
|
|
if (ToFileSystem) {
|
|
|
|
|
switch (*p) {
|
|
|
|
|
// characters that can be used "as is":
|
|
|
|
|
case '!':
|
|
|
|
|
case '@':
|
|
|
|
|
case '$':
|
|
|
|
|
case '%':
|
|
|
|
|
case '&':
|
|
|
|
|
case '(':
|
|
|
|
|
case ')':
|
|
|
|
|
case '+':
|
|
|
|
|
case ',':
|
|
|
|
|
case '-':
|
|
|
|
|
case '.':
|
|
|
|
|
case ';':
|
|
|
|
|
case '=':
|
|
|
|
|
case '0' ... '9':
|
|
|
|
|
case 'a' ... 'z':
|
2002-02-24 11:22:30 +01:00
|
|
|
|
case 'A' ... 'Z':
|
|
|
|
|
case '<EFBFBD>': case '<EFBFBD>':
|
|
|
|
|
case '<EFBFBD>': case '<EFBFBD>':
|
|
|
|
|
case '<EFBFBD>': case '<EFBFBD>':
|
|
|
|
|
case '<EFBFBD>':
|
|
|
|
|
break;
|
2002-02-10 14:21:36 +01:00
|
|
|
|
// characters that can be mapped to other characters:
|
|
|
|
|
case ' ': *p = '_'; break;
|
|
|
|
|
case '~': *p = '/'; break;
|
|
|
|
|
// characters that have to be encoded:
|
|
|
|
|
default: {
|
|
|
|
|
int l = p - s;
|
|
|
|
|
s = (char *)realloc(s, strlen(s) + 10);
|
|
|
|
|
p = s + l;
|
|
|
|
|
char buf[4];
|
|
|
|
|
sprintf(buf, "#%02X", (unsigned char)*p);
|
|
|
|
|
memmove(p + 2, p, strlen(p) + 1);
|
|
|
|
|
strncpy(p, buf, 3);
|
|
|
|
|
p += 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
switch (*p) {
|
|
|
|
|
// mapped characters:
|
|
|
|
|
case '_': *p = ' '; break;
|
|
|
|
|
case '/': *p = '~'; break;
|
|
|
|
|
// encodes characters:
|
|
|
|
|
case '#': {
|
|
|
|
|
if (strlen(p) > 2) {
|
|
|
|
|
char buf[3];
|
|
|
|
|
sprintf(buf, "%c%c", *(p + 1), *(p + 2));
|
|
|
|
|
unsigned char c = strtol(buf, NULL, 16);
|
|
|
|
|
*p = c;
|
|
|
|
|
memmove(p + 1, p + 3, strlen(p) - 2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
// backwards compatibility:
|
|
|
|
|
case '\x01': *p = '\''; break;
|
|
|
|
|
case '\x02': *p = '/'; break;
|
|
|
|
|
case '\x03': *p = ':'; break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#else
|
2001-09-02 10:28:20 +02:00
|
|
|
|
for (struct tCharExchange *ce = CharExchange; ce->a && ce->b; ce++) {
|
|
|
|
|
if (*p == (ToFileSystem ? ce->a : ce->b)) {
|
|
|
|
|
*p = ToFileSystem ? ce->b : ce->a;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-02-10 14:21:36 +01:00
|
|
|
|
#endif
|
2001-09-02 10:28:20 +02:00
|
|
|
|
p++;
|
|
|
|
|
}
|
2001-06-16 10:36:13 +02:00
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
2002-02-03 15:55:04 +01:00
|
|
|
|
cRecording::cRecording(cTimer *Timer, const char *Title, const char *Subtitle, const char *Summary)
|
2000-03-11 11:22:37 +01:00
|
|
|
|
{
|
2002-01-20 14:05:28 +01:00
|
|
|
|
resume = RESUME_NOT_INITIALIZED;
|
2000-04-24 09:35:29 +02:00
|
|
|
|
titleBuffer = NULL;
|
2001-10-07 11:00:35 +02:00
|
|
|
|
sortBuffer = NULL;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
fileName = NULL;
|
2002-02-03 15:55:04 +01:00
|
|
|
|
name = NULL;
|
|
|
|
|
// set up the actual name:
|
|
|
|
|
if (isempty(Title))
|
|
|
|
|
Title = Channels.GetChannelNameByNumber(Timer->channel);
|
|
|
|
|
if (isempty(Subtitle))
|
|
|
|
|
Subtitle = " ";
|
|
|
|
|
char *macroTITLE = strstr(Timer->file, TIMERMACRO_TITLE);
|
|
|
|
|
char *macroEPISODE = strstr(Timer->file, TIMERMACRO_EPISODE);
|
|
|
|
|
if (macroTITLE || macroEPISODE) {
|
2001-09-02 15:21:54 +02:00
|
|
|
|
name = strdup(Timer->file);
|
2002-02-03 15:55:04 +01:00
|
|
|
|
name = strreplace(name, TIMERMACRO_TITLE, Title);
|
|
|
|
|
name = strreplace(name, TIMERMACRO_EPISODE, Subtitle);
|
|
|
|
|
if (Timer->IsSingleEvent()) {
|
|
|
|
|
Timer->SetFile(name); // this was an instant recording, so let's set the actual data
|
|
|
|
|
Timers.Save();
|
|
|
|
|
}
|
2001-09-02 15:21:54 +02:00
|
|
|
|
}
|
2002-02-03 15:55:04 +01:00
|
|
|
|
else if (Timer->IsSingleEvent() || !Setup.UseSubtitle)
|
|
|
|
|
name = strdup(Timer->file);
|
|
|
|
|
else
|
|
|
|
|
asprintf(&name, "%s~%s", Timer->file, Subtitle);
|
2000-11-01 16:04:57 +01:00
|
|
|
|
// substitute characters that would cause problems in file names:
|
2001-06-16 10:36:13 +02:00
|
|
|
|
strreplace(name, '\n', ' ');
|
2000-03-11 11:22:37 +01:00
|
|
|
|
start = Timer->StartTime();
|
|
|
|
|
priority = Timer->priority;
|
|
|
|
|
lifetime = Timer->lifetime;
|
2001-09-02 15:21:54 +02:00
|
|
|
|
// handle summary:
|
|
|
|
|
summary = !isempty(Timer->summary) ? strdup(Timer->summary) : NULL;
|
|
|
|
|
if (!summary) {
|
|
|
|
|
if (isempty(Subtitle))
|
|
|
|
|
Subtitle = "";
|
|
|
|
|
if (isempty(Summary))
|
|
|
|
|
Summary = "";
|
|
|
|
|
if (*Subtitle || *Summary)
|
2002-03-23 11:37:28 +01:00
|
|
|
|
asprintf(&summary, "%s\n\n%s%s%s", Title, Subtitle, (*Subtitle && *Summary) ? "\n\n" : "", Summary);
|
2001-09-02 15:21:54 +02:00
|
|
|
|
}
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cRecording::cRecording(const char *FileName)
|
|
|
|
|
{
|
2002-01-20 14:05:28 +01:00
|
|
|
|
resume = RESUME_NOT_INITIALIZED;
|
2000-04-24 09:35:29 +02:00
|
|
|
|
titleBuffer = NULL;
|
2001-10-07 11:00:35 +02:00
|
|
|
|
sortBuffer = NULL;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
fileName = strdup(FileName);
|
2000-07-28 13:44:31 +02:00
|
|
|
|
FileName += strlen(VideoDirectory) + 1;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
char *p = strrchr(FileName, '/');
|
|
|
|
|
|
|
|
|
|
name = NULL;
|
2000-07-24 16:43:04 +02:00
|
|
|
|
summary = NULL;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
if (p) {
|
|
|
|
|
time_t now = time(NULL);
|
2001-10-19 13:22:24 +02:00
|
|
|
|
struct tm tm_r;
|
|
|
|
|
struct tm t = *localtime_r(&now, &tm_r); // this initializes the time zone in 't'
|
2002-04-01 11:04:47 +02:00
|
|
|
|
t.tm_isdst = -1; // makes sure mktime() will determine the correct DST setting
|
2000-04-15 17:38:11 +02:00
|
|
|
|
if (7 == sscanf(p + 1, DATAFORMAT, &t.tm_year, &t.tm_mon, &t.tm_mday, &t.tm_hour, &t.tm_min, &priority, &lifetime)) {
|
2000-03-11 11:22:37 +01:00
|
|
|
|
t.tm_year -= 1900;
|
|
|
|
|
t.tm_mon--;
|
|
|
|
|
t.tm_sec = 0;
|
|
|
|
|
start = mktime(&t);
|
|
|
|
|
name = new char[p - FileName + 1];
|
|
|
|
|
strncpy(name, FileName, p - FileName);
|
|
|
|
|
name[p - FileName] = 0;
|
2002-02-10 14:21:36 +01:00
|
|
|
|
name = ExchangeChars(name, false);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
2000-07-24 16:43:04 +02:00
|
|
|
|
// read an optional summary file:
|
|
|
|
|
char *SummaryFileName = NULL;
|
|
|
|
|
asprintf(&SummaryFileName, "%s%s", fileName, SUMMARYFILESUFFIX);
|
|
|
|
|
int f = open(SummaryFileName, O_RDONLY);
|
|
|
|
|
if (f >= 0) {
|
|
|
|
|
struct stat buf;
|
|
|
|
|
if (fstat(f, &buf) == 0) {
|
|
|
|
|
int size = buf.st_size;
|
|
|
|
|
summary = new char[size + 1]; // +1 for terminating 0
|
|
|
|
|
if (summary) {
|
2001-08-12 15:22:48 +02:00
|
|
|
|
int rbytes = safe_read(f, summary, size);
|
2000-07-24 16:43:04 +02:00
|
|
|
|
if (rbytes >= 0) {
|
|
|
|
|
summary[rbytes] = 0;
|
|
|
|
|
if (rbytes != size)
|
|
|
|
|
esyslog(LOG_ERR, "%s: expected %d bytes but read %d", SummaryFileName, size, rbytes);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
LOG_ERROR_STR(SummaryFileName);
|
|
|
|
|
delete summary;
|
|
|
|
|
summary = NULL;
|
|
|
|
|
}
|
2001-06-02 10:47:40 +02:00
|
|
|
|
|
2000-07-24 16:43:04 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
esyslog(LOG_ERR, "can't allocate %d byte of memory for summary file '%s'", size + 1, SummaryFileName);
|
|
|
|
|
close(f);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
LOG_ERROR_STR(SummaryFileName);
|
|
|
|
|
}
|
|
|
|
|
else if (errno != ENOENT)
|
|
|
|
|
LOG_ERROR_STR(SummaryFileName);
|
|
|
|
|
delete SummaryFileName;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cRecording::~cRecording()
|
|
|
|
|
{
|
2000-04-24 09:35:29 +02:00
|
|
|
|
delete titleBuffer;
|
2001-10-07 11:00:35 +02:00
|
|
|
|
delete sortBuffer;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
delete fileName;
|
|
|
|
|
delete name;
|
2000-07-24 16:43:04 +02:00
|
|
|
|
delete summary;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-10-07 11:00:35 +02:00
|
|
|
|
char *cRecording::StripEpisodeName(char *s)
|
|
|
|
|
{
|
|
|
|
|
char *t = s, *s1 = NULL, *s2 = NULL;
|
|
|
|
|
while (*t) {
|
|
|
|
|
if (*t == '/') {
|
|
|
|
|
if (s1) {
|
|
|
|
|
if (s2)
|
|
|
|
|
s1 = s2;
|
|
|
|
|
s2 = t;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
s1 = t;
|
|
|
|
|
}
|
|
|
|
|
t++;
|
|
|
|
|
}
|
|
|
|
|
if (s1 && s2)
|
|
|
|
|
memmove(s1 + 1, s2, t - s2 + 1);
|
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *cRecording::SortName(void)
|
|
|
|
|
{
|
|
|
|
|
if (!sortBuffer) {
|
|
|
|
|
char *s = StripEpisodeName(strdup(FileName() + strlen(VideoDirectory) + 1));
|
|
|
|
|
int l = strxfrm(NULL, s, 0);
|
|
|
|
|
sortBuffer = new char[l];
|
|
|
|
|
strxfrm(sortBuffer, s, l);
|
|
|
|
|
delete s;
|
|
|
|
|
}
|
|
|
|
|
return sortBuffer;
|
|
|
|
|
}
|
|
|
|
|
|
2002-01-20 14:05:28 +01:00
|
|
|
|
int cRecording::GetResume(void)
|
|
|
|
|
{
|
|
|
|
|
if (resume == RESUME_NOT_INITIALIZED) {
|
|
|
|
|
cResumeFile ResumeFile(FileName());
|
|
|
|
|
resume = ResumeFile.Read();
|
|
|
|
|
}
|
|
|
|
|
return resume;
|
|
|
|
|
}
|
|
|
|
|
|
2001-10-07 11:00:35 +02:00
|
|
|
|
bool cRecording::operator< (const cListObject &ListObject)
|
|
|
|
|
{
|
|
|
|
|
cRecording *r = (cRecording *)&ListObject;
|
|
|
|
|
return strcasecmp(SortName(), r->SortName()) < 0;
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-11 11:22:37 +01:00
|
|
|
|
const char *cRecording::FileName(void)
|
|
|
|
|
{
|
|
|
|
|
if (!fileName) {
|
2001-10-19 13:22:24 +02:00
|
|
|
|
struct tm tm_r;
|
|
|
|
|
struct tm *t = localtime_r(&start, &tm_r);
|
2002-02-10 14:21:36 +01:00
|
|
|
|
name = ExchangeChars(name, true);
|
2000-07-28 13:44:31 +02:00
|
|
|
|
asprintf(&fileName, NAMEFORMAT, VideoDirectory, name, t->tm_year + 1900, t->tm_mon + 1, t->tm_mday, t->tm_hour, t->tm_min, priority, lifetime);
|
2002-02-10 14:21:36 +01:00
|
|
|
|
name = ExchangeChars(name, false);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
return fileName;
|
|
|
|
|
}
|
|
|
|
|
|
2002-01-20 14:05:28 +01:00
|
|
|
|
const char *cRecording::Title(char Delimiter, bool NewIndicator, int Level)
|
2000-04-23 15:38:16 +02:00
|
|
|
|
{
|
2002-01-20 14:05:28 +01:00
|
|
|
|
char New = NewIndicator && IsNew() ? '*' : ' ';
|
2000-04-23 15:38:16 +02:00
|
|
|
|
delete titleBuffer;
|
|
|
|
|
titleBuffer = NULL;
|
2002-01-20 14:05:28 +01:00
|
|
|
|
if (Level < 0 || Level == HierarchyLevels()) {
|
|
|
|
|
struct tm tm_r;
|
|
|
|
|
struct tm *t = localtime_r(&start, &tm_r);
|
2002-02-10 15:41:23 +01:00
|
|
|
|
char *s;
|
2002-01-20 14:05:28 +01:00
|
|
|
|
if (Level > 0 && (s = strrchr(name, '~')) != NULL)
|
|
|
|
|
s++;
|
|
|
|
|
else
|
|
|
|
|
s = name;
|
|
|
|
|
asprintf(&titleBuffer, "%02d.%02d%c%02d:%02d%c%c%s",
|
|
|
|
|
t->tm_mday,
|
|
|
|
|
t->tm_mon + 1,
|
|
|
|
|
Delimiter,
|
|
|
|
|
t->tm_hour,
|
|
|
|
|
t->tm_min,
|
|
|
|
|
New,
|
|
|
|
|
Delimiter,
|
|
|
|
|
s);
|
2002-02-10 15:41:23 +01:00
|
|
|
|
// let's not display a trailing '~':
|
|
|
|
|
stripspace(titleBuffer);
|
|
|
|
|
s = &titleBuffer[strlen(titleBuffer) - 1];
|
|
|
|
|
if (*s == '~')
|
|
|
|
|
*s = 0;
|
2002-01-20 14:05:28 +01:00
|
|
|
|
}
|
|
|
|
|
else if (Level < HierarchyLevels()) {
|
|
|
|
|
const char *s = name;
|
|
|
|
|
const char *p = s;
|
|
|
|
|
while (*++s) {
|
|
|
|
|
if (*s == '~') {
|
|
|
|
|
if (Level--)
|
|
|
|
|
p = s + 1;
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
titleBuffer = new char[s - p + 3];
|
|
|
|
|
*titleBuffer = Delimiter;
|
|
|
|
|
*(titleBuffer + 1) = Delimiter;
|
|
|
|
|
strn0cpy(titleBuffer + 2, p, s - p + 1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
return "";
|
2000-04-23 15:38:16 +02:00
|
|
|
|
return titleBuffer;
|
|
|
|
|
}
|
|
|
|
|
|
2001-09-01 13:38:09 +02:00
|
|
|
|
const char *cRecording::PrefixFileName(char Prefix)
|
|
|
|
|
{
|
|
|
|
|
const char *p = PrefixVideoFileName(FileName(), Prefix);
|
|
|
|
|
if (p) {
|
|
|
|
|
delete fileName;
|
|
|
|
|
fileName = strdup(p);
|
|
|
|
|
return fileName;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2002-01-20 14:05:28 +01:00
|
|
|
|
int cRecording::HierarchyLevels(void)
|
|
|
|
|
{
|
|
|
|
|
const char *s = name;
|
|
|
|
|
int level = 0;
|
|
|
|
|
while (*++s) {
|
|
|
|
|
if (*s == '~')
|
|
|
|
|
level++;
|
|
|
|
|
}
|
|
|
|
|
return level;
|
|
|
|
|
}
|
|
|
|
|
|
2000-07-24 16:43:04 +02:00
|
|
|
|
bool cRecording::WriteSummary(void)
|
|
|
|
|
{
|
|
|
|
|
if (summary) {
|
|
|
|
|
char *SummaryFileName = NULL;
|
|
|
|
|
asprintf(&SummaryFileName, "%s%s", fileName, SUMMARYFILESUFFIX);
|
|
|
|
|
FILE *f = fopen(SummaryFileName, "w");
|
|
|
|
|
if (f) {
|
|
|
|
|
if (fputs(summary, f) < 0)
|
|
|
|
|
LOG_ERROR_STR(SummaryFileName);
|
|
|
|
|
fclose(f);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
LOG_ERROR_STR(SummaryFileName);
|
|
|
|
|
delete SummaryFileName;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-11 11:22:37 +01:00
|
|
|
|
bool cRecording::Delete(void)
|
|
|
|
|
{
|
|
|
|
|
bool result = true;
|
|
|
|
|
char *NewName = strdup(FileName());
|
|
|
|
|
char *ext = strrchr(NewName, '.');
|
|
|
|
|
if (strcmp(ext, RECEXT) == 0) {
|
|
|
|
|
strncpy(ext, DELEXT, strlen(ext));
|
2002-03-16 12:19:14 +01:00
|
|
|
|
if (access(NewName, F_OK) == 0) {
|
|
|
|
|
// the new name already exists, so let's remove that one first:
|
|
|
|
|
isyslog(LOG_INFO, "removing recording %s", NewName);
|
|
|
|
|
RemoveVideoFile(NewName);
|
|
|
|
|
}
|
2000-03-11 11:22:37 +01:00
|
|
|
|
isyslog(LOG_INFO, "deleting recording %s", FileName());
|
2000-07-29 15:21:42 +02:00
|
|
|
|
result = RenameVideoFile(FileName(), NewName);
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
delete NewName;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cRecording::Remove(void)
|
|
|
|
|
{
|
2002-01-20 16:47:09 +01:00
|
|
|
|
// let's do a final safety check here:
|
|
|
|
|
if (!endswith(FileName(), DELEXT)) {
|
|
|
|
|
esyslog(LOG_ERR, "attempt to remove recording %s", FileName());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2000-03-11 11:22:37 +01:00
|
|
|
|
isyslog(LOG_INFO, "removing recording %s", FileName());
|
2000-07-29 15:21:42 +02:00
|
|
|
|
return RemoveVideoFile(FileName());
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// --- cRecordings -----------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
bool cRecordings::Load(bool Deleted)
|
|
|
|
|
{
|
|
|
|
|
Clear();
|
2002-01-27 15:14:45 +01:00
|
|
|
|
bool result = false;
|
|
|
|
|
char *cmd = NULL;
|
|
|
|
|
asprintf(&cmd, FINDCMD, VideoDirectory, Deleted ? "*" DELEXT : "*" RECEXT);
|
|
|
|
|
FILE *p = popen(cmd, "r");
|
|
|
|
|
if (p) {
|
|
|
|
|
char *s;
|
|
|
|
|
while ((s = readline(p)) != NULL) {
|
|
|
|
|
cRecording *r = new cRecording(s);
|
|
|
|
|
if (r->Name())
|
|
|
|
|
Add(r);
|
|
|
|
|
else
|
|
|
|
|
delete r;
|
|
|
|
|
}
|
|
|
|
|
pclose(p);
|
|
|
|
|
Sort();
|
|
|
|
|
result = Count() > 0;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
}
|
2002-01-27 15:14:45 +01:00
|
|
|
|
else
|
|
|
|
|
Interface->Error("Error while opening pipe!");
|
|
|
|
|
delete cmd;
|
2000-03-11 11:22:37 +01:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2002-01-20 14:05:28 +01:00
|
|
|
|
cRecording *cRecordings::GetByName(const char *FileName)
|
|
|
|
|
{
|
|
|
|
|
for (cRecording *recording = First(); recording; recording = Next(recording)) {
|
|
|
|
|
if (strcmp(recording->FileName(), FileName) == 0)
|
|
|
|
|
return recording;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2000-12-28 12:57:16 +01:00
|
|
|
|
// --- cMark -----------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
char *cMark::buffer = NULL;
|
|
|
|
|
|
|
|
|
|
cMark::cMark(int Position, const char *Comment)
|
|
|
|
|
{
|
|
|
|
|
position = Position;
|
|
|
|
|
comment = Comment ? strdup(Comment) : NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cMark::~cMark()
|
|
|
|
|
{
|
|
|
|
|
delete comment;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *cMark::ToText(void)
|
|
|
|
|
{
|
|
|
|
|
delete buffer;
|
|
|
|
|
asprintf(&buffer, "%s%s%s\n", IndexToHMSF(position, true), comment ? " " : "", comment ? comment : "");
|
|
|
|
|
return buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cMark::Parse(const char *s)
|
|
|
|
|
{
|
|
|
|
|
delete comment;
|
|
|
|
|
comment = NULL;
|
|
|
|
|
position = HMSFToIndex(s);
|
|
|
|
|
const char *p = strchr(s, ' ');
|
|
|
|
|
if (p) {
|
|
|
|
|
p = skipspace(p);
|
|
|
|
|
if (*p) {
|
|
|
|
|
comment = strdup(p);
|
|
|
|
|
comment[strlen(comment) - 1] = 0; // strips trailing newline
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cMark::Save(FILE *f)
|
|
|
|
|
{
|
|
|
|
|
return fprintf(f, ToText()) > 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// --- cMarks ----------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
bool cMarks::Load(const char *RecordingFileName)
|
|
|
|
|
{
|
|
|
|
|
const char *MarksFile = AddDirectory(RecordingFileName, MARKSFILESUFFIX);
|
2001-01-01 14:48:03 +01:00
|
|
|
|
if (cConfig<cMark>::Load(MarksFile)) {
|
2000-12-28 12:57:16 +01:00
|
|
|
|
Sort();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cMarks::Sort(void)
|
|
|
|
|
{
|
|
|
|
|
for (cMark *m1 = First(); m1; m1 = Next(m1)) {
|
|
|
|
|
for (cMark *m2 = Next(m1); m2; m2 = Next(m2)) {
|
|
|
|
|
if (m2->position < m1->position) {
|
|
|
|
|
swap(m1->position, m2->position);
|
|
|
|
|
swap(m1->comment, m2->comment);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cMark *cMarks::Add(int Position)
|
|
|
|
|
{
|
|
|
|
|
cMark *m = Get(Position);
|
|
|
|
|
if (!m) {
|
2001-01-01 14:48:03 +01:00
|
|
|
|
cConfig<cMark>::Add(m = new cMark(Position));
|
2000-12-28 12:57:16 +01:00
|
|
|
|
Sort();
|
|
|
|
|
}
|
|
|
|
|
return m;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cMark *cMarks::Get(int Position)
|
|
|
|
|
{
|
|
|
|
|
for (cMark *mi = First(); mi; mi = Next(mi)) {
|
|
|
|
|
if (mi->position == Position)
|
|
|
|
|
return mi;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cMark *cMarks::GetPrev(int Position)
|
|
|
|
|
{
|
|
|
|
|
for (cMark *mi = Last(); mi; mi = Prev(mi)) {
|
|
|
|
|
if (mi->position < Position)
|
|
|
|
|
return mi;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cMark *cMarks::GetNext(int Position)
|
|
|
|
|
{
|
|
|
|
|
for (cMark *mi = First(); mi; mi = Next(mi)) {
|
|
|
|
|
if (mi->position > Position)
|
|
|
|
|
return mi;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2001-09-23 14:02:11 +02:00
|
|
|
|
// --- cRecordingUserCommand -------------------------------------------------
|
|
|
|
|
|
|
|
|
|
const char *cRecordingUserCommand::command = NULL;
|
|
|
|
|
|
|
|
|
|
void cRecordingUserCommand::InvokeCommand(const char *State, const char *RecordingFileName)
|
|
|
|
|
{
|
|
|
|
|
if (command) {
|
|
|
|
|
char *cmd;
|
2002-01-26 12:04:32 +01:00
|
|
|
|
asprintf(&cmd, "%s %s \"%s\"", command, State, strescape(RecordingFileName, "\"$"));
|
2001-09-23 14:02:11 +02:00
|
|
|
|
isyslog(LOG_INFO, "executing '%s'", cmd);
|
2001-10-20 10:39:27 +02:00
|
|
|
|
SystemExec(cmd);
|
2001-09-23 14:02:11 +02:00
|
|
|
|
delete cmd;
|
|
|
|
|
}
|
|
|
|
|
}
|