2020-04-11 04:33:53 +03:00
|
|
|
const fs = require('fs')
|
2019-07-20 10:03:31 +03:00
|
|
|
const path = require('path')
|
2019-08-09 13:52:54 +03:00
|
|
|
const axios = require('axios')
|
2020-04-11 04:33:53 +03:00
|
|
|
const zlib = require('zlib')
|
2019-08-07 17:28:11 +03:00
|
|
|
const urlParser = require('url')
|
2019-11-02 12:45:09 +03:00
|
|
|
const escapeStringRegexp = require('escape-string-regexp')
|
|
|
|
const markdownInclude = require('markdown-include')
|
2020-04-18 17:16:56 +03:00
|
|
|
const iso6393 = require('iso-639-3')
|
2019-08-08 03:29:36 +03:00
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
const utils = {}
|
|
|
|
|
|
|
|
utils.supportedCategories = [
|
|
|
|
{
|
|
|
|
name: 'Auto',
|
|
|
|
id: 'auto',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Business',
|
|
|
|
id: 'business',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Classic',
|
|
|
|
id: 'classic',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Comedy',
|
|
|
|
id: 'comedy',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Documentary',
|
|
|
|
id: 'documentary',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Education',
|
|
|
|
id: 'education',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Entertainment',
|
|
|
|
id: 'entertainment',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Family',
|
|
|
|
id: 'family',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Fashion',
|
|
|
|
id: 'fashion',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Food',
|
|
|
|
id: 'food',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'General',
|
|
|
|
id: 'general',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Health',
|
|
|
|
id: 'health',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'History',
|
|
|
|
id: 'history',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Hobby',
|
|
|
|
id: 'hobby',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Kids',
|
|
|
|
id: 'kids',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Legislative',
|
|
|
|
id: 'legislative',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Lifestyle',
|
|
|
|
id: 'lifestyle',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Local',
|
|
|
|
id: 'local',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Movies',
|
|
|
|
id: 'movies',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Music',
|
|
|
|
id: 'music',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'News',
|
|
|
|
id: 'news',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Quiz',
|
|
|
|
id: 'quiz',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Religious',
|
|
|
|
id: 'religious',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Sci-Fi',
|
|
|
|
id: 'sci-fi',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Shop',
|
|
|
|
id: 'shop',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Sport',
|
|
|
|
id: 'sport',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Travel',
|
|
|
|
id: 'travel',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Weather',
|
|
|
|
id: 'weather',
|
|
|
|
nsfw: false
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'XXX',
|
|
|
|
id: 'xxx',
|
|
|
|
nsfw: true
|
|
|
|
}
|
|
|
|
]
|
2019-08-07 15:44:12 +03:00
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.code2flag = function (code) {
|
2020-05-06 18:01:36 +03:00
|
|
|
switch (code) {
|
|
|
|
case 'uk':
|
|
|
|
return '🇬🇧'
|
|
|
|
case 'int':
|
|
|
|
return '🌎'
|
2020-09-17 23:30:28 +03:00
|
|
|
case 'unsorted':
|
|
|
|
return ''
|
2020-05-06 18:01:36 +03:00
|
|
|
default:
|
|
|
|
return code
|
|
|
|
.toUpperCase()
|
|
|
|
.replace(/./g, char => String.fromCodePoint(char.charCodeAt(0) + 127397))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.code2name = function (code) {
|
|
|
|
switch (code.toLowerCase()) {
|
|
|
|
case 'int':
|
|
|
|
return 'International'
|
|
|
|
case 'us':
|
|
|
|
return 'United States'
|
|
|
|
}
|
|
|
|
|
2021-01-30 01:58:52 +03:00
|
|
|
const intlDisplayNames = new Intl.DisplayNames(['en'], {
|
2021-01-30 06:16:44 +03:00
|
|
|
style: 'narrow',
|
2021-01-30 01:58:52 +03:00
|
|
|
type: 'region'
|
|
|
|
})
|
|
|
|
|
|
|
|
try {
|
|
|
|
return intlDisplayNames.of(code.toUpperCase())
|
|
|
|
} catch (e) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.language2code = function (name) {
|
2021-01-30 01:58:52 +03:00
|
|
|
const lang = iso6393.find(l => l.name === name)
|
|
|
|
|
|
|
|
return lang && lang.iso6393 ? lang.iso6393 : null
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.sortBy = function (arr, fields) {
|
2019-11-02 15:43:41 +03:00
|
|
|
return arr.sort((a, b) => {
|
2020-04-11 04:33:53 +03:00
|
|
|
for (let field of fields) {
|
2020-05-04 16:23:03 +03:00
|
|
|
let propA = a[field] ? a[field].toLowerCase() : ''
|
|
|
|
let propB = b[field] ? b[field].toLowerCase() : ''
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
if (propA === 'undefined') {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
if (propB === 'undefined') {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
if (propA === 'other') {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
if (propB === 'other') {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
2020-05-04 16:23:03 +03:00
|
|
|
if (propA < propB) {
|
2020-04-11 04:33:53 +03:00
|
|
|
return -1
|
|
|
|
}
|
2020-05-04 16:23:03 +03:00
|
|
|
if (propA > propB) {
|
2020-04-11 04:33:53 +03:00
|
|
|
return 1
|
|
|
|
}
|
2019-11-02 15:43:41 +03:00
|
|
|
}
|
|
|
|
return 0
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.loadEPG = function (url) {
|
2019-08-07 16:51:34 +03:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
var buffer = []
|
2019-08-09 13:52:54 +03:00
|
|
|
axios({
|
|
|
|
method: 'get',
|
|
|
|
url: url,
|
2019-11-03 19:00:57 +03:00
|
|
|
responseType: 'stream',
|
2020-05-04 16:23:03 +03:00
|
|
|
timeout: 60000
|
2020-04-11 04:33:53 +03:00
|
|
|
})
|
2020-05-04 16:23:03 +03:00
|
|
|
.then(res => {
|
2020-04-11 04:33:53 +03: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)
|
|
|
|
})
|
|
|
|
})
|
2020-05-04 16:23:03 +03:00
|
|
|
.catch(e => {
|
2019-08-07 16:51:34 +03:00
|
|
|
reject(e)
|
|
|
|
})
|
2019-08-07 15:44:12 +03:00
|
|
|
})
|
2019-07-20 10:03:31 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.getBasename = function (filename) {
|
2021-01-30 01:58:52 +03:00
|
|
|
return path.basename(filename, path.extname(filename))
|
2019-08-07 16:51:34 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.filterPlaylists = function (arr, include = '', exclude = '') {
|
2021-01-30 01:58:52 +03:00
|
|
|
if (include) {
|
|
|
|
const included = include.split(',').map(filename => `channels/${filename}.m3u`)
|
2019-07-20 10:03:31 +03:00
|
|
|
|
2021-01-30 01:58:52 +03:00
|
|
|
return arr.filter(i => included.indexOf(i.url) > -1)
|
|
|
|
}
|
2019-07-20 10:03:31 +03:00
|
|
|
|
2021-01-30 01:58:52 +03:00
|
|
|
if (exclude) {
|
|
|
|
const excluded = exclude.split(',').map(filename => `channels/${filename}.m3u`)
|
2019-08-07 17:28:11 +03:00
|
|
|
|
2021-01-30 01:58:52 +03:00
|
|
|
return arr.filter(i => excluded.indexOf(i.url) === -1)
|
|
|
|
}
|
2019-08-07 17:28:11 +03:00
|
|
|
|
2021-01-30 01:58:52 +03:00
|
|
|
return arr
|
2019-08-07 17:28:11 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.generateTable = function (data, options) {
|
2019-11-03 20:07:43 +03:00
|
|
|
let output = '<table>\n'
|
2019-10-23 09:38:32 +03:00
|
|
|
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '\t<thead>\n\t\t<tr>'
|
2019-10-23 09:38:32 +03:00
|
|
|
for (let column of options.columns) {
|
|
|
|
output += `<th align="${column.align}">${column.name}</th>`
|
|
|
|
}
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '</tr>\n\t</thead>\n'
|
2019-10-23 09:38:32 +03:00
|
|
|
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '\t<tbody>\n'
|
2019-10-23 09:38:32 +03:00
|
|
|
for (let item of data) {
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '\t\t<tr>'
|
2019-10-23 09:38:32 +03: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++
|
|
|
|
}
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '</tr>\n'
|
2019-10-23 09:38:32 +03:00
|
|
|
}
|
2019-11-03 20:07:43 +03:00
|
|
|
output += '\t</tbody>\n'
|
2019-10-23 09:38:32 +03:00
|
|
|
|
|
|
|
output += '</table>'
|
|
|
|
|
|
|
|
return output
|
2019-10-09 05:33:52 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.createDir = function (dir) {
|
2021-01-30 01:58:52 +03:00
|
|
|
if (!fs.existsSync(dir)) {
|
|
|
|
fs.mkdirSync(dir)
|
|
|
|
}
|
2019-11-02 17:30:12 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.readFile = function (filename) {
|
2021-01-30 01:58:52 +03:00
|
|
|
return fs.readFileSync(path.resolve(__dirname) + `/../${filename}`, { encoding: 'utf8' })
|
2019-11-03 20:21:35 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.appendToFile = function (filename, data) {
|
2021-01-30 01:58:52 +03:00
|
|
|
fs.appendFileSync(path.resolve(__dirname) + '/../' + filename, data)
|
|
|
|
}
|
2020-04-25 19:58:32 +03:00
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.compileMarkdown = function (filepath) {
|
2021-01-30 01:58:52 +03:00
|
|
|
return markdownInclude.compileFiles(path.resolve(__dirname, filepath))
|
|
|
|
}
|
2020-04-25 19:58:32 +03:00
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.escapeStringRegexp = function (scring) {
|
2021-01-30 01:58:52 +03:00
|
|
|
return escapeStringRegexp(string)
|
|
|
|
}
|
2020-04-25 19:58:32 +03:00
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.createFile = function (filename, data = '') {
|
2021-01-30 01:58:52 +03:00
|
|
|
fs.writeFileSync(path.resolve(__dirname) + '/../' + filename, data)
|
2020-04-25 19:58:32 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.writeToLog = function (country, msg, url) {
|
2021-01-30 01:58:52 +03:00
|
|
|
var now = new Date()
|
|
|
|
var line = `${country}: ${msg} '${url}'`
|
|
|
|
this.appendToFile('error.log', now.toISOString() + ' ' + line + '\n')
|
2020-05-04 16:23:03 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.filterNSFW = function (arr) {
|
|
|
|
const sfwCategories = utils.supportedCategories.filter(c => !c.nsfw).map(c => c.name)
|
2021-01-27 18:54:22 +03:00
|
|
|
|
|
|
|
return arr.filter(i => sfwCategories.includes(i.category))
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
utils.sleep = function (ms) {
|
2021-01-20 01:02:24 +03:00
|
|
|
return function (x) {
|
|
|
|
return new Promise(resolve => setTimeout(() => resolve(x), ms))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-30 06:16:44 +03:00
|
|
|
module.exports = utils
|