diff --git a/sites/cosmote.gr/cosmote.gr.config.js b/sites/cosmote.gr/cosmote.gr.config.js
index b143ee8a..eb77c5c8 100644
--- a/sites/cosmote.gr/cosmote.gr.config.js
+++ b/sites/cosmote.gr/cosmote.gr.config.js
@@ -1,14 +1,14 @@
-const jsdom = require('jsdom')
-const { JSDOM } = jsdom
+const cheerio = require('cheerio')
const dayjs = require('dayjs')
const utc = require('dayjs/plugin/utc')
+const timezone = require('dayjs/plugin/timezone')
const customParseFormat = require('dayjs/plugin/customParseFormat')
dayjs.extend(utc)
+dayjs.extend(timezone)
dayjs.extend(customParseFormat)
module.exports = {
- delay: 7000,
site: 'cosmote.gr',
url: function ({ date, channel }) {
return `https://www.cosmote.gr/cosmotetv/residential/program/epg/programchannel?p_p_id=channelprogram_WAR_OTETVportlet&p_p_lifecycle=0&_channelprogram_WAR_OTETVportlet_platform=IPTV&_channelprogram_WAR_OTETVportlet_date=${date.format(
@@ -16,37 +16,30 @@ module.exports = {
)}&_channelprogram_WAR_OTETVportlet_articleTitleUrl=${channel.site_id}`
},
logo: function ({ content }) {
- const dom = new JSDOM(content)
- const img = dom.window.document.querySelector('img.channel_program-banner')
+ const $ = cheerio.load(content)
+ const imgSrc = $('img.channel_program-banner').attr('src')
- return img ? 'https://www.cosmote.gr' + img.src : null
+ return imgSrc ? `https://www.cosmote.gr${imgSrc}` : null
},
parser: function ({ date, content }) {
- const dom = new JSDOM(content)
- const items = dom.window.document.querySelectorAll(
- '#_channelprogram_WAR_OTETVportlet_programs > tr.visible-xs'
- )
let programs = []
+ const items = parseItems(content)
items.forEach(item => {
- const title = (item.querySelector('a') || { textContent: '' }).textContent
- const meta = (item.querySelector('td') || { innerHTML: '' }).innerHTML
- const startTime = (item.querySelector('.start-time') || { textContent: '' }).textContent
- const endTime = (item.querySelector('.end-time') || { textContent: '' }).textContent
- const category = meta.match(/\| (.+)
/)[1]
- const start = dayjs
- .utc(startTime, 'HH:mm')
- .set('D', date.get('D'))
- .set('M', date.get('M'))
- .set('y', date.get('y'))
- const stop = dayjs
- .utc(endTime, 'HH:mm')
- .set('D', date.get('D'))
- .set('M', date.get('M'))
- .set('y', date.get('y'))
-
+ const prev = programs[programs.length - 1]
+ const $item = cheerio.load(item)
+ let start = parseStart($item, date)
+ if (prev && start.isBefore(prev.start)) {
+ start = start.add(1, 'd')
+ date = date.add(1, 'd')
+ }
+ let stop = parseStop($item, date)
+ if (stop.isBefore(start)) {
+ stop = stop.add(1, 'd')
+ date = date.add(1, 'd')
+ }
programs.push({
- title,
- category,
+ title: parseTitle($item),
+ category: parseCategory($item),
start,
stop
})
@@ -55,3 +48,39 @@ module.exports = {
return programs
}
}
+
+function parseTitle($item) {
+ return $item('.channel_program-table--program > a').text()
+}
+
+function parseCategory($item) {
+ const typeString = $item('.channel_program-table--program_type')
+ .children()
+ .remove()
+ .end()
+ .text()
+ .trim()
+ const [_, category] = typeString.match(/\| (.*)/) || [null, null]
+
+ return category
+}
+
+function parseStart($item, date) {
+ const timeString = $item('span.start-time').text()
+ const dateString = `${date.format('YYYY-MM-DD')} ${timeString}`
+
+ return dayjs.tz(dateString, 'YYYY-MM-DD HH:mm', 'Europe/Athens')
+}
+
+function parseStop($item, date) {
+ const timeString = $item('span.end-time').text()
+ const dateString = `${date.format('YYYY-MM-DD')} ${timeString}`
+
+ return dayjs.tz(dateString, 'YYYY-MM-DD HH:mm', 'Europe/Athens')
+}
+
+function parseItems(content) {
+ const $ = cheerio.load(content)
+
+ return $('#_channelprogram_WAR_OTETVportlet_programs > tr.visible-xs').toArray()
+}
diff --git a/sites/cosmote.gr/cosmote.gr.test.js b/sites/cosmote.gr/cosmote.gr.test.js
new file mode 100644
index 00000000..bcbe68f2
--- /dev/null
+++ b/sites/cosmote.gr/cosmote.gr.test.js
@@ -0,0 +1,68 @@
+// npx epg-grabber --config=sites/cosmote.gr/cosmote.gr.config.js --channels=sites/cosmote.gr/cosmote.gr_gr.channels.xml --output=.gh-pages/guides/gr/cosmote.gr.epg.xml --days=2
+
+const { parser, url, logo } = require('./cosmote.gr.config.js')
+const dayjs = require('dayjs')
+const utc = require('dayjs/plugin/utc')
+const timezone = require('dayjs/plugin/timezone')
+const customParseFormat = require('dayjs/plugin/customParseFormat')
+
+dayjs.extend(utc)
+dayjs.extend(timezone)
+dayjs.extend(customParseFormat)
+
+const date = dayjs.utc('2021-11-25', 'YYYY-MM-DD').startOf('d')
+const channel = {
+ site_id: '4e',
+ xmltv_id: '4E.gr'
+}
+const content = `
`
+
+it('can generate valid url', () => {
+ expect(url({ channel, date })).toBe(
+ 'https://www.cosmote.gr/cosmotetv/residential/program/epg/programchannel?p_p_id=channelprogram_WAR_OTETVportlet&p_p_lifecycle=0&_channelprogram_WAR_OTETVportlet_platform=IPTV&_channelprogram_WAR_OTETVportlet_date=25-11-2021&_channelprogram_WAR_OTETVportlet_articleTitleUrl=4e'
+ )
+})
+
+it('can generate valid logo url', () => {
+ expect(logo({ content })).toBe(
+ 'https://www.cosmote.gr/portal/image/journal/article?img_id=56014332&t=1544180299920'
+ )
+})
+
+it('can parse response', () => {
+ const result = parser({ content, channel, date }).map(p => {
+ p.start = p.start.toJSON()
+ p.stop = p.stop.toJSON()
+ return p
+ })
+
+ expect(result).toMatchObject([
+ {
+ start: '2021-11-24T23:15:00.000Z',
+ stop: '2021-11-24T23:50:00.000Z',
+ title: `Μαθαίνω να Είμαι Γονιός`,
+ category: 'Ντοκιμαντέρ'
+ },
+ {
+ start: '2021-11-25T21:00:00.000Z',
+ stop: '2021-11-25T21:45:00.000Z',
+ title: `Μικρό Απόδειπνο`,
+ category: 'Special'
+ },
+ {
+ start: '2021-11-25T21:45:00.000Z',
+ stop: '2021-11-26T01:00:00.000Z',
+ title: `Πανηγυρική Αρχιερατική Θεία Λειτουργία`,
+ category: 'Special'
+ }
+ ])
+})
+
+it('can handle empty guide', () => {
+ const result = parser({
+ date,
+ channel,
+ content: ``
+ })
+ expect(result).toMatchObject([])
+})