wip
This commit is contained in:
parent
9332dcc592
commit
482e8c705e
|
@ -2,15 +2,13 @@
|
||||||
|
|
||||||
const { Command } = require('commander')
|
const { Command } = require('commander')
|
||||||
const program = new Command()
|
const program = new Command()
|
||||||
const fs = require('fs')
|
|
||||||
const path = require('path')
|
|
||||||
const EPGGrabber = require('../src/index')
|
|
||||||
const utils = require('../src/utils')
|
|
||||||
const { name, version, description } = require('../package.json')
|
|
||||||
const { merge } = require('lodash')
|
const { merge } = require('lodash')
|
||||||
const { gzip } = require('node-gzip')
|
const { gzip } = require('node-gzip')
|
||||||
const { createLogger, format, transports } = require('winston')
|
const file = require('../src/file')
|
||||||
const { combine, timestamp, printf } = format
|
const EPGGrabber = require('../src/index')
|
||||||
|
const { create: createLogger } = require('../src/logger')
|
||||||
|
const { parseInteger, getUTCDate } = require('../src/utils')
|
||||||
|
const { name, version, description } = require('../package.json')
|
||||||
|
|
||||||
program
|
program
|
||||||
.name(name)
|
.name(name)
|
||||||
|
@ -36,39 +34,13 @@ program
|
||||||
.parse(process.argv)
|
.parse(process.argv)
|
||||||
|
|
||||||
const options = program.opts()
|
const options = program.opts()
|
||||||
|
const logger = createLogger(options)
|
||||||
const fileFormat = printf(({ level, message, timestamp }) => {
|
|
||||||
return `[${timestamp}] ${level.toUpperCase()}: ${message}`
|
|
||||||
})
|
|
||||||
|
|
||||||
const consoleFormat = printf(({ level, message, timestamp }) => {
|
|
||||||
if (level === 'error') return ` Error: ${message}`
|
|
||||||
|
|
||||||
return message
|
|
||||||
})
|
|
||||||
|
|
||||||
const t = [new transports.Console({ format: consoleFormat })]
|
|
||||||
|
|
||||||
if (options.log) {
|
|
||||||
t.push(
|
|
||||||
new transports.File({
|
|
||||||
filename: path.resolve(options.log),
|
|
||||||
format: combine(timestamp(), fileFormat),
|
|
||||||
options: { flags: 'w' }
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
const logger = createLogger({
|
|
||||||
level: options.logLevel,
|
|
||||||
transports: t
|
|
||||||
})
|
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
logger.info('Starting...')
|
logger.info('Starting...')
|
||||||
|
|
||||||
logger.info(`Loading '${options.config}'...`)
|
logger.info(`Loading '${options.config}'...`)
|
||||||
let config = require(path.resolve(options.config))
|
let config = require(file.resolve(options.config))
|
||||||
config = merge(config, {
|
config = merge(config, {
|
||||||
days: options.days,
|
days: options.days,
|
||||||
debug: options.debug,
|
debug: options.debug,
|
||||||
|
@ -83,22 +55,27 @@ async function main() {
|
||||||
if (options.cacheTtl) config.request.cache.ttl = options.cacheTtl
|
if (options.cacheTtl) config.request.cache.ttl = options.cacheTtl
|
||||||
if (options.channels) config.channels = options.channels
|
if (options.channels) config.channels = options.channels
|
||||||
else if (config.channels)
|
else if (config.channels)
|
||||||
config.channels = path.join(path.dirname(options.config), config.channels)
|
config.channels = file.join(file.dirname(options.config), config.channels)
|
||||||
else throw new Error("The required 'channels' property is missing")
|
else throw new Error("The required 'channels' property is missing")
|
||||||
|
|
||||||
if (!config.channels) return logger.error('Path to [site].channels.xml is missing')
|
if (!config.channels) return logger.error('Path to [site].channels.xml is missing')
|
||||||
logger.info(`Loading '${config.channels}'...`)
|
logger.info(`Loading '${config.channels}'...`)
|
||||||
const channelsXML = fs.readFileSync(path.resolve(config.channels), { encoding: 'utf-8' })
|
const grabber = new EPGGrabber(config)
|
||||||
const { channels } = utils.parseChannels(channelsXML)
|
|
||||||
|
const channelsXML = file.read(config.channels)
|
||||||
|
const { channels } = grabber.parseChannels(channelsXML)
|
||||||
|
|
||||||
let programs = []
|
let programs = []
|
||||||
let i = 1
|
let i = 1
|
||||||
let days = options.days || 1
|
let days = options.days || 1
|
||||||
const total = channels.length * days
|
const total = channels.length * days
|
||||||
const utcDate = utils.getUTCDate()
|
const utcDate = getUTCDate()
|
||||||
const dates = Array.from({ length: config.days }, (_, i) => utcDate.add(i, 'd'))
|
const dates = Array.from({ length: config.days }, (_, i) => utcDate.add(i, 'd'))
|
||||||
const grabber = new EPGGrabber(config)
|
|
||||||
for (let channel of channels) {
|
for (let channel of channels) {
|
||||||
|
if (!channel.logo && config.logo) {
|
||||||
|
channel.logo = await grabber.loadLogo(channel)
|
||||||
|
}
|
||||||
|
|
||||||
for (let date of dates) {
|
for (let date of dates) {
|
||||||
await grabber
|
await grabber
|
||||||
.grab(channel, date, (data, err) => {
|
.grab(channel, date, (data, err) => {
|
||||||
|
@ -118,15 +95,15 @@ async function main() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const xml = utils.convertToXMLTV({ config, channels, programs })
|
const xml = grabber.generateXMLTV({ config, channels, programs })
|
||||||
let outputPath = options.output || config.output
|
let outputPath = options.output || config.output
|
||||||
if (options.gzip) {
|
if (options.gzip) {
|
||||||
outputPath = outputPath || 'guide.xml.gz'
|
outputPath = outputPath || 'guide.xml.gz'
|
||||||
const compressed = await gzip(xml)
|
const compressed = await gzip(xml)
|
||||||
utils.writeToFile(outputPath, compressed)
|
file.write(outputPath, compressed)
|
||||||
} else {
|
} else {
|
||||||
outputPath = outputPath || 'guide.xml'
|
outputPath = outputPath || 'guide.xml'
|
||||||
utils.writeToFile(outputPath, xml)
|
file.write(outputPath, xml)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`File '${outputPath}' successfully saved`)
|
logger.info(`File '${outputPath}' successfully saved`)
|
||||||
|
@ -134,7 +111,3 @@ async function main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
main()
|
main()
|
||||||
|
|
||||||
function parseInteger(val) {
|
|
||||||
return val ? parseInt(val) : null
|
|
||||||
}
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,26 @@
|
||||||
|
const convert = require('xml-js')
|
||||||
|
|
||||||
|
module.exports.parse = parse
|
||||||
|
|
||||||
|
function parse(xml) {
|
||||||
|
const result = convert.xml2js(xml)
|
||||||
|
const siteTag = result.elements.find(el => el.name === 'site') || {}
|
||||||
|
if (!siteTag.elements) return []
|
||||||
|
const site = siteTag.attributes.site
|
||||||
|
|
||||||
|
const channelsTag = siteTag.elements.find(el => el.name === 'channels')
|
||||||
|
if (!channelsTag.elements) return []
|
||||||
|
|
||||||
|
const channels = channelsTag.elements
|
||||||
|
.filter(el => el.name === 'channel')
|
||||||
|
.map(el => {
|
||||||
|
const channel = el.attributes
|
||||||
|
if (!el.elements) throw new Error(`Channel '${channel.xmltv_id}' has no valid name`)
|
||||||
|
channel.name = el.elements.find(el => el.type === 'text').text
|
||||||
|
channel.site = channel.site || site
|
||||||
|
|
||||||
|
return channel
|
||||||
|
})
|
||||||
|
|
||||||
|
return { site, channels }
|
||||||
|
}
|
|
@ -0,0 +1,143 @@
|
||||||
|
const { merge } = require('lodash')
|
||||||
|
const { CurlGenerator } = require('curl-generator')
|
||||||
|
const axios = require('axios').default
|
||||||
|
const axiosCookieJarSupport = require('axios-cookiejar-support').default
|
||||||
|
const { setupCache } = require('axios-cache-interceptor')
|
||||||
|
const { isPromise, getUTCDate } = require('./utils')
|
||||||
|
|
||||||
|
axiosCookieJarSupport(axios)
|
||||||
|
|
||||||
|
module.exports.create = create
|
||||||
|
module.exports.buildRequest = buildRequest
|
||||||
|
module.exports.parseResponse = parseResponse
|
||||||
|
|
||||||
|
let timeout
|
||||||
|
|
||||||
|
class Client {
|
||||||
|
constructor() {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function create(config) {
|
||||||
|
const client = setupCache(
|
||||||
|
axios.create({
|
||||||
|
headers: {
|
||||||
|
'User-Agent':
|
||||||
|
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
client.interceptors.request.use(
|
||||||
|
function (request) {
|
||||||
|
if (config.debug) {
|
||||||
|
console.log('Request:', JSON.stringify(request, null, 2))
|
||||||
|
}
|
||||||
|
return request
|
||||||
|
},
|
||||||
|
function (error) {
|
||||||
|
return Promise.reject(error)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
client.interceptors.response.use(
|
||||||
|
function (response) {
|
||||||
|
if (config.debug) {
|
||||||
|
const data =
|
||||||
|
isObject(response.data) || Array.isArray(response.data)
|
||||||
|
? JSON.stringify(response.data)
|
||||||
|
: response.data.toString()
|
||||||
|
console.log(
|
||||||
|
'Response:',
|
||||||
|
JSON.stringify(
|
||||||
|
{
|
||||||
|
headers: response.headers,
|
||||||
|
data,
|
||||||
|
cached: response.cached
|
||||||
|
},
|
||||||
|
null,
|
||||||
|
2
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
clearTimeout(timeout)
|
||||||
|
return response
|
||||||
|
},
|
||||||
|
function (error) {
|
||||||
|
clearTimeout(timeout)
|
||||||
|
return Promise.reject(error)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
return client
|
||||||
|
}
|
||||||
|
|
||||||
|
async function buildRequest({ channel, date, config }) {
|
||||||
|
date = typeof date === 'string' ? getUTCDate(date) : date
|
||||||
|
const CancelToken = axios.CancelToken
|
||||||
|
const source = CancelToken.source()
|
||||||
|
const request = { ...config.request }
|
||||||
|
timeout = setTimeout(() => {
|
||||||
|
source.cancel('Connection timeout')
|
||||||
|
}, request.timeout)
|
||||||
|
request.headers = await getRequestHeaders({ channel, date, config })
|
||||||
|
request.url = await getRequestUrl({ channel, date, config })
|
||||||
|
request.data = await getRequestData({ channel, date, config })
|
||||||
|
request.cancelToken = source.token
|
||||||
|
|
||||||
|
if (config.curl) {
|
||||||
|
const curl = CurlGenerator({
|
||||||
|
url: request.url,
|
||||||
|
method: request.method,
|
||||||
|
headers: request.headers,
|
||||||
|
body: request.data
|
||||||
|
})
|
||||||
|
console.log(curl)
|
||||||
|
}
|
||||||
|
|
||||||
|
return request
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseResponse(response) {
|
||||||
|
return {
|
||||||
|
content: response.data.toString(),
|
||||||
|
buffer: response.data,
|
||||||
|
headers: response.headers,
|
||||||
|
request: response.request,
|
||||||
|
cached: response.cached
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getRequestHeaders({ channel, date, config }) {
|
||||||
|
if (typeof config.request.headers === 'function') {
|
||||||
|
const headers = config.request.headers({ channel, date })
|
||||||
|
if (isPromise(headers)) {
|
||||||
|
return await headers
|
||||||
|
}
|
||||||
|
return headers
|
||||||
|
}
|
||||||
|
|
||||||
|
return config.request.headers || null
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getRequestData({ channel, date, config }) {
|
||||||
|
if (typeof config.request.data === 'function') {
|
||||||
|
const data = config.request.data({ channel, date })
|
||||||
|
if (isPromise(data)) {
|
||||||
|
return await data
|
||||||
|
}
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
return config.request.data || null
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getRequestUrl({ channel, date, config }) {
|
||||||
|
if (typeof config.url === 'function') {
|
||||||
|
const url = config.url({ channel, date })
|
||||||
|
if (isPromise(url)) {
|
||||||
|
return await url
|
||||||
|
}
|
||||||
|
return url
|
||||||
|
}
|
||||||
|
return config.url
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
const tough = require('tough-cookie')
|
||||||
|
const { merge } = require('lodash')
|
||||||
|
|
||||||
|
module.exports.load = load
|
||||||
|
|
||||||
|
function load(config) {
|
||||||
|
if (!config.site) throw new Error("The required 'site' property is missing")
|
||||||
|
if (!config.url) throw new Error("The required 'url' property is missing")
|
||||||
|
if (typeof config.url !== 'function' && typeof config.url !== 'string')
|
||||||
|
throw new Error("The 'url' property should return the function or string")
|
||||||
|
if (!config.parser) throw new Error("The required 'parser' function is missing")
|
||||||
|
if (typeof config.parser !== 'function')
|
||||||
|
throw new Error("The 'parser' property should return the function")
|
||||||
|
if (config.logo && typeof config.logo !== 'function')
|
||||||
|
throw new Error("The 'logo' property should return the function")
|
||||||
|
|
||||||
|
const defaultConfig = {
|
||||||
|
days: 1,
|
||||||
|
lang: 'en',
|
||||||
|
delay: 3000,
|
||||||
|
output: 'guide.xml',
|
||||||
|
request: {
|
||||||
|
method: 'GET',
|
||||||
|
maxContentLength: 5 * 1024 * 1024,
|
||||||
|
timeout: 5000,
|
||||||
|
withCredentials: true,
|
||||||
|
jar: new tough.CookieJar(),
|
||||||
|
responseType: 'arraybuffer',
|
||||||
|
cache: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return merge(defaultConfig, config)
|
||||||
|
}
|
|
@ -0,0 +1,33 @@
|
||||||
|
const fs = require('fs')
|
||||||
|
const path = require('path')
|
||||||
|
|
||||||
|
module.exports.read = read
|
||||||
|
module.exports.write = write
|
||||||
|
module.exports.resolve = resolve
|
||||||
|
module.exports.join = join
|
||||||
|
module.exports.dirname = dirname
|
||||||
|
|
||||||
|
function read(filepath) {
|
||||||
|
return fs.readFileSync(path.resolve(filepath), { encoding: 'utf-8' })
|
||||||
|
}
|
||||||
|
|
||||||
|
function write(filepath, data) {
|
||||||
|
const dir = path.resolve(path.dirname(filepath))
|
||||||
|
if (!fs.existsSync(dir)) {
|
||||||
|
fs.mkdirSync(dir, { recursive: true })
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.writeFileSync(path.resolve(filepath), data)
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolve(filepath) {
|
||||||
|
return path.resolve(filepath)
|
||||||
|
}
|
||||||
|
|
||||||
|
function join(path1, path2) {
|
||||||
|
return path.join(path1, path2)
|
||||||
|
}
|
||||||
|
|
||||||
|
function dirname(filepath) {
|
||||||
|
return path.dirname(filepath)
|
||||||
|
}
|
61
src/index.js
61
src/index.js
|
@ -1,41 +1,48 @@
|
||||||
const utils = require('./utils')
|
const { merge } = require('lodash')
|
||||||
|
const { create: createClient, buildRequest, parseResponse } = require('./client')
|
||||||
|
const { generate: generateXMLTV } = require('./xmltv')
|
||||||
|
const { parse: parseChannels } = require('./channels')
|
||||||
|
const { parse: parsePrograms } = require('./programs')
|
||||||
|
const { load: loadConfig } = require('./config')
|
||||||
|
const { sleep, isPromise } = require('./utils')
|
||||||
|
|
||||||
class EPGGrabber {
|
class EPGGrabber {
|
||||||
constructor(config = {}) {
|
constructor(config = {}) {
|
||||||
this.config = utils.loadConfig(config)
|
this.config = loadConfig(config)
|
||||||
this.client = utils.createClient(config)
|
this.client = createClient(config)
|
||||||
|
}
|
||||||
|
|
||||||
|
async loadLogo(channel) {
|
||||||
|
const logo = this.config.logo({ channel })
|
||||||
|
if (isPromise(logo)) {
|
||||||
|
return await logo
|
||||||
|
}
|
||||||
|
return logo
|
||||||
}
|
}
|
||||||
|
|
||||||
async grab(channel, date, cb = () => {}) {
|
async grab(channel, date, cb = () => {}) {
|
||||||
date = typeof date === 'string' ? utils.getUTCDate(date) : date
|
await sleep(this.config.delay)
|
||||||
channel.lang = channel.lang || this.config.lang || null
|
|
||||||
|
|
||||||
let programs = []
|
return buildRequest({ channel, date, config: this.config })
|
||||||
const item = { date, channel }
|
.then(this.client)
|
||||||
await utils
|
.then(parseResponse)
|
||||||
.buildRequest(item, this.config)
|
.then(data => merge({ channel, date, config: this.config }, data))
|
||||||
.then(request => utils.fetchData(this.client, request))
|
.then(parsePrograms)
|
||||||
.then(response => utils.parseResponse(item, response, this.config))
|
.then(programs => {
|
||||||
.then(results => {
|
cb({ channel, date, programs })
|
||||||
item.programs = results
|
|
||||||
cb(item, null)
|
return programs
|
||||||
programs = programs.concat(results)
|
|
||||||
})
|
})
|
||||||
.catch(error => {
|
.catch(err => {
|
||||||
item.programs = []
|
if (this.config.debug) console.log('Error:', JSON.stringify(err, null, 2))
|
||||||
if (this.config.debug) {
|
cb({ channel, date, programs: [] }, err)
|
||||||
console.log('Error:', JSON.stringify(error, null, 2))
|
|
||||||
}
|
return []
|
||||||
cb(item, error)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
await utils.sleep(this.config.delay)
|
|
||||||
|
|
||||||
return programs
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
EPGGrabber.convertToXMLTV = utils.convertToXMLTV
|
EPGGrabber.prototype.generateXMLTV = generateXMLTV
|
||||||
EPGGrabber.parseChannels = utils.parseChannels
|
EPGGrabber.prototype.parseChannels = parseChannels
|
||||||
|
|
||||||
module.exports = EPGGrabber
|
module.exports = EPGGrabber
|
||||||
|
|
|
@ -0,0 +1,33 @@
|
||||||
|
const { createLogger, format, transports } = require('winston')
|
||||||
|
const { combine, timestamp, printf } = format
|
||||||
|
|
||||||
|
module.exports.create = create
|
||||||
|
|
||||||
|
function create(options) {
|
||||||
|
const fileFormat = printf(({ level, message, timestamp }) => {
|
||||||
|
return `[${timestamp}] ${level.toUpperCase()}: ${message}`
|
||||||
|
})
|
||||||
|
|
||||||
|
const consoleFormat = printf(({ level, message, timestamp }) => {
|
||||||
|
if (level === 'error') return ` Error: ${message}`
|
||||||
|
|
||||||
|
return message
|
||||||
|
})
|
||||||
|
|
||||||
|
const t = [new transports.Console({ format: consoleFormat })]
|
||||||
|
|
||||||
|
if (options.log) {
|
||||||
|
t.push(
|
||||||
|
new transports.File({
|
||||||
|
filename: path.resolve(options.log),
|
||||||
|
format: combine(timestamp(), fileFormat),
|
||||||
|
options: { flags: 'w' }
|
||||||
|
})
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return createLogger({
|
||||||
|
level: options.logLevel,
|
||||||
|
transports: t
|
||||||
|
})
|
||||||
|
}
|
|
@ -0,0 +1,29 @@
|
||||||
|
const dayjs = require('dayjs')
|
||||||
|
const { isPromise } = require('./utils')
|
||||||
|
|
||||||
|
module.exports.parse = parse
|
||||||
|
|
||||||
|
async function parse(data) {
|
||||||
|
const { config, channel } = data
|
||||||
|
let programs = config.parser(data)
|
||||||
|
|
||||||
|
if (isPromise(programs)) {
|
||||||
|
programs = await programs
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Array.isArray(programs)) {
|
||||||
|
throw new Error('Parser should return an array')
|
||||||
|
}
|
||||||
|
|
||||||
|
return programs
|
||||||
|
.filter(i => i)
|
||||||
|
.map(program => {
|
||||||
|
program.channel = channel.xmltv_id
|
||||||
|
|
||||||
|
return program
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function toBaseObject(data) {
|
||||||
|
return data
|
||||||
|
}
|
509
src/utils.js
509
src/utils.js
|
@ -1,127 +1,34 @@
|
||||||
const fs = require('fs')
|
|
||||||
const { padStart } = require('lodash')
|
|
||||||
const path = require('path')
|
|
||||||
const axios = require('axios').default
|
|
||||||
const axiosCookieJarSupport = require('axios-cookiejar-support').default
|
|
||||||
const { setupCache } = require('axios-cache-interceptor')
|
|
||||||
const tough = require('tough-cookie')
|
|
||||||
const convert = require('xml-js')
|
|
||||||
const { merge } = require('lodash')
|
|
||||||
const dayjs = require('dayjs')
|
const dayjs = require('dayjs')
|
||||||
const utc = require('dayjs/plugin/utc')
|
const utc = require('dayjs/plugin/utc')
|
||||||
const { CurlGenerator } = require('curl-generator')
|
|
||||||
dayjs.extend(utc)
|
dayjs.extend(utc)
|
||||||
axiosCookieJarSupport(axios)
|
|
||||||
|
|
||||||
let timeout
|
module.exports.sleep = sleep
|
||||||
const utils = {}
|
module.exports.getUTCDate = getUTCDate
|
||||||
const defaultUserAgent =
|
module.exports.isPromise = isPromise
|
||||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
module.exports.isObject = isObject
|
||||||
|
module.exports.escapeString = escapeString
|
||||||
|
module.exports.parseInteger = parseInteger
|
||||||
|
|
||||||
utils.loadConfig = function (config) {
|
function sleep(ms) {
|
||||||
if (!config.site) throw new Error("The required 'site' property is missing")
|
|
||||||
if (!config.url) throw new Error("The required 'url' property is missing")
|
|
||||||
if (typeof config.url !== 'function' && typeof config.url !== 'string')
|
|
||||||
throw new Error("The 'url' property should return the function or string")
|
|
||||||
if (!config.parser) throw new Error("The required 'parser' function is missing")
|
|
||||||
if (typeof config.parser !== 'function')
|
|
||||||
throw new Error("The 'parser' property should return the function")
|
|
||||||
if (config.logo && typeof config.logo !== 'function')
|
|
||||||
throw new Error("The 'logo' property should return the function")
|
|
||||||
|
|
||||||
const defaultConfig = {
|
|
||||||
days: 1,
|
|
||||||
lang: 'en',
|
|
||||||
delay: 3000,
|
|
||||||
output: 'guide.xml',
|
|
||||||
request: {
|
|
||||||
method: 'GET',
|
|
||||||
maxContentLength: 5 * 1024 * 1024,
|
|
||||||
timeout: 5000,
|
|
||||||
withCredentials: true,
|
|
||||||
jar: new tough.CookieJar(),
|
|
||||||
responseType: 'arraybuffer',
|
|
||||||
cache: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return merge(defaultConfig, config)
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.createClient = function (config) {
|
|
||||||
const client = setupCache(axios.create())
|
|
||||||
client.interceptors.request.use(
|
|
||||||
function (request) {
|
|
||||||
if (config.debug) {
|
|
||||||
console.log('Request:', JSON.stringify(request, null, 2))
|
|
||||||
}
|
|
||||||
return request
|
|
||||||
},
|
|
||||||
function (error) {
|
|
||||||
return Promise.reject(error)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
client.interceptors.response.use(
|
|
||||||
function (response) {
|
|
||||||
if (config.debug) {
|
|
||||||
const data =
|
|
||||||
utils.isObject(response.data) || Array.isArray(response.data)
|
|
||||||
? JSON.stringify(response.data)
|
|
||||||
: response.data.toString()
|
|
||||||
console.log(
|
|
||||||
'Response:',
|
|
||||||
JSON.stringify(
|
|
||||||
{
|
|
||||||
headers: response.headers,
|
|
||||||
data,
|
|
||||||
cached: response.cached
|
|
||||||
},
|
|
||||||
null,
|
|
||||||
2
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
clearTimeout(timeout)
|
|
||||||
return response
|
|
||||||
},
|
|
||||||
function (error) {
|
|
||||||
clearTimeout(timeout)
|
|
||||||
return Promise.reject(error)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
return client
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.parseChannels = function (xml) {
|
|
||||||
const result = convert.xml2js(xml)
|
|
||||||
const siteTag = result.elements.find(el => el.name === 'site') || {}
|
|
||||||
if (!siteTag.elements) return []
|
|
||||||
const site = siteTag.attributes.site
|
|
||||||
|
|
||||||
const channelsTag = siteTag.elements.find(el => el.name === 'channels')
|
|
||||||
if (!channelsTag.elements) return []
|
|
||||||
|
|
||||||
const channels = channelsTag.elements
|
|
||||||
.filter(el => el.name === 'channel')
|
|
||||||
.map(el => {
|
|
||||||
const channel = el.attributes
|
|
||||||
if (!el.elements) throw new Error(`Channel '${channel.xmltv_id}' has no valid name`)
|
|
||||||
channel.name = el.elements.find(el => el.type === 'text').text
|
|
||||||
channel.site = channel.site || site
|
|
||||||
|
|
||||||
return channel
|
|
||||||
})
|
|
||||||
|
|
||||||
return { site, channels }
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.sleep = function (ms) {
|
|
||||||
return new Promise(resolve => setTimeout(resolve, ms))
|
return new Promise(resolve => setTimeout(resolve, ms))
|
||||||
}
|
}
|
||||||
|
|
||||||
utils.escapeString = function (string, defaultValue = '') {
|
function isObject(a) {
|
||||||
|
return !!a && a.constructor === Object
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPromise(promise) {
|
||||||
|
return !!promise && typeof promise.then === 'function'
|
||||||
|
}
|
||||||
|
|
||||||
|
function getUTCDate(d = null) {
|
||||||
|
if (typeof d === 'string') return dayjs.utc(d).startOf('d')
|
||||||
|
|
||||||
|
return dayjs.utc().startOf('d')
|
||||||
|
}
|
||||||
|
|
||||||
|
function escapeString(string, defaultValue = '') {
|
||||||
if (!string) return defaultValue
|
if (!string) return defaultValue
|
||||||
|
|
||||||
const regex = new RegExp(
|
const regex = new RegExp(
|
||||||
|
@ -149,372 +56,6 @@ utils.escapeString = function (string, defaultValue = '') {
|
||||||
.trim()
|
.trim()
|
||||||
}
|
}
|
||||||
|
|
||||||
utils.convertToXMLTV = function ({ channels, programs, date = dayjs.utc() }) {
|
function parseInteger(val) {
|
||||||
let output = `<?xml version="1.0" encoding="UTF-8" ?><tv date="${dayjs(date).format(
|
return val ? parseInt(val) : null
|
||||||
'YYYYMMDD'
|
|
||||||
)}">\r\n`
|
|
||||||
for (let channel of channels) {
|
|
||||||
const id = utils.escapeString(channel['xmltv_id'])
|
|
||||||
const displayName = utils.escapeString(channel.name)
|
|
||||||
output += `<channel id="${id}"><display-name>${displayName}</display-name>`
|
|
||||||
if (channel.logo) {
|
|
||||||
const logo = utils.escapeString(channel.logo)
|
|
||||||
output += `<icon src="${logo}"/>`
|
|
||||||
}
|
|
||||||
if (channel.site) {
|
|
||||||
const url = channel.site ? 'https://' + channel.site : null
|
|
||||||
output += `<url>${url}</url>`
|
|
||||||
}
|
|
||||||
output += `</channel>\r\n`
|
|
||||||
}
|
|
||||||
|
|
||||||
for (let program of programs) {
|
|
||||||
if (!program) continue
|
|
||||||
|
|
||||||
const channel = utils.escapeString(program.channel)
|
|
||||||
const title = utils.escapeString(program.title)
|
|
||||||
const description = utils.escapeString(program.description)
|
|
||||||
const categories = Array.isArray(program.category) ? program.category : [program.category]
|
|
||||||
const start = program.start ? dayjs.unix(program.start).utc().format('YYYYMMDDHHmmss ZZ') : ''
|
|
||||||
const stop = program.stop ? dayjs.unix(program.stop).utc().format('YYYYMMDDHHmmss ZZ') : ''
|
|
||||||
const lang = program.lang || 'en'
|
|
||||||
const xmltv_ns = createXMLTVNS(program.season, program.episode)
|
|
||||||
const onscreen = createOnScreen(program.season, program.episode)
|
|
||||||
const date = program.date || ''
|
|
||||||
const credits = createCredits({
|
|
||||||
director: program.director,
|
|
||||||
actor: program.actor,
|
|
||||||
writer: program.writer,
|
|
||||||
adapter: program.adapter,
|
|
||||||
producer: program.producer,
|
|
||||||
composer: program.composer,
|
|
||||||
editor: program.editor,
|
|
||||||
presenter: program.presenter,
|
|
||||||
commentator: program.commentator,
|
|
||||||
guest: program.guest
|
|
||||||
})
|
|
||||||
const icon = utils.escapeString(program.icon)
|
|
||||||
const sub_title = utils.escapeString(program.sub_title)
|
|
||||||
const url = program.url ? createURL(program.url, channel) : ''
|
|
||||||
|
|
||||||
if (start && stop && title) {
|
|
||||||
output += `<programme start="${start}" stop="${stop}" channel="${channel}"><title lang="${lang}">${title}</title>`
|
|
||||||
|
|
||||||
if (sub_title) {
|
|
||||||
output += `<sub-title>${sub_title}</sub-title>`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (description) {
|
|
||||||
output += `<desc lang="${lang}">${description}</desc>`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (categories.length) {
|
|
||||||
categories.forEach(category => {
|
|
||||||
if (category) {
|
|
||||||
output += `<category lang="${lang}">${utils.escapeString(category)}</category>`
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if (url) {
|
|
||||||
output += url
|
|
||||||
}
|
|
||||||
|
|
||||||
if (xmltv_ns) {
|
|
||||||
output += `<episode-num system="xmltv_ns">${xmltv_ns}</episode-num>`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (onscreen) {
|
|
||||||
output += `<episode-num system="onscreen">${onscreen}</episode-num>`
|
|
||||||
}
|
|
||||||
if (date) {
|
|
||||||
output += `<date>${date}</date>`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (icon) {
|
|
||||||
output += `<icon src="${icon}"/>`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (credits) {
|
|
||||||
output += `<credits>${credits}</credits>`
|
|
||||||
}
|
|
||||||
|
|
||||||
output += '</programme>\r\n'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
output += '</tv>'
|
|
||||||
|
|
||||||
function createXMLTVNS(s, e) {
|
|
||||||
if (!e) return null
|
|
||||||
s = s || 1
|
|
||||||
|
|
||||||
return `${s - 1}.${e - 1}.0/1`
|
|
||||||
}
|
|
||||||
|
|
||||||
function createOnScreen(s, e) {
|
|
||||||
if (!e) return null
|
|
||||||
s = s || 1
|
|
||||||
|
|
||||||
s = padStart(s, 2, '0')
|
|
||||||
e = padStart(e, 2, '0')
|
|
||||||
|
|
||||||
return `S${s}E${e}`
|
|
||||||
}
|
|
||||||
|
|
||||||
function createURL(urlObj, channel = '') {
|
|
||||||
const urls = Array.isArray(urlObj) ? urlObj : [urlObj]
|
|
||||||
let output = ''
|
|
||||||
for (let url of urls) {
|
|
||||||
if (typeof url === 'string' || url instanceof String) {
|
|
||||||
url = { value: url }
|
|
||||||
}
|
|
||||||
|
|
||||||
let attr = url.system ? ` system="${url.system}"` : ''
|
|
||||||
if (url.value.includes('http')) {
|
|
||||||
output += `<url${attr}>${url.value}</url>`
|
|
||||||
} else if (channel) {
|
|
||||||
let chan = channels.find(c => c.xmltv_id.localeCompare(channel) === 0)
|
|
||||||
if (chan && chan.site) {
|
|
||||||
output += `<url${attr}>https://${chan.site}${url.value}</url>`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
function createImage(imgObj, channel = '') {
|
|
||||||
const imgs = Array.isArray(imgObj) ? imgObj : [imgObj]
|
|
||||||
let output = ''
|
|
||||||
for (let img of imgs) {
|
|
||||||
if (typeof img === 'string' || img instanceof String) {
|
|
||||||
img = { value: img }
|
|
||||||
}
|
|
||||||
|
|
||||||
const imageTypes = ['poster', 'backdrop', 'still', 'person', 'character']
|
|
||||||
const imageSizes = ['1', '2', '3']
|
|
||||||
const imageOrients = ['P', 'L']
|
|
||||||
|
|
||||||
let attr = ''
|
|
||||||
|
|
||||||
if (img.type && imageTypes.some(el => img.type.includes(el))) {
|
|
||||||
attr += ` type="${img.type}"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (img.size && imageSizes.some(el => img.size.includes(el))) {
|
|
||||||
attr += ` size="${img.size}"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (img.orient && imageOrients.some(el => img.orient.includes(el))) {
|
|
||||||
attr += ` orient="${img.orient}"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (img.system) {
|
|
||||||
attr += ` system="${img.system}"`
|
|
||||||
}
|
|
||||||
|
|
||||||
if (img.value.includes('http')) {
|
|
||||||
output += `<image${attr}>${img.value}</image>`
|
|
||||||
} else if (channel) {
|
|
||||||
let chan = channels.find(c => c.xmltv_id.localeCompare(channel) === 0)
|
|
||||||
if (chan && chan.site) {
|
|
||||||
output += `<image${attr}>https://${chan.site}${img.value}</image>`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
|
|
||||||
function createCredits(obj) {
|
|
||||||
let cast = Object.entries(obj)
|
|
||||||
.filter(x => x[1])
|
|
||||||
.map(([name, value]) => ({ name, value }))
|
|
||||||
|
|
||||||
let output = ''
|
|
||||||
for (let type of cast) {
|
|
||||||
const r = Array.isArray(type.value) ? type.value : [type.value]
|
|
||||||
for (let person of r) {
|
|
||||||
if (typeof person === 'string' || person instanceof String) {
|
|
||||||
person = { value: person }
|
|
||||||
}
|
|
||||||
|
|
||||||
let attr = ''
|
|
||||||
if (type.name.localeCompare('actor') === 0 && type.value.role) {
|
|
||||||
attr += ` role="${type.value.role}"`
|
|
||||||
}
|
|
||||||
if (type.name.localeCompare('actor') === 0 && type.value.guest) {
|
|
||||||
attr += ` guest="${type.value.guest}"`
|
|
||||||
}
|
|
||||||
output += `<${type.name}${attr}>${person.value}`
|
|
||||||
|
|
||||||
if (person.url) {
|
|
||||||
output += createURL(person.url)
|
|
||||||
}
|
|
||||||
if (person.image) {
|
|
||||||
output += createImage(person.image)
|
|
||||||
}
|
|
||||||
|
|
||||||
output += `</${type.name}>`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
|
||||||
return output
|
|
||||||
}
|
}
|
||||||
|
|
||||||
utils.writeToFile = function (filename, data) {
|
|
||||||
const dir = path.resolve(path.dirname(filename))
|
|
||||||
if (!fs.existsSync(dir)) {
|
|
||||||
fs.mkdirSync(dir, { recursive: true })
|
|
||||||
}
|
|
||||||
|
|
||||||
fs.writeFileSync(path.resolve(filename), data)
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.buildRequest = async function (item, config) {
|
|
||||||
const CancelToken = axios.CancelToken
|
|
||||||
const source = CancelToken.source()
|
|
||||||
const request = { ...config.request }
|
|
||||||
timeout = setTimeout(() => {
|
|
||||||
source.cancel('Connection timeout')
|
|
||||||
}, request.timeout)
|
|
||||||
const headers = await utils.getRequestHeaders(item, config)
|
|
||||||
request.headers = { 'User-Agent': defaultUserAgent, ...headers }
|
|
||||||
request.url = await utils.getRequestUrl(item, config)
|
|
||||||
request.data = await utils.getRequestData(item, config)
|
|
||||||
request.cancelToken = source.token
|
|
||||||
|
|
||||||
if (config.curl) {
|
|
||||||
const curl = CurlGenerator({
|
|
||||||
url: request.url,
|
|
||||||
method: request.method,
|
|
||||||
headers: request.headers,
|
|
||||||
body: request.data
|
|
||||||
})
|
|
||||||
console.log(curl)
|
|
||||||
}
|
|
||||||
|
|
||||||
return request
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.fetchData = function (client, request) {
|
|
||||||
return client(request)
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.getRequestHeaders = async function (item, config) {
|
|
||||||
if (typeof config.request.headers === 'function') {
|
|
||||||
const headers = config.request.headers(item)
|
|
||||||
if (this.isPromise(headers)) {
|
|
||||||
return await headers
|
|
||||||
}
|
|
||||||
return headers
|
|
||||||
}
|
|
||||||
return config.request.headers || null
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.getRequestData = async function (item, config) {
|
|
||||||
if (typeof config.request.data === 'function') {
|
|
||||||
const data = config.request.data(item)
|
|
||||||
if (this.isPromise(data)) {
|
|
||||||
return await data
|
|
||||||
}
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
return config.request.data || null
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.getRequestUrl = async function (item, config) {
|
|
||||||
if (typeof config.url === 'function') {
|
|
||||||
const url = config.url(item)
|
|
||||||
if (this.isPromise(url)) {
|
|
||||||
return await url
|
|
||||||
}
|
|
||||||
return url
|
|
||||||
}
|
|
||||||
return config.url
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.getUTCDate = function (d = null) {
|
|
||||||
if (typeof d === 'string') return dayjs.utc(d).startOf('d')
|
|
||||||
|
|
||||||
return dayjs.utc().startOf('d')
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.parseResponse = async (item, response, config) => {
|
|
||||||
const data = merge(item, config, {
|
|
||||||
content: response.data.toString(),
|
|
||||||
buffer: response.data,
|
|
||||||
headers: response.headers,
|
|
||||||
request: response.request,
|
|
||||||
cached: response.cached
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!item.channel.logo && config.logo) {
|
|
||||||
data.channel.logo = await utils.loadLogo(data, config)
|
|
||||||
}
|
|
||||||
|
|
||||||
return await utils.parsePrograms(data, config)
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.parsePrograms = async function (data, config) {
|
|
||||||
let programs = config.parser(data)
|
|
||||||
|
|
||||||
if (this.isPromise(programs)) {
|
|
||||||
programs = await programs
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!Array.isArray(programs)) {
|
|
||||||
throw new Error('Parser should return an array')
|
|
||||||
}
|
|
||||||
|
|
||||||
const channel = data.channel
|
|
||||||
return programs
|
|
||||||
.filter(i => i)
|
|
||||||
.map(program => {
|
|
||||||
return {
|
|
||||||
title: program.title,
|
|
||||||
description: program.description || null,
|
|
||||||
category: program.category || null,
|
|
||||||
season: program.season || null,
|
|
||||||
episode: program.episode || null,
|
|
||||||
sub_title: program.sub_title || null,
|
|
||||||
url: program.url || null,
|
|
||||||
icon: program.icon || null,
|
|
||||||
channel: channel.xmltv_id,
|
|
||||||
lang: program.lang || channel.lang || config.lang || 'en',
|
|
||||||
start: program.start ? dayjs(program.start).unix() : null,
|
|
||||||
stop: program.stop ? dayjs(program.stop).unix() : null,
|
|
||||||
date: program.date || null,
|
|
||||||
director: program.director || null,
|
|
||||||
actor: program.actor || null,
|
|
||||||
writer: program.writer || null,
|
|
||||||
adapter: program.adapter || null,
|
|
||||||
producer: program.producer || null,
|
|
||||||
composer: program.composer || null,
|
|
||||||
editor: program.editor || null,
|
|
||||||
presenter: program.presenter || null,
|
|
||||||
commentator: program.commentator || null,
|
|
||||||
guest: program.guest || null
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.loadLogo = async function (options, config) {
|
|
||||||
const logo = config.logo(options)
|
|
||||||
if (this.isPromise(logo)) {
|
|
||||||
return await logo
|
|
||||||
}
|
|
||||||
return logo
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.isPromise = function (promise) {
|
|
||||||
return !!promise && typeof promise.then === 'function'
|
|
||||||
}
|
|
||||||
|
|
||||||
utils.isObject = function (a) {
|
|
||||||
return !!a && a.constructor === Object
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = utils
|
|
||||||
|
|
|
@ -0,0 +1,224 @@
|
||||||
|
const { padStart } = require('lodash')
|
||||||
|
const { escapeString, getUTCDate } = require('./utils')
|
||||||
|
const dayjs = require('dayjs')
|
||||||
|
const utc = require('dayjs/plugin/utc')
|
||||||
|
|
||||||
|
dayjs.extend(utc)
|
||||||
|
|
||||||
|
module.exports.generate = generate
|
||||||
|
|
||||||
|
function generate({ channels, programs, date = getUTCDate() }) {
|
||||||
|
let output = `<?xml version="1.0" encoding="UTF-8" ?><tv date="${dayjs(date).format(
|
||||||
|
'YYYYMMDD'
|
||||||
|
)}">\r\n`
|
||||||
|
for (let channel of channels) {
|
||||||
|
const id = escapeString(channel['xmltv_id'])
|
||||||
|
const displayName = escapeString(channel.name)
|
||||||
|
output += `<channel id="${id}"><display-name>${displayName}</display-name>`
|
||||||
|
if (channel.logo) {
|
||||||
|
const logo = escapeString(channel.logo)
|
||||||
|
output += `<icon src="${logo}"/>`
|
||||||
|
}
|
||||||
|
if (channel.site) {
|
||||||
|
const url = channel.site ? 'https://' + channel.site : null
|
||||||
|
output += `<url>${url}</url>`
|
||||||
|
}
|
||||||
|
output += `</channel>\r\n`
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let program of programs) {
|
||||||
|
if (!program) continue
|
||||||
|
|
||||||
|
const channel = escapeString(program.channel)
|
||||||
|
const title = escapeString(program.title)
|
||||||
|
const description = escapeString(program.description)
|
||||||
|
const categories = Array.isArray(program.category) ? program.category : [program.category]
|
||||||
|
const start = program.start ? dayjs.unix(program.start).utc().format('YYYYMMDDHHmmss ZZ') : ''
|
||||||
|
const stop = program.stop ? dayjs.unix(program.stop).utc().format('YYYYMMDDHHmmss ZZ') : ''
|
||||||
|
const lang = program.lang || 'en'
|
||||||
|
const xmltv_ns = createXMLTVNS(program.season, program.episode)
|
||||||
|
const onscreen = createOnScreen(program.season, program.episode)
|
||||||
|
const date = program.date || ''
|
||||||
|
const credits = createCredits({
|
||||||
|
director: program.director,
|
||||||
|
actor: program.actor,
|
||||||
|
writer: program.writer,
|
||||||
|
adapter: program.adapter,
|
||||||
|
producer: program.producer,
|
||||||
|
composer: program.composer,
|
||||||
|
editor: program.editor,
|
||||||
|
presenter: program.presenter,
|
||||||
|
commentator: program.commentator,
|
||||||
|
guest: program.guest
|
||||||
|
})
|
||||||
|
const icon = escapeString(program.icon)
|
||||||
|
const sub_title = escapeString(program.sub_title)
|
||||||
|
const url = program.url ? createURL(program.url, channel) : ''
|
||||||
|
|
||||||
|
if (start && stop && title) {
|
||||||
|
output += `<programme start="${start}" stop="${stop}" channel="${channel}"><title lang="${lang}">${title}</title>`
|
||||||
|
|
||||||
|
if (sub_title) {
|
||||||
|
output += `<sub-title>${sub_title}</sub-title>`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (description) {
|
||||||
|
output += `<desc lang="${lang}">${description}</desc>`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (categories.length) {
|
||||||
|
categories.forEach(category => {
|
||||||
|
if (category) {
|
||||||
|
output += `<category lang="${lang}">${escapeString(category)}</category>`
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (url) {
|
||||||
|
output += url
|
||||||
|
}
|
||||||
|
|
||||||
|
if (xmltv_ns) {
|
||||||
|
output += `<episode-num system="xmltv_ns">${xmltv_ns}</episode-num>`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (onscreen) {
|
||||||
|
output += `<episode-num system="onscreen">${onscreen}</episode-num>`
|
||||||
|
}
|
||||||
|
if (date) {
|
||||||
|
output += `<date>${date}</date>`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (icon) {
|
||||||
|
output += `<icon src="${icon}"/>`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (credits) {
|
||||||
|
output += `<credits>${credits}</credits>`
|
||||||
|
}
|
||||||
|
|
||||||
|
output += '</programme>\r\n'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
output += '</tv>'
|
||||||
|
|
||||||
|
return output
|
||||||
|
}
|
||||||
|
|
||||||
|
function createXMLTVNS(s, e) {
|
||||||
|
if (!e) return null
|
||||||
|
s = s || 1
|
||||||
|
|
||||||
|
return `${s - 1}.${e - 1}.0/1`
|
||||||
|
}
|
||||||
|
|
||||||
|
function createOnScreen(s, e) {
|
||||||
|
if (!e) return null
|
||||||
|
s = s || 1
|
||||||
|
|
||||||
|
s = padStart(s, 2, '0')
|
||||||
|
e = padStart(e, 2, '0')
|
||||||
|
|
||||||
|
return `S${s}E${e}`
|
||||||
|
}
|
||||||
|
|
||||||
|
function createURL(urlObj, channel = '') {
|
||||||
|
const urls = Array.isArray(urlObj) ? urlObj : [urlObj]
|
||||||
|
let output = ''
|
||||||
|
for (let url of urls) {
|
||||||
|
if (typeof url === 'string' || url instanceof String) {
|
||||||
|
url = { value: url }
|
||||||
|
}
|
||||||
|
|
||||||
|
let attr = url.system ? ` system="${url.system}"` : ''
|
||||||
|
if (url.value.includes('http')) {
|
||||||
|
output += `<url${attr}>${url.value}</url>`
|
||||||
|
} else if (channel) {
|
||||||
|
let chan = channels.find(c => c.xmltv_id.localeCompare(channel) === 0)
|
||||||
|
if (chan && chan.site) {
|
||||||
|
output += `<url${attr}>https://${chan.site}${url.value}</url>`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output
|
||||||
|
}
|
||||||
|
|
||||||
|
function createImage(imgObj, channel = '') {
|
||||||
|
const imgs = Array.isArray(imgObj) ? imgObj : [imgObj]
|
||||||
|
let output = ''
|
||||||
|
for (let img of imgs) {
|
||||||
|
if (typeof img === 'string' || img instanceof String) {
|
||||||
|
img = { value: img }
|
||||||
|
}
|
||||||
|
|
||||||
|
const imageTypes = ['poster', 'backdrop', 'still', 'person', 'character']
|
||||||
|
const imageSizes = ['1', '2', '3']
|
||||||
|
const imageOrients = ['P', 'L']
|
||||||
|
|
||||||
|
let attr = ''
|
||||||
|
|
||||||
|
if (img.type && imageTypes.some(el => img.type.includes(el))) {
|
||||||
|
attr += ` type="${img.type}"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (img.size && imageSizes.some(el => img.size.includes(el))) {
|
||||||
|
attr += ` size="${img.size}"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (img.orient && imageOrients.some(el => img.orient.includes(el))) {
|
||||||
|
attr += ` orient="${img.orient}"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (img.system) {
|
||||||
|
attr += ` system="${img.system}"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (img.value.includes('http')) {
|
||||||
|
output += `<image${attr}>${img.value}</image>`
|
||||||
|
} else if (channel) {
|
||||||
|
let chan = channels.find(c => c.xmltv_id.localeCompare(channel) === 0)
|
||||||
|
if (chan && chan.site) {
|
||||||
|
output += `<image${attr}>https://${chan.site}${img.value}</image>`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output
|
||||||
|
}
|
||||||
|
|
||||||
|
function createCredits(obj) {
|
||||||
|
let cast = Object.entries(obj)
|
||||||
|
.filter(x => x[1])
|
||||||
|
.map(([name, value]) => ({ name, value }))
|
||||||
|
|
||||||
|
let output = ''
|
||||||
|
for (let type of cast) {
|
||||||
|
const r = Array.isArray(type.value) ? type.value : [type.value]
|
||||||
|
for (let person of r) {
|
||||||
|
if (typeof person === 'string' || person instanceof String) {
|
||||||
|
person = { value: person }
|
||||||
|
}
|
||||||
|
|
||||||
|
let attr = ''
|
||||||
|
if (type.name.localeCompare('actor') === 0 && type.value.role) {
|
||||||
|
attr += ` role="${type.value.role}"`
|
||||||
|
}
|
||||||
|
if (type.name.localeCompare('actor') === 0 && type.value.guest) {
|
||||||
|
attr += ` guest="${type.value.guest}"`
|
||||||
|
}
|
||||||
|
output += `<${type.name}${attr}>${person.value}`
|
||||||
|
|
||||||
|
if (person.url) {
|
||||||
|
output += createURL(person.url)
|
||||||
|
}
|
||||||
|
if (person.image) {
|
||||||
|
output += createImage(person.image)
|
||||||
|
}
|
||||||
|
|
||||||
|
output += `</${type.name}>`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return output
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
import { parse as parseChannels } from '../src/channels'
|
||||||
|
import path from 'path'
|
||||||
|
import fs from 'fs'
|
||||||
|
|
||||||
|
it('can parse valid channels.xml', () => {
|
||||||
|
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
||||||
|
const { channels } = parseChannels(file)
|
||||||
|
expect(channels).toEqual([
|
||||||
|
{
|
||||||
|
name: '1 TV',
|
||||||
|
xmltv_id: '1TV.com',
|
||||||
|
site_id: '1',
|
||||||
|
site: 'example.com',
|
||||||
|
lang: 'fr',
|
||||||
|
logo: 'https://example.com/logos/1TV.png'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '2 TV',
|
||||||
|
xmltv_id: '2TV.com',
|
||||||
|
site_id: '2',
|
||||||
|
site: 'example.com',
|
||||||
|
lang: undefined,
|
||||||
|
logo: undefined
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
|
@ -0,0 +1,47 @@
|
||||||
|
import { buildRequest, create as createClient } from '../src/client'
|
||||||
|
|
||||||
|
const config = {
|
||||||
|
days: 1,
|
||||||
|
lang: 'en',
|
||||||
|
delay: 3000,
|
||||||
|
output: 'guide.xml',
|
||||||
|
request: {
|
||||||
|
method: 'POST',
|
||||||
|
maxContentLength: 5 * 1024 * 1024,
|
||||||
|
timeout: 5000,
|
||||||
|
withCredentials: true,
|
||||||
|
responseType: 'arraybuffer',
|
||||||
|
cache: false,
|
||||||
|
data: { accountID: '123' },
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
Cookie: 'abc=123',
|
||||||
|
'User-Agent':
|
||||||
|
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
url: 'http://example.com/20210319/1tv.json'
|
||||||
|
}
|
||||||
|
|
||||||
|
it('can build request', done => {
|
||||||
|
buildRequest({ config })
|
||||||
|
.then(request => {
|
||||||
|
expect(request).toMatchObject({
|
||||||
|
data: { accountID: '123' },
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
Cookie: 'abc=123',
|
||||||
|
'User-Agent':
|
||||||
|
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
||||||
|
},
|
||||||
|
maxContentLength: 5242880,
|
||||||
|
method: 'POST',
|
||||||
|
responseType: 'arraybuffer',
|
||||||
|
timeout: 5000,
|
||||||
|
url: 'http://example.com/20210319/1tv.json',
|
||||||
|
withCredentials: true
|
||||||
|
})
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
.catch(done)
|
||||||
|
})
|
|
@ -0,0 +1,26 @@
|
||||||
|
import { load as loadConfig } from '../src/config'
|
||||||
|
import path from 'path'
|
||||||
|
import fs from 'fs'
|
||||||
|
|
||||||
|
it('can load config', () => {
|
||||||
|
const config = loadConfig(require(path.resolve('./tests/input/example.com.config.js')))
|
||||||
|
expect(config).toMatchObject({
|
||||||
|
days: 1,
|
||||||
|
delay: 3000,
|
||||||
|
lang: 'en',
|
||||||
|
site: 'example.com'
|
||||||
|
})
|
||||||
|
expect(config.request).toMatchObject({
|
||||||
|
timeout: 5000,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
Cookie: 'abc=123'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
expect(typeof config.request.data).toEqual('function')
|
||||||
|
expect(typeof config.url).toEqual('function')
|
||||||
|
expect(typeof config.logo).toEqual('function')
|
||||||
|
expect(config.request.data()).toEqual({ accountID: '123' })
|
||||||
|
expect(config.url()).toEqual('http://example.com/20210319/1tv.json')
|
||||||
|
expect(config.logo()).toEqual('http://example.com/logos/1TV.png?x=шеллы&sid=777')
|
||||||
|
})
|
|
@ -0,0 +1,66 @@
|
||||||
|
import { parse as parsePrograms } from '../src/programs'
|
||||||
|
|
||||||
|
const channel = { xmltv_id: '1tv', lang: 'en' }
|
||||||
|
|
||||||
|
it('can parse programs', done => {
|
||||||
|
const config = {
|
||||||
|
parser: () => [
|
||||||
|
{
|
||||||
|
title: 'Title',
|
||||||
|
description: 'Description',
|
||||||
|
lang: 'en',
|
||||||
|
category: ['Category1', 'Category2'],
|
||||||
|
icon: 'https://example.com/image.jpg',
|
||||||
|
season: 9,
|
||||||
|
episode: 238,
|
||||||
|
start: 1640995200,
|
||||||
|
stop: 1640998800
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
parsePrograms({ channel, config })
|
||||||
|
.then(programs => {
|
||||||
|
expect(programs).toMatchObject([
|
||||||
|
{
|
||||||
|
title: 'Title',
|
||||||
|
description: 'Description',
|
||||||
|
lang: 'en',
|
||||||
|
category: ['Category1', 'Category2'],
|
||||||
|
icon: 'https://example.com/image.jpg',
|
||||||
|
season: 9,
|
||||||
|
episode: 238,
|
||||||
|
start: 1640995200,
|
||||||
|
stop: 1640998800,
|
||||||
|
channel: '1tv'
|
||||||
|
}
|
||||||
|
])
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
.catch(done)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can parse programs async', done => {
|
||||||
|
const config = {
|
||||||
|
parser: async () => [
|
||||||
|
{
|
||||||
|
title: 'Title',
|
||||||
|
description: 'Description',
|
||||||
|
lang: 'en',
|
||||||
|
category: ['Category1', 'Category2'],
|
||||||
|
icon: 'https://example.com/image.jpg',
|
||||||
|
season: 9,
|
||||||
|
episode: 238,
|
||||||
|
start: 1640995200,
|
||||||
|
stop: 1640998800
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
parsePrograms({ channel, config })
|
||||||
|
.then(programs => {
|
||||||
|
expect(programs.length).toBe(1)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
.catch(done)
|
||||||
|
})
|
|
@ -1,401 +1,11 @@
|
||||||
import mockAxios from 'jest-mock-axios'
|
import { escapeString } from '../src/utils'
|
||||||
import utils from '../src/utils'
|
|
||||||
import axios from 'axios'
|
|
||||||
import path from 'path'
|
|
||||||
import fs from 'fs'
|
|
||||||
|
|
||||||
jest.useFakeTimers('modern').setSystemTime(new Date('2022-05-05'))
|
|
||||||
|
|
||||||
it('can load valid config.js', () => {
|
|
||||||
const config = utils.loadConfig(require(path.resolve('./tests/input/example.com.config.js')))
|
|
||||||
expect(config).toMatchObject({
|
|
||||||
days: 1,
|
|
||||||
delay: 3000,
|
|
||||||
lang: 'en',
|
|
||||||
site: 'example.com'
|
|
||||||
})
|
|
||||||
expect(config.request).toMatchObject({
|
|
||||||
timeout: 5000,
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
Cookie: 'abc=123'
|
|
||||||
}
|
|
||||||
})
|
|
||||||
expect(typeof config.request.data).toEqual('function')
|
|
||||||
expect(typeof config.url).toEqual('function')
|
|
||||||
expect(typeof config.logo).toEqual('function')
|
|
||||||
expect(config.request.data()).toEqual({ accountID: '123' })
|
|
||||||
expect(config.url()).toEqual('http://example.com/20210319/1tv.json')
|
|
||||||
expect(config.logo()).toEqual('http://example.com/logos/1TV.png?x=шеллы&sid=777')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can parse valid channels.xml', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
expect(channels).toEqual([
|
|
||||||
{
|
|
||||||
name: '1 TV',
|
|
||||||
xmltv_id: '1TV.com',
|
|
||||||
site_id: '1',
|
|
||||||
site: 'example.com',
|
|
||||||
lang: 'fr',
|
|
||||||
logo: 'https://example.com/logos/1TV.png'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: '2 TV',
|
|
||||||
xmltv_id: '2TV.com',
|
|
||||||
site_id: '2',
|
|
||||||
site: 'example.com',
|
|
||||||
lang: undefined,
|
|
||||||
logo: undefined
|
|
||||||
}
|
|
||||||
])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
sub_title: 'Sub-title & 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
url: 'http://example.com/title.html',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: 'Test',
|
|
||||||
season: 9,
|
|
||||||
episode: 239,
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it',
|
|
||||||
date: '20220505',
|
|
||||||
director: {
|
|
||||||
value: 'Director 1',
|
|
||||||
url: { value: 'http://example.com/director1.html', system: 'TestSystem' }
|
|
||||||
},
|
|
||||||
actor: ['Actor 1', 'Actor 2'],
|
|
||||||
writer: 'Writer 1'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><sub-title>Sub-title & 1</sub-title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><url>http://example.com/title.html</url><episode-num system="xmltv_ns">8.238.0/1</episode-num><episode-num system="onscreen">S09E239</episode-num><date>20220505</date><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><director>Director 1<url system="TestSystem">http://example.com/director1.html</url></director><actor>Actor 1</actor><actor>Actor 2</actor><writer>Writer 1</writer></credits></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string without season number', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: 'Test',
|
|
||||||
episode: 239,
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><episode-num system="xmltv_ns">0.238.0/1</episode-num><episode-num system="onscreen">S01E239</episode-num><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string without episode number', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: 'Test',
|
|
||||||
season: 1,
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string without categories', () => {
|
|
||||||
const channels = [
|
|
||||||
{
|
|
||||||
name: '1 TV',
|
|
||||||
xmltv_id: '1TV.com',
|
|
||||||
site_id: '1',
|
|
||||||
site: 'example.com',
|
|
||||||
lang: 'fr',
|
|
||||||
logo: 'https://example.com/logos/1TV.png'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const config = { site: 'example.com' }
|
|
||||||
const output = utils.convertToXMLTV({ config, channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string with multiple categories', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: ['Test1', 'Test2'],
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string with multiple urls', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: ['Test1', 'Test2'],
|
|
||||||
url: [
|
|
||||||
'https://example.com/noattr.html',
|
|
||||||
{ value: 'https://example.com/attr.html', system: 'TestSystem' }
|
|
||||||
],
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><url>https://example.com/noattr.html</url><url system="TestSystem">https://example.com/attr.html</url><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string with multiple images', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: ['Test1', 'Test2'],
|
|
||||||
url: [
|
|
||||||
'https://example.com/noattr.html',
|
|
||||||
{ value: 'https://example.com/attr.html', system: 'TestSystem' }
|
|
||||||
],
|
|
||||||
actor: {
|
|
||||||
value: 'Actor 1',
|
|
||||||
image: [
|
|
||||||
'https://example.com/image1.jpg',
|
|
||||||
{
|
|
||||||
value: 'https://example.com/image2.jpg',
|
|
||||||
type: 'person',
|
|
||||||
size: '2',
|
|
||||||
system: 'TestSystem',
|
|
||||||
orient: 'P'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><url>https://example.com/noattr.html</url><url system="TestSystem">https://example.com/attr.html</url><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><actor>Actor 1<image>https://example.com/image1.jpg</image><image type="person" size="2" orient="P" system="TestSystem">https://example.com/image2.jpg</image></actor></credits></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can convert object to xmltv string with multiple credits member', () => {
|
|
||||||
const file = fs.readFileSync('./tests/input/example.com.channels.xml', { encoding: 'utf-8' })
|
|
||||||
const { channels } = utils.parseChannels(file)
|
|
||||||
const programs = [
|
|
||||||
{
|
|
||||||
title: 'Program 1',
|
|
||||||
sub_title: 'Sub-title 1',
|
|
||||||
description: 'Description for Program 1',
|
|
||||||
url: 'http://example.com/title.html',
|
|
||||||
start: 1616133600,
|
|
||||||
stop: 1616135400,
|
|
||||||
category: 'Test',
|
|
||||||
season: 9,
|
|
||||||
episode: 239,
|
|
||||||
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
|
||||||
channel: '1TV.com',
|
|
||||||
lang: 'it',
|
|
||||||
date: '20220505',
|
|
||||||
director: {
|
|
||||||
value: 'Director 1',
|
|
||||||
url: { value: 'http://example.com/director1.html', system: 'TestSystem' }
|
|
||||||
},
|
|
||||||
actor: {
|
|
||||||
value: 'Actor 1',
|
|
||||||
role: 'Manny',
|
|
||||||
guest: 'yes',
|
|
||||||
url: { value: 'http://example.com/actor1.html', system: 'TestSystem' }
|
|
||||||
},
|
|
||||||
writer: [
|
|
||||||
{ value: 'Writer 1', url: { value: 'http://example.com/w1.html', system: 'TestSystem' } },
|
|
||||||
{ value: 'Writer 2', url: { value: 'http://example.com/w2.html', system: 'TestSystem' } }
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
const output = utils.convertToXMLTV({ channels, programs })
|
|
||||||
expect(output).toBe(
|
|
||||||
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><sub-title>Sub-title 1</sub-title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><url>http://example.com/title.html</url><episode-num system="xmltv_ns">8.238.0/1</episode-num><episode-num system="onscreen">S09E239</episode-num><date>20220505</date><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><director>Director 1<url system="TestSystem">http://example.com/director1.html</url></director><actor role="Manny" guest="yes">Actor 1<url system="TestSystem">http://example.com/actor1.html</url></actor><writer>Writer 1<url system="TestSystem">http://example.com/w1.html</url></writer><writer>Writer 2<url system="TestSystem">http://example.com/w2.html</url></writer></credits></programme>\r\n</tv>'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can escape string', () => {
|
it('can escape string', () => {
|
||||||
const string = 'Música тест dun. &<>"\'\r\n'
|
const string = 'Música тест dun. &<>"\'\r\n'
|
||||||
expect(utils.escapeString(string)).toBe('Música тест dun. &<>"'')
|
expect(escapeString(string)).toBe('Música тест dun. &<>"'')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('can escape url', () => {
|
it('can escape url', () => {
|
||||||
const string = 'http://example.com/logos/1TV.png?param1=val¶m2=val'
|
const string = 'http://example.com/logos/1TV.png?param1=val¶m2=val'
|
||||||
expect(utils.escapeString(string)).toBe(
|
expect(escapeString(string)).toBe('http://example.com/logos/1TV.png?param1=val&param2=val')
|
||||||
'http://example.com/logos/1TV.png?param1=val&param2=val'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can fetch data', () => {
|
|
||||||
const request = {
|
|
||||||
data: { accountID: '123' },
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
Cookie: 'abc=123',
|
|
||||||
'User-Agent':
|
|
||||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
|
||||||
},
|
|
||||||
maxContentLength: 5242880,
|
|
||||||
method: 'POST',
|
|
||||||
responseType: 'arraybuffer',
|
|
||||||
timeout: 5000,
|
|
||||||
url: 'http://example.com/20210319/1tv.json',
|
|
||||||
withCredentials: true
|
|
||||||
}
|
|
||||||
utils.fetchData(mockAxios, request).then(jest.fn).catch(jest.fn)
|
|
||||||
expect(mockAxios).toHaveBeenCalledWith(
|
|
||||||
expect.objectContaining({
|
|
||||||
data: { accountID: '123' },
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
Cookie: 'abc=123',
|
|
||||||
'User-Agent':
|
|
||||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
|
||||||
},
|
|
||||||
method: 'POST',
|
|
||||||
responseType: 'arraybuffer',
|
|
||||||
timeout: 5000,
|
|
||||||
url: 'http://example.com/20210319/1tv.json',
|
|
||||||
withCredentials: true
|
|
||||||
})
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can build request async', done => {
|
|
||||||
const config = utils.loadConfig(require(path.resolve('./tests/input/async.config.js')))
|
|
||||||
|
|
||||||
utils
|
|
||||||
.buildRequest({}, config)
|
|
||||||
.then(request => {
|
|
||||||
expect(request).toMatchObject({
|
|
||||||
data: { accountID: '123' },
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
Cookie: 'abc=123',
|
|
||||||
'User-Agent':
|
|
||||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36 Edg/79.0.309.71'
|
|
||||||
},
|
|
||||||
maxContentLength: 5242880,
|
|
||||||
method: 'POST',
|
|
||||||
responseType: 'arraybuffer',
|
|
||||||
timeout: 5000,
|
|
||||||
url: 'http://example.com/20210319/1tv.json',
|
|
||||||
withCredentials: true
|
|
||||||
})
|
|
||||||
done()
|
|
||||||
})
|
|
||||||
.catch(done)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can load logo async', done => {
|
|
||||||
const config = utils.loadConfig(require(path.resolve('./tests/input/async.config.js')))
|
|
||||||
|
|
||||||
utils
|
|
||||||
.loadLogo({}, config)
|
|
||||||
.then(logo => {
|
|
||||||
expect(logo).toBe('http://example.com/logos/1TV.png?x=шеллы&sid=777')
|
|
||||||
done()
|
|
||||||
})
|
|
||||||
.catch(done)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can parse programs', done => {
|
|
||||||
const config = utils.loadConfig(require(path.resolve('./tests/input/example.com.config.js')))
|
|
||||||
|
|
||||||
utils
|
|
||||||
.parsePrograms({ channel: { xmltv_id: '1tv', lang: 'en' } }, config)
|
|
||||||
.then(programs => {
|
|
||||||
expect(programs).toMatchObject([
|
|
||||||
{
|
|
||||||
title: 'Title',
|
|
||||||
description: 'Description',
|
|
||||||
lang: 'en',
|
|
||||||
category: ['Category1', 'Category2'],
|
|
||||||
icon: 'https://example.com/image.jpg',
|
|
||||||
season: 9,
|
|
||||||
episode: 238,
|
|
||||||
start: 1640995200,
|
|
||||||
stop: 1640998800
|
|
||||||
}
|
|
||||||
])
|
|
||||||
done()
|
|
||||||
})
|
|
||||||
.catch(done)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('can parse programs async', done => {
|
|
||||||
const config = utils.loadConfig(require(path.resolve('./tests/input/async.config.js')))
|
|
||||||
|
|
||||||
utils
|
|
||||||
.parsePrograms({ channel: { xmltv_id: '1tv', lang: 'en' } }, config)
|
|
||||||
.then(programs => {
|
|
||||||
expect(programs.length).toBe(0)
|
|
||||||
done()
|
|
||||||
})
|
|
||||||
.catch(done)
|
|
||||||
})
|
})
|
||||||
|
|
|
@ -0,0 +1,220 @@
|
||||||
|
import xmltv from '../src/xmltv'
|
||||||
|
|
||||||
|
jest.useFakeTimers('modern').setSystemTime(new Date('2022-05-05'))
|
||||||
|
|
||||||
|
const channels = [
|
||||||
|
{
|
||||||
|
xmltv_id: '1TV.com',
|
||||||
|
name: '1 TV',
|
||||||
|
logo: 'https://example.com/logos/1TV.png',
|
||||||
|
site: 'example.com'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
xmltv_id: '2TV.com',
|
||||||
|
name: '2 TV',
|
||||||
|
site: 'example.com'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
it('can generate xmltv', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
sub_title: 'Sub-title & 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
url: 'http://example.com/title.html',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: 'Test',
|
||||||
|
season: 9,
|
||||||
|
episode: 239,
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it',
|
||||||
|
date: '20220505',
|
||||||
|
director: {
|
||||||
|
value: 'Director 1',
|
||||||
|
url: { value: 'http://example.com/director1.html', system: 'TestSystem' }
|
||||||
|
},
|
||||||
|
actor: ['Actor 1', 'Actor 2'],
|
||||||
|
writer: 'Writer 1'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><sub-title>Sub-title & 1</sub-title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><url>http://example.com/title.html</url><episode-num system="xmltv_ns">8.238.0/1</episode-num><episode-num system="onscreen">S09E239</episode-num><date>20220505</date><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><director>Director 1<url system="TestSystem">http://example.com/director1.html</url></director><actor>Actor 1</actor><actor>Actor 2</actor><writer>Writer 1</writer></credits></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv without season number', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: 'Test',
|
||||||
|
episode: 239,
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><episode-num system="xmltv_ns">0.238.0/1</episode-num><episode-num system="onscreen">S01E239</episode-num><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv without episode number', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: 'Test',
|
||||||
|
season: 1,
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv without categories', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const config = { site: 'example.com' }
|
||||||
|
const output = xmltv.generate({ config, channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv with multiple categories', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: ['Test1', 'Test2'],
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv with multiple urls', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: ['Test1', 'Test2'],
|
||||||
|
url: [
|
||||||
|
'https://example.com/noattr.html',
|
||||||
|
{ value: 'https://example.com/attr.html', system: 'TestSystem' }
|
||||||
|
],
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><url>https://example.com/noattr.html</url><url system="TestSystem">https://example.com/attr.html</url><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv with multiple images', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: ['Test1', 'Test2'],
|
||||||
|
url: [
|
||||||
|
'https://example.com/noattr.html',
|
||||||
|
{ value: 'https://example.com/attr.html', system: 'TestSystem' }
|
||||||
|
],
|
||||||
|
actor: {
|
||||||
|
value: 'Actor 1',
|
||||||
|
image: [
|
||||||
|
'https://example.com/image1.jpg',
|
||||||
|
{
|
||||||
|
value: 'https://example.com/image2.jpg',
|
||||||
|
type: 'person',
|
||||||
|
size: '2',
|
||||||
|
system: 'TestSystem',
|
||||||
|
orient: 'P'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><desc lang="it">Description for Program 1</desc><category lang="it">Test1</category><category lang="it">Test2</category><url>https://example.com/noattr.html</url><url system="TestSystem">https://example.com/attr.html</url><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><actor>Actor 1<image>https://example.com/image1.jpg</image><image type="person" size="2" orient="P" system="TestSystem">https://example.com/image2.jpg</image></actor></credits></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can generate xmltv with multiple credits member', () => {
|
||||||
|
const programs = [
|
||||||
|
{
|
||||||
|
title: 'Program 1',
|
||||||
|
sub_title: 'Sub-title 1',
|
||||||
|
description: 'Description for Program 1',
|
||||||
|
url: 'http://example.com/title.html',
|
||||||
|
start: 1616133600,
|
||||||
|
stop: 1616135400,
|
||||||
|
category: 'Test',
|
||||||
|
season: 9,
|
||||||
|
episode: 239,
|
||||||
|
icon: 'https://example.com/images/Program1.png?x=шеллы&sid=777',
|
||||||
|
channel: '1TV.com',
|
||||||
|
lang: 'it',
|
||||||
|
date: '20220505',
|
||||||
|
director: {
|
||||||
|
value: 'Director 1',
|
||||||
|
url: { value: 'http://example.com/director1.html', system: 'TestSystem' }
|
||||||
|
},
|
||||||
|
actor: {
|
||||||
|
value: 'Actor 1',
|
||||||
|
role: 'Manny',
|
||||||
|
guest: 'yes',
|
||||||
|
url: { value: 'http://example.com/actor1.html', system: 'TestSystem' }
|
||||||
|
},
|
||||||
|
writer: [
|
||||||
|
{ value: 'Writer 1', url: { value: 'http://example.com/w1.html', system: 'TestSystem' } },
|
||||||
|
{ value: 'Writer 2', url: { value: 'http://example.com/w2.html', system: 'TestSystem' } }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
const output = xmltv.generate({ channels, programs })
|
||||||
|
expect(output).toBe(
|
||||||
|
'<?xml version="1.0" encoding="UTF-8" ?><tv date="20220505">\r\n<channel id="1TV.com"><display-name>1 TV</display-name><icon src="https://example.com/logos/1TV.png"/><url>https://example.com</url></channel>\r\n<channel id="2TV.com"><display-name>2 TV</display-name><url>https://example.com</url></channel>\r\n<programme start="20210319060000 +0000" stop="20210319063000 +0000" channel="1TV.com"><title lang="it">Program 1</title><sub-title>Sub-title 1</sub-title><desc lang="it">Description for Program 1</desc><category lang="it">Test</category><url>http://example.com/title.html</url><episode-num system="xmltv_ns">8.238.0/1</episode-num><episode-num system="onscreen">S09E239</episode-num><date>20220505</date><icon src="https://example.com/images/Program1.png?x=шеллы&sid=777"/><credits><director>Director 1<url system="TestSystem">http://example.com/director1.html</url></director><actor role="Manny" guest="yes">Actor 1<url system="TestSystem">http://example.com/actor1.html</url></actor><writer>Writer 1<url system="TestSystem">http://example.com/w1.html</url></writer><writer>Writer 2<url system="TestSystem">http://example.com/w2.html</url></writer></credits></programme>\r\n</tv>'
|
||||||
|
)
|
||||||
|
})
|
Loading…
Reference in New Issue