/* * dvbProvider.cpp * * Created on: 12.10.2012 * Author: savop */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; namespace upnp { class ChannelTitle : public cListObject { private: int channelNo; string channelName; string title; string pattern; void replaceAll(std::string& str, const std::string& from, const std::string& to) { if(from.empty()) return; size_t start_pos = 0; while((start_pos = str.find(from, start_pos)) != std::string::npos) { str.replace(start_pos, from.length(), to); start_pos += to.length(); // In case 'to' contains 'from', like replacing 'x' with 'yx' } } public: ChannelTitle() : channelNo(-1) , pattern("%chan% %- |title%") {} void Clear(){ channelNo = -1; channelName = string(); title = string(); } void SetChannelNumber(int no){ channelNo = no; } void SetChannelName(const string& name){ channelName = name; } void SetTitle(const string& t){ title = t; } string ToString(){ stringstream output; size_t startPos = 0, endPos = 0, delim = 0, opt = 0; string delimiter, options, var; while((startPos = pattern.find_first_of('%', startPos)) != string::npos){ // Copy the content from endPos to startPos output << pattern.substr(endPos + 1, startPos - endPos - (endPos > 0 ? 1 : 0 )); if((endPos = pattern.find_first_of('%', startPos+1)) != string::npos){ if((delim = pattern.find_first_of('|', startPos+1)) != string::npos && delim < endPos){ delimiter = pattern.substr(startPos + 1, delim - startPos - 1); startPos = delim; } if((opt = pattern.find_first_of(';', startPos+1)) != string::npos && opt < endPos){ options = pattern.substr(opt + 1, endPos - opt - 1); } else { opt = endPos; } var = pattern.substr(startPos + 1, opt - startPos - 1); if (var.compare("no") == 0 && channelNo > 0){ int width = atoi(options.c_str()); output << delimiter; if(width > 0) output << setfill('0') << setw(width); output << channelNo; } else if(var.compare("chan") == 0 && !channelName.empty()){ output << delimiter << channelName; } else if(var.compare("title") == 0 && !title.empty()){ output << delimiter << title; } } startPos = endPos + 1; } return output.str(); } bool Parse(const char* str){ if(!str || strcmp(str, "") == 0){ pattern = "%chan% - %title%"; } else { pattern = str; } return true; } }; class DVBProfiler : public cMediaProfiler { public: DVBProfiler(){ stringstream file; file << cPlugin::ConfigDirectory(PLUGIN_NAME_I18N) << "/channelTitle.conf"; channelTitleConfig.Load(file.str().c_str(), true); } virtual bool CanHandleSchema(const string& schema){ #if VDRVERSNUM > 10704 if( schema.find("vdr",0) == 0 || schema.find("rec",0) == 0 ){ #else if( schema.find("vdr",0) == 0 ) { #endif return true; } else { return false; } } virtual bool GetMetadata(const string& uri, cMetadata& metadata, cUPnPResourceProvider*){ if (uri.find("vdr",0) == 0){ return GetChannelMetadata(uri, metadata); } #if VDRVERSNUM > 10704 else if (uri.find("rec",0) == 0){ return GetRecordingMetadata(uri, metadata); } #endif else { return false; } } ::cConfig channelTitleConfig; private: #if VDRVERSNUM > 10704 bool GetRecordingMetadata(const string& u, cMetadata& metadata){ string videoDir = string(VideoDirectory), uri = u.substr(6), recStr = videoDir + "/" + uri; cRecording* recording = Recordings.GetByName(recStr.c_str()); if(!recording) return false; // Sorry, currently unsupported. if(recording->IsPesRecording()) return false; char* fileBuf = strdup(recording->Name()); fileBuf = ExchangeChars(fileBuf, true); string fs = fileBuf; free(fileBuf); int pos = 0; if((pos = fs.find_last_of('/')) != string::npos){ fs = fs.substr(0,pos+1); } else { fs = ""; } fs = u.substr(0,6) + fs; const cRecordingInfo* info = recording->Info(); metadata.SetObjectIDByUri(u); metadata.SetParentIDByUri(fs); metadata.SetProperty(cMetadata::Property(property::object::KEY_RESTRICTED, true)); metadata.SetProperty(cMetadata::Property(property::object::KEY_TITLE, string(info->Title()?info->Title():recording->Title()))); metadata.SetProperty(cMetadata::Property(property::object::KEY_DESCRIPTION, string(info->ShortText()?info->ShortText():""))); metadata.SetProperty(cMetadata::Property(property::object::KEY_LONG_DESCRIPTION, string(info->Description()?info->Description():""))); boost::posix_time::ptime date = boost::posix_time::from_time_t(info->GetEvent()->StartTime()); metadata.SetProperty(cMetadata::Property(property::object::KEY_DATE, boost::gregorian::to_iso_extended_string(date.date()))); cMetadata::Resource resource; char filename[1024]; strncpy(filename, recording->FileName(), 1024); char* pFileNumber = filename + strlen(filename); sprintf(pFileNumber, "/%05d.ts", 1); FILE *fd = fopen(filename, "r"); if(!fd) return false; int pmtV, patV, pid; unsigned char buf[TS_SIZE]; cPatPmtParser parser; while(fread(buf, 1, TS_SIZE, fd) == TS_SIZE){ if(buf[0] == TS_SYNC_BYTE){ pid = TsPid(buf); if(pid == 0){ parser.ParsePat(buf, TS_SIZE); } else #if VDRVERSNUM < 10733 if (pid == parser.PmtPid()) #else if (parser.IsPmtPid(pid)) #endif { parser.ParsePmt(buf, TS_SIZE); if(parser.GetVersions(patV, pmtV)) break; } } } fclose(fd); int i = 1; size_t size = 0; sprintf(pFileNumber, "/%05d.ts", i); while(access(filename, F_OK) == 0){ fd = fopen(filename, "r"); fseek(fd, 0, SEEK_END); size += ftell(fd); sprintf(pFileNumber, "/%05d.ts", ++i); fclose(fd); } DLNA4thField fourthfield; string contentType, upnpclass; if(parser.Vtype() != 0){ switch (parser.Vtype()) { case 0x02: fourthfield = DLNA4thField("MPEG_TS_SD_EU_ISO", DLNA_OPERATION_RANGE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_BYTE_BASED_SEEK | DLNA_FLAG_BACKGROUND_TRANSFER | DLNA_FLAG_CONNECTION_STALLING | DLNA_FLAG_VERSION_1_5 ); break; case 0x1B: fourthfield = DLNA4thField("AVC_TS_HD_EU_ISO", DLNA_OPERATION_RANGE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_BYTE_BASED_SEEK | DLNA_FLAG_BACKGROUND_TRANSFER | DLNA_FLAG_CONNECTION_STALLING | DLNA_FLAG_VERSION_1_5 ); break; default: return false; } contentType = "video/mpeg"; upnpclass = "object.item.videoItem.videoBroadcast"; } else { int Atype = 0; for(int i = 0; (Atype = parser.Atype(i)) != 0; ++i){ switch(Atype){ case 0x03: case 0x04: #ifdef DLNA_STRICT fourthfield = DLNA4thField("MP2_MPS", DLNA_OPERATION_RANGE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_BYTE_BASED_SEEK | DLNA_FLAG_BACKGROUND_TRANSFER | DLNA_FLAG_CONNECTION_STALLING | DLNA_FLAG_VERSION_1_5 ); contentType = "audio/mpeg"; upnpclass = "object.item.audioItem.audioBroadcast"; #else fourthfield = DLNA4thField("MPEG_TS_SD_EU_ISO", DLNA_OPERATION_RANGE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_BYTE_BASED_SEEK | DLNA_FLAG_BACKGROUND_TRANSFER | DLNA_FLAG_CONNECTION_STALLING | DLNA_FLAG_VERSION_1_5 ); contentType = "video/mpeg"; upnpclass = "object.item.videoItem.videoBroadcast"; #endif goto validProfile; default: break; } } // No compatible audio codec found. return false; } // Found a valid profile validProfile: metadata.SetProperty(cMetadata::Property(property::object::KEY_CLASS, upnpclass)); resource.SetProtocolInfo(ProtocolInfo(contentType, fourthfield).ToString()); resource.SetSize(size); resource.SetResourceUri(u); int seconds = 0; const cEvent* event = info->GetEvent(); if(event && event->Duration() > 0){ seconds = event->Duration(); } #if VDRVERSNUM > 10723 else if(recording->LengthInSeconds() > 0){ seconds = recording->LengthInSeconds(); } #endif if(seconds){ boost::posix_time::time_duration duration = boost::posix_time::seconds(seconds); resource.SetDuration(boost::posix_time::to_simple_string(duration)); } metadata.AddResource(resource); return true; } #endif bool GetChannelMetadata(const string& uri, cMetadata& metadata){ tChannelID channelID = tChannelID::FromString(uri.substr(uri.find_last_of('/')+1).c_str()); if(!channelID.Valid()) return false; cChannel* channel = Channels.GetByChannelID(channelID); if(channel == NULL) return false; // First of all we can set the default information. string parentUri = uri.substr(0, uri.find_last_of('/')+1); metadata.SetObjectIDByUri(uri); metadata.SetParentIDByUri(parentUri); // TODO: implement check for radio stations (this is ...audioItem.audioBroadcast) metadata.SetProperty(cMetadata::Property(property::object::KEY_CLASS, string("object.item.videoItem.videoBroadcast"))); metadata.SetProperty(cMetadata::Property(property::object::KEY_RESTRICTED, true)); metadata.SetProperty(cMetadata::Property(property::object::KEY_CHANNEL_NAME, string(channel->Name()))); metadata.SetProperty(cMetadata::Property(property::object::KEY_CHANNEL_NR, (long int)channel->Number())); // Now, we try to get the present event of the schedule { cSchedulesLock lock; const cSchedules* schedules = cSchedules::Schedules(lock); const cSchedule* schedule = (schedules) ? schedules->GetSchedule(channelID) : NULL; const cEvent* event = (schedule) ? schedule->GetPresentEvent() : NULL; if(event){ boost::posix_time::ptime startTime, endTime; startTime = boost::posix_time::from_time_t(event->StartTime()); endTime = boost::posix_time::from_time_t(event->EndTime()); metadata.SetProperty(cMetadata::Property(property::object::KEY_DATE, boost::gregorian::to_iso_extended_string(startTime.date()))); metadata.SetProperty(cMetadata::Property(property::object::KEY_SCHEDULED_START, boost::posix_time::to_iso_extended_string(startTime))); metadata.SetProperty(cMetadata::Property(property::object::KEY_SCHEDULED_END, boost::posix_time::to_iso_extended_string(endTime))); metadata.SetProperty(cMetadata::Property(property::object::KEY_DESCRIPTION, string(event->ShortText()?event->ShortText():""))); metadata.SetProperty(cMetadata::Property(property::object::KEY_LONG_DESCRIPTION, string(event->Description()?event->Description():""))); } ChannelTitle* titleConfig = channelTitleConfig.First(); if(titleConfig){ titleConfig->Clear(); if(event) titleConfig->SetTitle(event->Title()); titleConfig->SetChannelName(channel->Name()); titleConfig->SetChannelNumber(channel->Number()); metadata.SetProperty(cMetadata::Property(property::object::KEY_TITLE, titleConfig->ToString())); } else { stringstream ss; ss << channel->Name(); if(event) ss << " - " << event->Title(); metadata.SetProperty(cMetadata::Property(property::object::KEY_TITLE, ss.str())); } cMetadata::Resource resource; resource.SetResourceUri(uri); // TODO: implement check for radio stations DLNA4thField fourthfield; switch (channel->Vtype()) { case 0x02: fourthfield = DLNA4thField("MPEG_TS_SD_EU_ISO", DLNA_OPERATION_NONE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_SN_INCREASE | DLNA_FLAG_VERSION_1_5 ); break; case 0x1B: fourthfield = DLNA4thField("AVC_TS_HD_EU_ISO", DLNA_OPERATION_NONE, DLNA_PLAYSPEEDS_NONE, DLNA_CONVERSION_NONE, DLNA_FLAG_STREAMING_TRANSFER | DLNA_FLAG_SN_INCREASE | DLNA_FLAG_VERSION_1_5 ); break; default: return false; } resource.SetProtocolInfo(ProtocolInfo("video/mpeg", fourthfield).ToString()); if(event){ boost::posix_time::time_duration duration = boost::posix_time::seconds(event->Duration()); resource.SetDuration(boost::posix_time::to_simple_string(duration)); } metadata.AddResource(resource); } stringstream ss; cMetadata::Resource thumbnail; ss.str(string()); ss << "channelIcons/" << channel->Name() << ".jpg"; stringstream filename, uriStrm; filename << cMediaServer::GetInstance()->GetWebserver().GetThumbnailDir() << ss.str(); uriStrm << "thumb://" << ss.str(); struct stat fileStat; if(stat(filename.str().c_str(), &fileStat) == 0){ thumbnail.SetResourceUri(uriStrm.str()); thumbnail.SetProtocolInfo(ProtocolInfo("image/jpeg", DLNA4thField("JPEG_TN")).ToString()); thumbnail.SetSize(fileStat.st_size); metadata.AddResource(thumbnail); } return true; } }; UPNP_REGISTER_MEDIA_PLUGIN(DVBProfiler); }