This commit is contained in:
Aleksandr Statciuk 2022-01-09 16:09:19 +03:00
parent fa8549c3e9
commit d068f5083f
11 changed files with 142749 additions and 5 deletions

View 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)
})
}
}

View 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, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
.replace(/"/g, '&quot;')
.replace(/'/g, '&apos;')
.replace(/\n|\r/g, ' ')
.replace(/ +/g, ' ')
.trim()
}

View file

@ -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]')}`)

View file

@ -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

File diff suppressed because it is too large Load diff

127493
scripts/output/channels.json Normal file

File diff suppressed because it is too large Load diff

13000
scripts/output/programs.json Normal file

File diff suppressed because it is too large Load diff