Files
iptv/scripts/commands/playlist/update.ts

145 lines
4.6 KiB
TypeScript
Raw Normal View History

2023-09-22 05:17:22 +03:00
import { Logger, Storage, Collection, Dictionary } from '@freearhey/core'
2023-09-17 04:08:50 +03:00
import { DATA_DIR, STREAMS_DIR } from '../../constants'
2023-09-22 05:17:22 +03:00
import { IssueLoader, PlaylistParser } from '../../core'
2023-09-17 04:08:50 +03:00
import { Stream, Playlist, Channel, Issue } from '../../models'
import validUrl from 'valid-url'
2023-09-15 18:40:35 +03:00
let processedIssues = new Collection()
let streams: Collection
let groupedChannels: Dictionary
2025-02-27 20:55:39 +03:00
let issues: Collection
2023-09-15 18:40:35 +03:00
async function main() {
const logger = new Logger({ disabled: true })
const loader = new IssueLoader()
2025-02-27 20:55:39 +03:00
logger.info('loading issues...')
issues = await loader.load()
2023-09-17 04:08:50 +03:00
logger.info('loading channels from api...')
const dataStorage = new Storage(DATA_DIR)
const channelsContent = await dataStorage.json('channels.json')
2023-09-15 18:40:35 +03:00
groupedChannels = new Collection(channelsContent)
.map(data => new Channel(data))
.keyBy((channel: Channel) => channel.id)
2023-09-17 04:08:50 +03:00
logger.info('loading streams...')
const streamsStorage = new Storage(STREAMS_DIR)
const parser = new PlaylistParser({ storage: streamsStorage })
const files = await streamsStorage.list('**/*.m3u')
streams = await parser.parse(files)
2023-09-15 18:40:35 +03:00
logger.info('removing broken streams...')
await removeStreams(loader)
logger.info('edit stream description...')
await editStreams(loader)
logger.info('add new streams...')
await addStreams(loader)
logger.info('saving...')
const groupedStreams = streams.groupBy((stream: Stream) => stream.filepath)
for (let filepath of groupedStreams.keys()) {
2023-09-18 01:51:20 +03:00
let streams = groupedStreams.get(filepath) || []
streams = streams.filter((stream: Stream) => stream.removed === false)
2023-09-15 18:40:35 +03:00
const playlist = new Playlist(streams, { public: false })
await streamsStorage.save(filepath, playlist.toString())
}
const output = processedIssues.map(issue_number => `closes #${issue_number}`).join(', ')
console.log(`OUTPUT=${output}`)
}
main()
async function removeStreams(loader: IssueLoader) {
2025-02-27 20:55:39 +03:00
const requests = issues.filter(
issue => issue.labels.includes('streams:remove') && issue.labels.includes('approved')
)
requests.forEach((issue: Issue) => {
2023-09-17 04:08:50 +03:00
const data = issue.data
2025-03-09 19:53:25 +03:00
if (data.missing('brokenLinks')) return
2023-09-15 18:40:35 +03:00
2025-03-09 19:53:25 +03:00
const brokenLinks = data.getString('brokenLinks').split(/\r?\n/).filter(Boolean)
2024-12-26 03:10:36 +03:00
let changed = false
brokenLinks.forEach(link => {
const found: Stream = streams.first((_stream: Stream) => _stream.url === link.trim())
if (found) {
found.removed = true
changed = true
}
})
if (changed) processedIssues.add(issue.number)
2023-09-15 18:40:35 +03:00
})
}
async function editStreams(loader: IssueLoader) {
2025-02-27 20:55:39 +03:00
const requests = issues.filter(
issue => issue.labels.includes('streams:edit') && issue.labels.includes('approved')
)
requests.forEach((issue: Issue) => {
2023-09-17 04:08:50 +03:00
const data = issue.data
2025-03-09 19:53:25 +03:00
if (data.missing('streamUrl')) return
2023-09-15 18:40:35 +03:00
let stream = streams.first(
2025-03-09 19:53:25 +03:00
(_stream: Stream) => _stream.url === data.getString('streamUrl')
2023-09-15 18:40:35 +03:00
) as Stream
if (!stream) return
2025-03-09 19:53:25 +03:00
if (data.has('channelId')) {
const channel = groupedChannels.get(data.getString('channelId'))
2023-09-15 18:40:35 +03:00
if (!channel) return
2025-03-09 19:53:25 +03:00
stream.channel = data.getString('channelId')
2023-09-15 18:40:35 +03:00
stream.filepath = `${channel.country.toLowerCase()}.m3u`
stream.line = -1
stream.name = channel.name
}
2025-02-22 12:54:00 +03:00
if (data.has('label')) stream.label = data.getString('label')
if (data.has('quality')) stream.quality = data.getString('quality')
2025-03-09 19:53:25 +03:00
if (data.has('httpUserAgent')) stream.httpUserAgent = data.getString('httpUserAgent')
if (data.has('httpReferrer')) stream.httpReferrer = data.getString('httpReferrer')
2023-09-15 18:40:35 +03:00
2023-09-17 04:08:50 +03:00
processedIssues.add(issue.number)
2023-09-15 18:40:35 +03:00
})
}
async function addStreams(loader: IssueLoader) {
2025-02-27 20:55:39 +03:00
const requests = issues.filter(
issue => issue.labels.includes('streams:add') && issue.labels.includes('approved')
)
requests.forEach((issue: Issue) => {
2023-09-17 04:08:50 +03:00
const data = issue.data
2025-03-09 19:53:25 +03:00
if (data.missing('channelId') || data.missing('streamUrl')) return
if (streams.includes((_stream: Stream) => _stream.url === data.getString('streamUrl'))) return
if (!validUrl.isUri(data.getString('streamUrl'))) return
2023-09-15 18:40:35 +03:00
2025-03-09 19:53:25 +03:00
const channel = groupedChannels.get(data.getString('channelId'))
2023-09-15 18:40:35 +03:00
if (!channel) return
const stream = new Stream({
2025-03-09 19:53:25 +03:00
channel: data.getString('channelId'),
url: data.getString('streamUrl'),
2025-02-22 12:54:00 +03:00
label: data.getString('label'),
quality: data.getString('quality'),
2025-03-09 19:53:25 +03:00
httpUserAgent: data.getString('httpUserAgent'),
httpReferrer: data.getString('httpReferrer'),
2023-09-15 18:40:35 +03:00
filepath: `${channel.country.toLowerCase()}.m3u`,
line: -1,
2025-03-09 19:53:25 +03:00
name: data.getString('channelName') || channel.name
2023-09-15 18:40:35 +03:00
})
streams.add(stream)
2023-09-17 04:08:50 +03:00
processedIssues.add(issue.number)
2023-09-15 18:40:35 +03:00
})
}