iptv/scripts/helper.js

307 lines
7.0 KiB
JavaScript
Raw Normal View History

2020-04-11 03:33:53 +02:00
const fs = require('fs')
2019-07-20 09:03:31 +02:00
const path = require('path')
2019-11-02 10:54:11 +01:00
const playlistParser = require('iptv-playlist-parser')
const axios = require('axios')
2020-04-11 03:33:53 +02:00
const zlib = require('zlib')
2019-10-30 16:44:26 +01:00
const epgParser = require('epg-parser')
const urlParser = require('url')
2019-11-02 10:45:09 +01:00
const escapeStringRegexp = require('escape-string-regexp')
const markdownInclude = require('markdown-include')
2020-04-18 16:16:56 +02:00
const iso6393 = require('iso-639-3')
let cache = {}
2019-11-02 10:45:09 +01:00
let helper = {}
2020-04-11 03:33:53 +02:00
helper.sortBy = function (arr, fields) {
2019-11-02 13:43:41 +01:00
return arr.sort((a, b) => {
2020-04-11 03:33:53 +02:00
for (let field of fields) {
if (a[field].toLowerCase() < b[field].toLowerCase()) {
return -1
}
if (a[field].toLowerCase() > b[field].toLowerCase()) {
return 1
}
2019-11-02 13:43:41 +01:00
}
return 0
})
}
2020-04-11 03:33:53 +02:00
helper.createDir = function (dir) {
2019-11-02 11:26:21 +01:00
if (!fs.existsSync(dir)) {
fs.mkdirSync(dir)
}
}
2020-04-11 03:33:53 +02:00
helper.compileMarkdown = function (filepath) {
2019-11-02 10:45:09 +01:00
return markdownInclude.compileFiles(path.resolve(__dirname, filepath))
}
2019-07-20 09:03:31 +02:00
2020-04-11 03:33:53 +02:00
helper.escapeStringRegexp = function (scring) {
2019-11-02 10:45:09 +01:00
return escapeStringRegexp(string)
}
2019-11-01 15:15:33 +01:00
2020-04-11 03:33:53 +02:00
helper.getISO6391Name = function (code) {
2020-04-18 16:16:56 +02:00
const lang = iso6393.find((l) => l.iso6393 === code.toLowerCase())
return lang && lang.name ? lang.name : null
2019-11-02 10:45:09 +01:00
}
2020-04-11 03:33:53 +02:00
helper.getISO6391Code = function (name) {
2020-04-18 16:16:56 +02:00
const lang = iso6393.find((l) => l.name === name)
return lang && lang.iso6393 ? lang.iso6393 : null
2019-07-20 09:03:31 +02:00
}
2020-04-11 03:33:53 +02:00
helper.parsePlaylist = function (filename) {
2019-11-02 10:54:11 +01:00
const content = this.readFile(filename)
const result = playlistParser.parse(content)
2019-08-07 15:51:34 +02:00
2019-10-07 01:45:06 +02:00
return new Playlist(result)
2019-07-20 09:03:31 +02:00
}
2020-04-11 03:33:53 +02:00
helper.parseEPG = async function (url) {
2019-11-03 14:03:35 +01:00
const content = await this.getEPG(url)
2019-10-30 16:44:26 +01:00
const result = epgParser.parse(content)
const channels = {}
2020-04-11 03:33:53 +02:00
for (let channel of result.channels) {
channels[channel.id] = channel
}
2020-04-11 03:33:53 +02:00
return Promise.resolve({
url,
channels,
})
}
2020-04-11 03:33:53 +02:00
helper.getEPG = function (url) {
2019-08-07 15:51:34 +02:00
return new Promise((resolve, reject) => {
var buffer = []
axios({
method: 'get',
url: url,
2019-11-03 17:00:57 +01:00
responseType: 'stream',
timeout: 60000,
2020-04-11 03:33:53 +02:00
})
.then((res) => {
2020-04-11 03:33:53 +02:00
let stream
if (/\.gz$/i.test(url)) {
let gunzip = zlib.createGunzip()
res.data.pipe(gunzip)
stream = gunzip
} else {
stream = res.data
}
stream
.on('data', function (data) {
buffer.push(data.toString())
})
.on('end', function () {
resolve(buffer.join(''))
})
.on('error', function (e) {
reject(e)
})
})
.catch((e) => {
2019-08-07 15:51:34 +02:00
reject(e)
})
})
2019-07-20 09:03:31 +02:00
}
2020-04-11 03:33:53 +02:00
helper.readFile = function (filename) {
return fs.readFileSync(path.resolve(__dirname) + `/../${filename}`, { encoding: 'utf8' })
2019-08-07 15:51:34 +02:00
}
2020-04-11 03:33:53 +02:00
helper.appendToFile = function (filename, data) {
2019-07-20 09:32:57 +02:00
fs.appendFileSync(path.resolve(__dirname) + '/../' + filename, data)
2019-07-20 09:03:31 +02:00
}
2020-04-11 03:33:53 +02:00
helper.createFile = function (filename, data = '') {
2019-07-20 09:32:57 +02:00
fs.writeFileSync(path.resolve(__dirname) + '/../' + filename, data)
2019-07-20 09:03:31 +02:00
}
2020-04-11 03:33:53 +02:00
helper.getBasename = function (filename) {
return path.basename(filename, path.extname(filename))
}
2020-04-11 03:33:53 +02:00
helper.getUrlPath = function (u) {
let parsed = urlParser.parse(u)
let searchQuery = parsed.search || ''
2019-09-07 23:51:06 +02:00
let path = parsed.host + parsed.pathname + searchQuery
2019-09-07 23:51:06 +02:00
return path.toLowerCase()
}
2020-04-11 03:33:53 +02:00
helper.generateTable = function (data, options) {
let output = '<table>\n'
2019-10-23 08:38:32 +02:00
output += '\t<thead>\n\t\t<tr>'
2019-10-23 08:38:32 +02:00
for (let column of options.columns) {
output += `<th align="${column.align}">${column.name}</th>`
}
output += '</tr>\n\t</thead>\n'
2019-10-23 08:38:32 +02:00
output += '\t<tbody>\n'
2019-10-23 08:38:32 +02:00
for (let item of data) {
output += '\t\t<tr>'
2019-10-23 08:38:32 +02:00
let i = 0
for (let prop in item) {
const column = options.columns[i]
let nowrap = column.nowrap
let align = column.align
output += `<td align="${align}"${nowrap ? ' nowrap' : ''}>${item[prop]}</td>`
i++
}
output += '</tr>\n'
2019-10-23 08:38:32 +02:00
}
output += '\t</tbody>\n'
2019-10-23 08:38:32 +02:00
output += '</table>'
return output
2019-10-09 04:33:52 +02:00
}
2020-04-11 03:33:53 +02:00
helper.createChannel = function (data) {
return new Channel(data)
2019-11-02 15:30:12 +01:00
}
2020-04-11 03:33:53 +02:00
helper.writeToLog = function (country, msg, url) {
2019-11-03 18:21:35 +01:00
var now = new Date()
var line = `${country}: ${msg} '${url}'`
this.appendToFile('error.log', now.toISOString() + ' ' + line + '\n')
}
2020-04-11 03:33:53 +02:00
helper.parseMessage = function (err, u) {
if (!err || !err.message) return
2019-11-03 18:21:35 +01:00
const msgArr = err.message.split('\n')
2020-04-11 03:33:53 +02:00
if (msgArr.length === 0) return
2019-11-03 18:21:35 +01:00
const line = msgArr.find((line) => {
2019-11-03 18:21:35 +01:00
return line.indexOf(u) === 0
})
2020-04-11 03:33:53 +02:00
if (!line) return
2019-11-03 18:21:35 +01:00
return line.replace(`${u}: `, '')
}
2019-11-02 10:45:09 +01:00
class Playlist {
constructor(data) {
this.header = data.header
this.items = data.items
2019-11-03 14:03:35 +01:00
this.changed = false
2019-11-02 10:45:09 +01:00
}
getHeader() {
let parts = ['#EXTM3U']
2020-04-11 03:33:53 +02:00
for (let key in this.header.attrs) {
2019-11-02 10:45:09 +01:00
let value = this.header.attrs[key]
2020-04-11 03:33:53 +02:00
if (value) {
2019-11-02 10:45:09 +01:00
parts.push(`${key}="${value}"`)
}
}
return `${parts.join(' ')}\n`
}
}
class Channel {
constructor(data) {
this.id = data.tvg.id
this.name = data.tvg.name
this.language = data.tvg.language
.split(';')
.filter((l) => !!helper.getISO6391Code(l))
.join(';')
this.logo = data.tvg.logo
this.group = this._filterGroup(data.group.title)
2019-11-02 10:45:09 +01:00
this.url = data.url
this.title = data.name.trim()
this.userAgent = data.http['user-agent']
this.referrer = data.http['referrer']
2019-11-02 10:45:09 +01:00
}
_filterGroup(groupTitle) {
2020-04-11 03:33:53 +02:00
if (!groupTitle) return ''
const supportedCategories = [
'Auto',
'Business',
'Classic',
'Comedy',
'Documentary',
'Education',
'Entertainment',
'Family',
'Fashion',
'Food',
'General',
'Health',
'History',
'Hobby',
'Kids',
'Legislative',
'Lifestyle',
'Local',
'Movies',
'Music',
'News',
'Quiz',
'Religious',
'Sci-Fi',
'Shop',
'Sport',
'Travel',
'Weather',
'XXX',
2020-04-11 03:33:53 +02:00
]
const groupIndex = supportedCategories
.map((g) => g.toLowerCase())
2020-04-11 03:33:53 +02:00
.indexOf(groupTitle.toLowerCase())
if (groupIndex === -1) {
2019-11-02 10:45:09 +01:00
groupTitle = ''
} else {
groupTitle = supportedCategories[groupIndex]
}
return groupTitle
}
toString() {
2019-11-02 15:21:19 +01:00
const country = this.countryCode.toUpperCase()
2020-04-11 03:33:53 +02:00
const epg = this.id && this.epg ? this.epg : ''
let info = `-1 tvg-id="${this.id}" tvg-name="${this.name}" tvg-language="${this.language}" tvg-logo="${this.logo}" tvg-country="${country}" tvg-url="${epg}" group-title="${this.group}",${this.title}`
if (this.referrer) {
info += `\n#EXTVLCOPT:http-referrer=${this.referrer}`
}
if (this.userAgent) {
info += `\n#EXTVLCOPT:http-user-agent=${this.userAgent}`
}
2019-11-02 15:21:19 +01:00
return '#EXTINF:' + info + '\n' + this.url + '\n'
}
toShortString() {
let info = `-1 tvg-id="${this.id}" tvg-name="${this.name}" tvg-language="${this.language}" tvg-logo="${this.logo}" group-title="${this.group}",${this.title}`
if (this.referrer) {
info += `\n#EXTVLCOPT:http-referrer=${this.referrer}`
}
if (this.userAgent) {
info += `\n#EXTVLCOPT:http-user-agent=${this.userAgent}`
}
2019-11-02 10:45:09 +01:00
return '#EXTINF:' + info + '\n' + this.url + '\n'
}
}
2020-04-11 03:33:53 +02:00
module.exports = helper