radiomanifest.js/radiomanifest.js

342 lines
8 KiB
JavaScript
Raw Normal View History

2021-11-18 15:35:32 +01:00
const fetch = require('isomorphic-unfetch')
2022-01-30 01:06:35 +01:00
const shows = require('./shows.js')
2022-01-30 02:23:11 +01:00
const calendar = require('./calendar.js')
2021-11-18 15:35:32 +01:00
2021-11-25 22:33:17 +01:00
function getStreaminfoUrl (siteurl) {
return siteurl + '/streaminfo.json' // XXX: improve this logic
}
function getManifestUrl (siteurl) {
return siteurl + '/radiomanifest.xml' // XXX: improve this logic
}
2021-11-18 15:35:32 +01:00
2021-12-26 20:00:17 +01:00
function getAttribute(el, attr, default_value) {
if(el.hasAttribute(attr))
return el.getAttribute(attr);
return default_value;
}
2022-01-30 14:01:38 +01:00
/**
* Represents everything we know about a radio. This includes, but is not limited to, radiomanifest.xml.
*/
2021-11-25 22:33:17 +01:00
class Radio {
2022-01-30 14:01:38 +01:00
/**
* @param {Array} [sources] optional
* @param {string} [scheduleURL]
* @param {string} [showsURL]
* @param {string} [feed]
*/
2022-01-30 02:23:11 +01:00
constructor (sources, scheduleURL, showsURL, feed) {
2021-11-25 22:33:17 +01:00
this.streaming = new RadioStreaming(sources)
2022-01-30 02:23:11 +01:00
this.scheduleURL = scheduleURL
2022-01-30 00:26:07 +01:00
this.showsURL = showsURL
2021-11-25 22:33:17 +01:00
this.feed = feed
this.name = ''
2022-01-30 14:49:56 +01:00
this.description = ''
this.logo = null
2022-01-30 00:26:07 +01:00
this.shows = []
2022-01-30 02:23:11 +01:00
this.schedule = null
2021-11-18 15:35:32 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* @returns {RadioStreaming}
*/
2021-11-25 22:33:17 +01:00
getStreaming () {
return this.streaming
2021-11-18 15:35:32 +01:00
}
2021-11-25 22:33:17 +01:00
setName (name) {
this.name = name
}
2021-11-18 15:35:32 +01:00
2022-01-30 14:49:56 +01:00
/**
* The radio name, as inferred by stream-meta
*
* @returns {string}
*/
getName() {
return this.name
}
setDescription(desc) {
this.description = desc
}
/**
* The description of the radio, as inferred by stream-meta
*
* @returns {string}
*/
getDescription() {
return this.description
}
setLogo(logo) {
this.logo = logo
}
/**
* @returns {string} the URL of the logo, or `null` if not found
*/
getLogo() {
return this.logo
}
2022-01-30 14:01:38 +01:00
/**
*
* @returns {Array<RadioShow>}
*/
2022-01-30 00:26:07 +01:00
getShows() {
return this.shows
}
2022-01-30 14:01:38 +01:00
/**
* @returns {RadioShow} The lookup is exact and case-sensitive. If no such show can be found, `null`
* is returned.
*/
2021-11-25 22:33:17 +01:00
getShowByName (showName) {
2022-01-30 02:23:11 +01:00
if (this.shows === undefined) return null
2022-01-30 00:26:07 +01:00
return this.shows.find(s => s.name === showName)
2021-11-18 15:35:32 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* @returns {RadioSchedule} If no schedule is present, `null` is returned.
*/
2021-11-18 15:35:32 +01:00
getSchedule () {
2022-01-30 02:23:11 +01:00
return this.schedule
2021-11-18 15:35:32 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* Find if a show is running at the given moment. If there's none, `null` is returned.
* If possible, a complete {@link RadioShow} including full informations (from shows.xml) is returned.
* If, instead, we know from the `schedule` that there must be a show, but have no additional detail, a
* {@link RadioShow} object will be created on the fly.
*
* @param {ICAL.Time} [now] If omitted, the current time is used.
* @returns {RadioShow} If we don't know what's going on at the given moment, `null` is returned.
*/
2022-01-30 02:38:37 +01:00
getShowAtTime (now) {
if (this.schedule === undefined || this.schedule === null) return null
return this.getSchedule().getNowShow(now)
2021-11-18 15:35:32 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* This static method can create a Radio object from a valid radiomanifest.xml file
*
* @param xml An already parsed xml block
* @returns {Radio}
*/
2021-11-25 22:33:17 +01:00
static fromDOM (xml) {
const doc = xml.cloneNode(true)
let res = doc.evaluate('/radio-manifest/streaming/source', doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null)
2021-11-25 22:33:17 +01:00
const sources = []
for (let i = 0; i < res.snapshotLength; i++) {
const src = res.snapshotItem(i)
2021-11-25 22:33:17 +01:00
if (!src.hasAttribute('priority')) {
src.setAttribute('priority', '0')
} else if (parseInt(src.getAttribute('priority'), 10) < 0) {
continue
}
sources.push(src)
}
res = doc.evaluate('/radio-manifest/schedule', doc)
2021-11-25 22:33:17 +01:00
const scheduleEl = res.iterateNext()
2022-01-30 02:23:11 +01:00
let scheduleURL = null
2021-11-25 22:33:17 +01:00
if (scheduleEl !== null) {
2022-01-30 02:23:11 +01:00
scheduleURL = scheduleEl.getAttribute('src')
2021-11-16 23:22:16 +01:00
}
2021-11-25 22:33:17 +01:00
res = xml.evaluate('/radio-manifest/shows', xml)
const showsEl = res.iterateNext()
2022-01-30 00:26:07 +01:00
let showsURL = null
2021-11-25 22:33:17 +01:00
if (showsEl !== null) {
2022-01-30 00:26:07 +01:00
showsURL = showsEl.getAttribute('src')
2021-11-25 22:33:17 +01:00
}
2021-11-16 23:22:16 +01:00
2021-11-25 22:33:17 +01:00
res = xml.evaluate('/radio-manifest/feed', xml)
const feedEl = res.iterateNext()
let feed = null
if (feedEl !== null) {
feed = feedEl.getAttribute('src')
}
2021-11-16 23:22:16 +01:00
2022-01-30 02:23:11 +01:00
const manifest = new Radio(sources, scheduleURL, showsURL, feed)
2021-11-25 22:33:17 +01:00
return manifest
}
2021-11-16 23:22:16 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* Represents the streaming capabilities of a radio.
* This has probably been derived from the <streaming> block in radiomanifest.xml
*/
2021-12-26 20:00:17 +01:00
class RadioStreaming {
constructor (sources) {
this.sources = sources.sort(
(a,b) => this.getPriority(a) < this.getPriority(a)
)
}
2021-11-19 18:08:09 +01:00
2022-01-30 14:01:38 +01:00
/**
* Get the list of possible options that are provided to the user
* @returns {Array<string>}
*/
2021-12-26 20:00:17 +01:00
getOptions() {
return this.sources.map(function (x) {
return x.getAttribute('name')
})
}
2022-01-30 14:01:38 +01:00
/**
* @private
*/
2021-12-26 20:00:17 +01:00
getPriority(element) {
return parseInt(getAttribute(element, 'priority', '1'))
}
2022-01-30 14:01:38 +01:00
/**
* @private
*/
2021-12-26 20:00:17 +01:00
getTopPrioritySources() {
var topPriority = this.getPriority(this.sources[0])
return this.sources.filter(
(src) => parseInt(src.getAttribute('priority'), 10) === topPriority
)
}
2022-01-30 14:01:38 +01:00
/**
* @return {string} url of the source. Note that this is much probably a playlist, in M3U format
*/
2021-12-26 20:00:17 +01:00
getSource(name) {
if (name === undefined) {
return this.getTopPrioritySources()[0]
}
const s = this.sources.find(function (x) {
return x.getAttribute('name') === name
})
if (s === undefined) return s
return s.getAttribute('src')
}
2022-01-30 14:01:38 +01:00
/**
* This is your go-to function whenever you need a list of URLs to play.
* They will be picked honoring priorities, and expanding the playlist source
* @return {Array<string>}
*/
2021-12-26 20:00:17 +01:00
async pickURLs() {
var allSources = this.getTopPrioritySources()
var allAudios = []
for(let src of allSources) {
let url = src.getAttribute('src')
let resp = await fetch(url)
allAudios.unshift(... parseM3U(await resp.text()))
}
return allAudios
2021-11-18 15:35:32 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* Just like {@link RadioStreaming#pickURLs}, but get a single URL
* @return {string}
*/
async pickURL() {
var allAudios = await this.pickURLs()
return allAudios[0]
}
2021-11-16 23:22:16 +01:00
}
2022-01-30 14:01:38 +01:00
/**
* Create everything you need - **you should start from here**
*
* @param {string} siteurl URL of website you want to load
* @param {Object} options options. Currenly unused
* @return {Radio}
*/
2021-11-19 18:08:09 +01:00
async function get (siteurl, options) {
let resp = await fetch(getManifestUrl(siteurl))
2021-11-18 15:35:32 +01:00
let text = await resp.text()
2021-11-19 18:08:09 +01:00
const parser = new DOMParser()
const dom = parser.parseFromString(text, 'text/xml')
2021-11-25 22:33:17 +01:00
const manifest = Radio.fromDOM(dom)
2021-11-19 18:08:09 +01:00
2022-01-30 00:26:07 +01:00
try {
2022-01-30 01:06:35 +01:00
manifest.shows = await shows.get(manifest)
2022-01-30 00:26:07 +01:00
} catch (e) {
console.error("Error while fetching shows file", e)
}
2022-01-30 02:23:11 +01:00
try {
manifest.schedule = await calendar.get(manifest)
if (manifest.schedule !== undefined)
manifest.schedule.radio = manifest
} catch (e) {
console.error("Error while fetching shows file", e)
}
2021-11-18 15:35:32 +01:00
resp = null
try {
2021-11-19 18:08:09 +01:00
resp = await fetch(getStreaminfoUrl(siteurl))
2021-11-18 15:35:32 +01:00
} catch (e) {
2021-12-05 17:40:17 +01:00
true
2021-11-18 15:35:32 +01:00
}
2021-12-26 19:59:45 +01:00
if(resp !== null) {
try {
text = await resp.text()
2022-01-30 14:49:56 +01:00
const data = JSON.parse(text)
const name = data['icy-name']
if (name !== undefined) {
manifest.setName(name)
}
const desc = data['icy-description']
if (desc !== undefined) {
manifest.setDescription(desc)
}
const logo = data['icy-logo']
if (desc !== undefined) {
manifest.setLogo(logo)
}
2021-12-26 19:59:45 +01:00
} catch (e) {
if (e instanceof SyntaxError) {
true
} else {
console.error('Error', e)
throw e
}
}
}
2021-11-19 18:08:09 +01:00
2021-11-18 15:35:32 +01:00
return manifest
2021-11-16 23:22:16 +01:00
}
2022-01-30 00:26:07 +01:00
2021-11-19 18:08:09 +01:00
function parseM3U (body) {
2021-12-26 19:59:45 +01:00
return body.split('\n').filter((line) => {
if (line.startsWith('#')) {
2021-11-18 15:35:32 +01:00
return false
} else {
2021-12-26 19:59:45 +01:00
try {
new URL(line); return true
} catch {
return false
}
2021-11-18 15:35:32 +01:00
}
})
2021-11-16 23:22:16 +01:00
}
2021-11-19 17:03:22 +01:00
2022-01-30 00:26:07 +01:00
2021-11-19 17:03:22 +01:00
module.exports = {
2021-11-19 18:08:09 +01:00
get: get,
objs: {
Radio: Radio,
RadioStreaming: RadioStreaming
},
parsers: {
M3U: parseM3U,
2022-01-30 00:26:07 +01:00
radioManifest: Radio.fromDOM,
2022-01-30 01:06:35 +01:00
shows: shows.parse,
2021-11-19 18:08:09 +01:00
}
2021-11-19 17:03:22 +01:00
}