Files
epg/sites/directv.com/directv.com.config.js
2026-04-06 23:20:33 +02:00

171 lines
5.4 KiB
JavaScript

const axios = require('axios')
const dayjs = require('dayjs')
const utc = require('dayjs/plugin/utc')
dayjs.extend(utc)
let token = null
async function fetchToken() {
if (token) return token
try {
token = await axios
.post('https://api.cld.dtvce.com/authn-tokengo/v3/v2/tokens?client_id=DTVE_DFW_WEB_Chrome_G', null, {
headers: {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/145.0.0.0 Safari/537.36',
'cache-control': 'no-cache',
'origin': 'https://www.directv.com',
'pragma': 'no-cache',
'priority': 'u=1, i',
'referer': 'https://www.directv.com/',
'sec-ch-ua': '"Chromium";v="146", "Not-A.Brand";v="24", "Brave";v="146"',
'sec-ch-ua-mobile': '?0',
'sec-ch-ua-platform': '"Windows"',
'sec-fetch-dest': 'empty',
'sec-fetch-mode': 'cors',
'sec-fetch-site': 'cross-site',
'sec-gpc': '1'
}
})
.then(r => r.data)
.then(d => d.access_token)
return token
} catch (error) {
console.error('Error fetching token (potential geo-block or API issue):', error)
return null
}
}
module.exports = {
site: 'directv.com',
days: 2,
request: {
cache: {
ttl: 60 * 60 * 1000 // 1 hour
},
async headers() {
await fetchToken()
return {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/145.0.0.0 Safari/537.36',
Authorization: `Bearer ${token}`,
'cache-control': 'no-cache',
'origin': 'https://www.directv.com',
'pragma': 'no-cache',
'priority': 'u=1, i',
'referer': 'https://www.directv.com/',
'sec-ch-ua': '"Chromium";v="146", "Not-A.Brand";v="24", "Brave";v="146"',
'sec-ch-ua-mobile': '?0',
'sec-ch-ua-platform': '"Windows"',
'sec-fetch-dest': 'empty',
'sec-fetch-mode': 'cors',
'sec-fetch-site': 'cross-site',
'sec-gpc': '1'
}
}
},
url({ date, channel }) {
return `https://api.cld.dtvce.com/discovery/edge/schedule/v1/service/schedule?startTime=${date.valueOf()}&endTime=${date.add(24, 'hour').valueOf()}&channelIds=${channel.site_id}&include4K=false&is4Kcompatible=false&includeTVOD=true`
},
async parser({ content, channel }) {
const programs = []
const items = parseItems(content, channel)
for (let item of items) {
if (item.programID === '-1') continue
const start = parseStart(item)
const stop = parseStop(item)
programs.push({
title: item.title,
sub_title: item.episodeTitle,
description: parseDescription(item),
rating: parseRating(item),
date: parseFullReleaseDate(item) ?? parseYear(item),
category: parseCategory(item),
season: item.seasonNumber,
episode: item.episodeNumber,
image: parseImage(item),
start,
stop
})
}
return programs
},
async channels() {
// alternate https://www.directv.com/dtvassets/dtv/dev/uf/CHLUP/chnlListingPageData.json
// though i don't think you could fetch the schedule from the API with this
let channels = []
const html = await axios
.get('https://api.cld.dtvce.com/discovery/metadata/channel/v5/service/allchannels?sort=OrdCh%253DASC', {
headers: {
Authorization: `Bearer ${await fetchToken()}`,
'Accept-Language': 'en-US,en;q=0.5',
Connection: 'keep-alive'
}
})
.then(r => r.data)
.catch(console.log)
const data = html?.channelInfoList
if (data && Array.isArray(data)) {
data.forEach(item => {
channels.push({
lang: 'en',
site_id: item.resourceId,
name: item.channelName,
icon: item.imageList && item.imageList.length > 0 ? item.imageList[0].imageUrl : null
})
})
}
return channels
}
}
function parseDescription(item) {
return item ? item.description : null
}
function parseCategory(item) {
return item && item.genres ? item.genres : null
}
// DirecTV are the only ones to put the episode/movie's full release date. Kudos to them.
function parseFullReleaseDate(item) {
return item ? item.originalAirDate : null
}
function parseYear(item) {
return item ? item.releaseYear : null
}
function parseRating(item) {
return item.parentalRating
? {
system: 'MPA',
value: item.parentalRating
}
: null
}
function parseImage(item) {
return item.images?.length > 0 ? item.images[0].defaultImageUrl : null
}
function parseStart(item) {
return dayjs.utc(item.consumables?.[0]?.startTime)
}
function parseStop(item) {
return dayjs.utc(item.consumables?.[0]?.endTime)
}
function parseItems(content, channel) {
try {
const data = JSON.parse(content)
if (!data) return []
if (!Array.isArray(data.schedules)) return []
const channelData = data.schedules.find(i => i.channelId === channel.site_id)
return channelData?.contents && Array.isArray(channelData.contents) ? channelData.contents : []
} catch (error) {
console.error('Error parsing content:', error)
return []
}
}