mirror of
https://github.com/iptv-org/epg.git
synced 2025-05-09 16:40:07 -04:00
wip
This commit is contained in:
parent
fa8549c3e9
commit
d068f5083f
11 changed files with 142749 additions and 5 deletions
115
scripts/commands/export-results.js
Normal file
115
scripts/commands/export-results.js
Normal file
|
@ -0,0 +1,115 @@
|
|||
const { db, logger, file, parser } = require('../core')
|
||||
const _ = require('lodash')
|
||||
|
||||
const LOGS_PATH = process.env.LOGS_PATH || 'scripts/logs'
|
||||
const OUTPUT_PATH = process.env.OUTPUT_PATH || 'scripts/output'
|
||||
|
||||
let channels = []
|
||||
let programs = []
|
||||
|
||||
async function main() {
|
||||
await setUp()
|
||||
|
||||
await createChannelsJson()
|
||||
await createProgramsJson()
|
||||
}
|
||||
|
||||
main()
|
||||
|
||||
async function createChannelsJson() {
|
||||
logger.info('Creating channels.json...')
|
||||
|
||||
let items = channels
|
||||
items = _.sortBy(items, item => item.name)
|
||||
|
||||
let buffer = {}
|
||||
items.forEach(item => {
|
||||
if (!buffer[item.xmltv_id]) {
|
||||
const countryCode = item.xmltv_id.split('.')[1]
|
||||
|
||||
buffer[item.xmltv_id] = {
|
||||
id: item.xmltv_id,
|
||||
name: [item.name],
|
||||
logo: item.logo || null,
|
||||
country: countryCode ? countryCode.toUpperCase() : null
|
||||
}
|
||||
} else {
|
||||
if (!buffer[item.xmltv_id].logo && item.logo) {
|
||||
buffer[item.xmltv_id].logo = item.logo
|
||||
}
|
||||
|
||||
if (!buffer[item.xmltv_id].name.includes(item.name)) {
|
||||
buffer[item.xmltv_id].name.push(item.name)
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
items = Object.values(buffer)
|
||||
|
||||
await file.create(`${OUTPUT_PATH}/channels.json`, JSON.stringify(items, null, 2))
|
||||
}
|
||||
|
||||
async function createProgramsJson() {
|
||||
logger.info('Creating programs.json...')
|
||||
|
||||
let items = programs
|
||||
|
||||
items = _.sortBy(items, ['channel', 'start'])
|
||||
items = _.groupBy(items, 'channel')
|
||||
|
||||
for (let channel in items) {
|
||||
let programs = items[channel]
|
||||
programs = Object.values(_.groupBy(programs, i => i.site))[0]
|
||||
let slots = _.groupBy(programs, i => `${i.start}_${i.stop}`)
|
||||
|
||||
for (let slotId in slots) {
|
||||
let program = {
|
||||
channel,
|
||||
site: null,
|
||||
title: [],
|
||||
description: [],
|
||||
categories: [],
|
||||
icons: [],
|
||||
start: null,
|
||||
stop: null
|
||||
}
|
||||
|
||||
slots[slotId].forEach(item => {
|
||||
program.site = item.site
|
||||
if (item.title) program.title.push({ lang: item.lang, value: item.title })
|
||||
if (item.description)
|
||||
program.description.push({
|
||||
lang: item.lang,
|
||||
value: item.description
|
||||
})
|
||||
if (item.category) program.categories.push({ lang: item.lang, value: item.category })
|
||||
if (item.icon) program.icons.push(item.icon)
|
||||
program.start = item.start
|
||||
program.stop = item.stop
|
||||
})
|
||||
|
||||
slots[slotId] = program
|
||||
}
|
||||
|
||||
items[channel] = Object.values(slots)
|
||||
}
|
||||
// console.log(items)
|
||||
|
||||
await file.create(`${OUTPUT_PATH}/programs.json`, JSON.stringify(items, null, 2))
|
||||
}
|
||||
|
||||
async function setUp() {
|
||||
channels = await db.find({})
|
||||
|
||||
const files = await file.list(`${LOGS_PATH}/load-cluster/cluster_*.log`)
|
||||
for (const filepath of files) {
|
||||
const results = await parser.parseLogs(filepath)
|
||||
results.forEach(result => {
|
||||
let pm = result.programs.map(p => {
|
||||
p.site = result.site
|
||||
return p
|
||||
})
|
||||
programs = programs.concat(pm)
|
||||
})
|
||||
}
|
||||
}
|
144
scripts/commands/generate-guides.js
Normal file
144
scripts/commands/generate-guides.js
Normal file
|
@ -0,0 +1,144 @@
|
|||
const { db, logger, file, parser } = require('../core')
|
||||
const _ = require('lodash')
|
||||
const dayjs = require('dayjs')
|
||||
const utc = require('dayjs/plugin/utc')
|
||||
dayjs.extend(utc)
|
||||
|
||||
let programs = {}
|
||||
|
||||
const LOGS_PATH = process.env.LOGS_PATH || 'scripts/logs'
|
||||
const PUBLIC_PATH = process.env.PUBLIC_PATH || '.'
|
||||
|
||||
async function main() {
|
||||
await setUp()
|
||||
|
||||
await generateGuideXML()
|
||||
}
|
||||
|
||||
main()
|
||||
|
||||
async function generateGuideXML() {
|
||||
logger.info(`Generating guide.xml...`)
|
||||
|
||||
const channels = Object.keys(programs)
|
||||
let items = await db.find({ xmltv_id: { $in: channels } })
|
||||
items = _.sortBy(items, item => item.name)
|
||||
|
||||
let buffer = {}
|
||||
items.forEach(item => {
|
||||
if (!buffer[item.xmltv_id]) {
|
||||
const countryCode = item.xmltv_id.split('.')[1]
|
||||
|
||||
buffer[item.xmltv_id] = {
|
||||
id: item.xmltv_id,
|
||||
display_name: [item.name],
|
||||
logo: item.logo || null,
|
||||
country: countryCode ? countryCode.toUpperCase() : null,
|
||||
site: `https://${programs[item.xmltv_id][0].site}`
|
||||
}
|
||||
} else {
|
||||
if (!buffer[item.xmltv_id].logo && item.logo) {
|
||||
buffer[item.xmltv_id].logo = item.logo
|
||||
}
|
||||
|
||||
if (!buffer[item.xmltv_id].display_name.includes(item.name)) {
|
||||
buffer[item.xmltv_id].display_name.push(item.name)
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
items = Object.values(buffer)
|
||||
|
||||
let outputProgs = []
|
||||
for (let ip of Object.values(programs)) {
|
||||
outputProgs = outputProgs.concat(ip)
|
||||
}
|
||||
|
||||
const xml = convertToXMLTV({ channels: items, programs: outputProgs })
|
||||
await file.write('./guide.xml', xml)
|
||||
}
|
||||
|
||||
async function setUp() {
|
||||
const f = await file.read(`${PUBLIC_PATH}/scripts/output/programs.json`)
|
||||
programs = JSON.parse(f)
|
||||
}
|
||||
|
||||
function convertToXMLTV({ channels, programs }) {
|
||||
let output = `<?xml version="1.0" encoding="UTF-8" ?><tv>\r\n`
|
||||
for (let channel of channels) {
|
||||
output += `<channel id="${escapeString(channel.id)}">`
|
||||
channel.display_name.forEach(displayName => {
|
||||
output += `<display-name>${escapeString(displayName)}</display-name>`
|
||||
})
|
||||
if (channel.logo) {
|
||||
const logo = escapeString(channel.logo)
|
||||
output += `<icon src="${logo}"/>`
|
||||
}
|
||||
output += `<url>${channel.site}</url>`
|
||||
output += `</channel>\r\n`
|
||||
}
|
||||
|
||||
for (let program of programs) {
|
||||
if (!program) continue
|
||||
|
||||
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 icon = escapeString(program.icon)
|
||||
|
||||
if (start && stop) {
|
||||
output += `<programme start="${start}" stop="${stop}" channel="${escapeString(
|
||||
program.channel
|
||||
)}">`
|
||||
|
||||
program.title.forEach(title => {
|
||||
output += `<title lang="${title.lang}">${escapeString(title.value)}</title>`
|
||||
})
|
||||
|
||||
program.description.forEach(description => {
|
||||
output += `<desc lang="${description.lang}">${escapeString(description.value)}</desc>`
|
||||
})
|
||||
|
||||
program.categories.forEach(category => {
|
||||
output += `<category lang="${category.lang}">${escapeString(category.value)}</category>`
|
||||
})
|
||||
|
||||
program.icons.forEach(icon => {
|
||||
output += `<icon src="${icon}"/>`
|
||||
})
|
||||
|
||||
output += '</programme>\r\n'
|
||||
}
|
||||
}
|
||||
|
||||
output += '</tv>'
|
||||
|
||||
return output
|
||||
}
|
||||
|
||||
function escapeString(string, defaultValue = '') {
|
||||
if (!string) return defaultValue
|
||||
|
||||
const regex = new RegExp(
|
||||
'((?:[\0-\x08\x0B\f\x0E-\x1F\uFFFD\uFFFE\uFFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|(?:[^\uD800-\uDBFF]|^)[\uDC00-\uDFFF]))|([\\x7F-\\x84]|[\\x86-\\x9F]|[\\uFDD0-\\uFDEF]|(?:\\uD83F[\\uDFFE\\uDFFF])|(?:\\uD87F[\\uDF' +
|
||||
'FE\\uDFFF])|(?:\\uD8BF[\\uDFFE\\uDFFF])|(?:\\uD8FF[\\uDFFE\\uDFFF])|(?:\\uD93F[\\uDFFE\\uD' +
|
||||
'FFF])|(?:\\uD97F[\\uDFFE\\uDFFF])|(?:\\uD9BF[\\uDFFE\\uDFFF])|(?:\\uD9FF[\\uDFFE\\uDFFF])' +
|
||||
'|(?:\\uDA3F[\\uDFFE\\uDFFF])|(?:\\uDA7F[\\uDFFE\\uDFFF])|(?:\\uDABF[\\uDFFE\\uDFFF])|(?:\\' +
|
||||
'uDAFF[\\uDFFE\\uDFFF])|(?:\\uDB3F[\\uDFFE\\uDFFF])|(?:\\uDB7F[\\uDFFE\\uDFFF])|(?:\\uDBBF' +
|
||||
'[\\uDFFE\\uDFFF])|(?:\\uDBFF[\\uDFFE\\uDFFF])(?:[\\0-\\t\\x0B\\f\\x0E-\\u2027\\u202A-\\uD7FF\\' +
|
||||
'uE000-\\uFFFF]|[\\uD800-\\uDBFF][\\uDC00-\\uDFFF]|[\\uD800-\\uDBFF](?![\\uDC00-\\uDFFF])|' +
|
||||
'(?:[^\\uD800-\\uDBFF]|^)[\\uDC00-\\uDFFF]))',
|
||||
'g'
|
||||
)
|
||||
|
||||
string = String(string || '').replace(regex, '')
|
||||
|
||||
return string
|
||||
.replace(/&/g, '&')
|
||||
.replace(/</g, '<')
|
||||
.replace(/>/g, '>')
|
||||
.replace(/"/g, '"')
|
||||
.replace(/'/g, ''')
|
||||
.replace(/\n|\r/g, ' ')
|
||||
.replace(/ +/g, ' ')
|
||||
.trim()
|
||||
}
|
|
@ -4,6 +4,7 @@ const { db, logger, timer, file, parser } = require('../core')
|
|||
|
||||
const options = program
|
||||
.requiredOption('-c, --cluster-id <cluster-id>', 'The ID of cluster to load', parser.parseNumber)
|
||||
.option('-d, --days <days>', 'Number of days for which to grab the program', parser.parseNumber)
|
||||
.parse(process.argv)
|
||||
.opts()
|
||||
|
||||
|
@ -18,7 +19,7 @@ async function main() {
|
|||
logger.info(`Creating '${clusterLog}'...`)
|
||||
await file.create(clusterLog)
|
||||
const items = await db.find({ cluster_id: options.clusterId })
|
||||
let days = 2
|
||||
const days = options.days || 1
|
||||
const total = days * items.length
|
||||
logger.info(`Total ${total} requests`)
|
||||
|
||||
|
@ -39,7 +40,10 @@ async function main() {
|
|||
|
||||
if (i < total) i++
|
||||
})
|
||||
await file.append(clusterLog, JSON.stringify({ [item._id]: programs }) + '\n')
|
||||
await file.append(
|
||||
clusterLog,
|
||||
JSON.stringify({ _id: item._id, site: config.site, programs }) + '\n'
|
||||
)
|
||||
}
|
||||
|
||||
logger.info(`Done in ${timer.format('HH[h] mm[m] ss[s]')}`)
|
||||
|
|
|
@ -2,4 +2,4 @@ exports.db = require('./db')
|
|||
exports.logger = require('./logger')
|
||||
exports.file = require('./file')
|
||||
exports.parser = require('./parser')
|
||||
exports.timer = require('./timer')
|
||||
exports.timer = require('./timer')
|
||||
|
|
1384
scripts/data/countries.json
Normal file
1384
scripts/data/countries.json
Normal file
File diff suppressed because it is too large
Load diff
127493
scripts/output/channels.json
Normal file
127493
scripts/output/channels.json
Normal file
File diff suppressed because it is too large
Load diff
13000
scripts/output/programs.json
Normal file
13000
scripts/output/programs.json
Normal file
File diff suppressed because it is too large
Load diff
Loading…
Add table
Add a link
Reference in a new issue