mirror of
https://github.com/iptv-org/epg
synced 2026-05-07 09:57:06 -04:00
Update tvguide.com guide.
Test: ```sh npm test --- tvguide.com > test > run-script-os tvguide.com > test:win32 > SET "TZ=Pacific/Nauru" && npx jest --runInBand tvguide.com PASS sites/tvguide.com/tvguide.com.test.js √ can generate valid url (4 ms) √ can parse response (5 ms) √ can handle empty guide (1 ms) Test Suites: 1 passed, 1 total Tests: 3 passed, 3 total Snapshots: 0 total Time: 0.567 s, estimated 1 s Ran all test suites matching /tvguide.com/i. ``` Grab: ```sh npm run grab --- --site=tvguide.com > grab > npx tsx scripts/commands/epg/grab.ts --site=tvguide.com starting... config: output: guide.xml maxConnections: 1 gzip: false site: tvguide.com loading channels... found 1 channel(s) run #1: [1/2] tvguide.com (es) - UnivisionEast.us - Jan 12, 2025 (33 programs) [2/2] tvguide.com (es) - UnivisionEast.us - Jan 13, 2025 (25 programs) saving to "guide.xml"... done in 00h 00m 05s ``` Signed-off-by: Toha <tohenk@yahoo.com>
This commit is contained in:
@@ -2,97 +2,113 @@ const axios = require('axios')
|
||||
const dayjs = require('dayjs')
|
||||
const utc = require('dayjs/plugin/utc')
|
||||
const timezone = require('dayjs/plugin/timezone')
|
||||
const doFetch = require('@ntlab/sfetch')
|
||||
const debug = require('debug')('site:tvguide.com')
|
||||
|
||||
dayjs.extend(utc)
|
||||
dayjs.extend(timezone)
|
||||
|
||||
doFetch.setDebugger(debug).setCheckResult(false)
|
||||
|
||||
const providerId = '9100001138'
|
||||
const maxDuration = 240
|
||||
const segments = 1440 / maxDuration
|
||||
|
||||
module.exports = {
|
||||
site: 'tvguide.com',
|
||||
days: 2,
|
||||
url: function ({ date, channel }) {
|
||||
const [providerId, channelSourceIds] = channel.site_id.split('#')
|
||||
const url = `https://internal-prod.apigee.fandom.net/v1/xapi/tvschedules/tvguide/${providerId}/web?start=${date
|
||||
.startOf('d')
|
||||
.unix()}&duration=1440&channelSourceIds=${channelSourceIds}`
|
||||
|
||||
return url
|
||||
request: {
|
||||
cache: {
|
||||
ttl: 24 * 60 * 60 * 1000 // 1 day
|
||||
}
|
||||
},
|
||||
async parser({ content }) {
|
||||
async url({ date, segment = 1 }) {
|
||||
const params = []
|
||||
if (module.exports.apiKey === undefined) {
|
||||
module.exports.apiKey = await module.exports.fetchApiKey()
|
||||
debug('Got api key', module.exports.apiKey)
|
||||
}
|
||||
if (date) {
|
||||
if (segment > 1) {
|
||||
date = date.add((segment - 1) * maxDuration, 'm')
|
||||
}
|
||||
params.push(`start=${date.unix()}`, `duration=${maxDuration}`)
|
||||
}
|
||||
params.push(`apiKey=${module.exports.apiKey}`)
|
||||
|
||||
return date ?
|
||||
`https://backend.tvguide.com/tvschedules/tvguide/${providerId}/web?${params.join('&')}` :
|
||||
`https://backend.tvguide.com/tvschedules/tvguide/serviceprovider/${providerId}/sources/web?${params.join('&')}`
|
||||
},
|
||||
async parser({ content, date, channel }) {
|
||||
const programs = []
|
||||
const items = parseItems(content)
|
||||
for (let item of items) {
|
||||
const details = await loadProgramDetails(item)
|
||||
programs.push({
|
||||
title: item.title,
|
||||
sub_title: details.episodeTitle,
|
||||
description: details.description,
|
||||
season: details.seasonNumber,
|
||||
episode: details.episodeNumber,
|
||||
rating: parseRating(item),
|
||||
categories: parseCategories(details),
|
||||
start: parseTime(item.startTime),
|
||||
stop: parseTime(item.endTime)
|
||||
const f = data => {
|
||||
const result = []
|
||||
if (typeof data === 'string') {
|
||||
data = JSON.parse(data)
|
||||
}
|
||||
if (data && Array.isArray(data?.data?.items)) {
|
||||
data.data.items
|
||||
.filter(i => i.channel.sourceId.toString() === channel.site_id)
|
||||
.forEach(i => {
|
||||
result.push(...i.programSchedules.map(p => {
|
||||
return { i: p, url: p.programDetails }
|
||||
}))
|
||||
})
|
||||
}
|
||||
|
||||
return result
|
||||
}
|
||||
const queues = f(content)
|
||||
if (queues.length) {
|
||||
const parts = []
|
||||
for (let i = 2; i <= segments; i++) {
|
||||
parts.push(await module.exports.url({ date, segment: i }))
|
||||
}
|
||||
await doFetch(parts, (url, res) => {
|
||||
queues.push(...f(res))
|
||||
})
|
||||
await doFetch(queues, (queue, res) => {
|
||||
const item = res?.data?.item ? res.data.item : queue.i
|
||||
programs.push({
|
||||
title: item.title ? item.title : queue.i.title,
|
||||
sub_title: item.episodeNumber ? item.episodeTitle : null,
|
||||
description: item.description,
|
||||
season: item.seasonNumber,
|
||||
episode: item.episodeNumber,
|
||||
rating: item.rating ? { system: 'MPA', value: item.rating } : null,
|
||||
categories: Array.isArray(item.genres) ? item.genres.map(g => g.name) : null,
|
||||
start: dayjs.unix(item.startTime ? item.startTime : queue.i.startTime),
|
||||
stop: dayjs.unix(item.endTime ? item.endTime : queue.i.endTime)
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
return programs
|
||||
},
|
||||
async channels() {
|
||||
const providers = [9100001138]
|
||||
const channels = []
|
||||
const data = await axios
|
||||
.get(await this.url({}))
|
||||
.then(r => r.data)
|
||||
.catch(console.error)
|
||||
|
||||
let channels = []
|
||||
for (let providerId of providers) {
|
||||
const data = await axios
|
||||
.get(
|
||||
`https://internal-prod.apigee.fandom.net/v1/xapi/tvschedules/tvguide/serviceprovider/${providerId}/sources/web`
|
||||
)
|
||||
.then(r => r.data)
|
||||
.catch(console.log)
|
||||
|
||||
data.data.items.forEach(item => {
|
||||
channels.push({
|
||||
lang: 'en',
|
||||
site_id: `${providerId}#${item.sourceId}`,
|
||||
name: item.fullName
|
||||
})
|
||||
data.data.items.forEach(item => {
|
||||
channels.push({
|
||||
lang: 'en',
|
||||
site_id: item.sourceId,
|
||||
name: item.fullName.replace(/Channel|Schedule/g, '').trim()
|
||||
})
|
||||
}
|
||||
})
|
||||
|
||||
return channels
|
||||
},
|
||||
async fetchApiKey() {
|
||||
const data = await axios
|
||||
.get('https://www.tvguide.com/listings/')
|
||||
.then(r => r.data)
|
||||
.catch(console.error)
|
||||
|
||||
return data ? data.match(/apiKey=([a-zA-Z0-9]+)&/)[1] : null
|
||||
}
|
||||
}
|
||||
|
||||
function parseRating(item) {
|
||||
return item.rating ? { system: 'MPA', value: item.rating } : null
|
||||
}
|
||||
|
||||
function parseCategories(details) {
|
||||
return Array.isArray(details.genres) ? details.genres.map(g => g.name) : []
|
||||
}
|
||||
|
||||
function parseTime(timestamp) {
|
||||
return dayjs.unix(timestamp)
|
||||
}
|
||||
|
||||
function parseItems(content) {
|
||||
const data = JSON.parse(content)
|
||||
if (!data.data || !Array.isArray(data.data.items) || !data.data.items.length) return []
|
||||
|
||||
return data.data.items[0].programSchedules
|
||||
}
|
||||
|
||||
async function loadProgramDetails(item) {
|
||||
item.programDetails = item.programDetails.replace(
|
||||
'player1-backend-prod-internal.apigee.net',
|
||||
'internal-prod.apigee.fandom.net'
|
||||
)
|
||||
const data = await axios
|
||||
.get(item.programDetails)
|
||||
.then(r => r.data)
|
||||
.catch(err => {
|
||||
console.log(err.message)
|
||||
})
|
||||
if (!data || !data.data || !data.data.item) return {}
|
||||
|
||||
return data.data.item
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user