mirror of
https://github.com/iptv-org/epg
synced 2026-05-08 18:36:59 -04:00
Add rts.ch
This commit is contained in:
21
sites/rts.ch/readme.md
Normal file
21
sites/rts.ch/readme.md
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# rts.ch
|
||||||
|
|
||||||
|
https://rts.ch
|
||||||
|
|
||||||
|
### Download the guide
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run grab --- --site=rts.ch
|
||||||
|
```
|
||||||
|
|
||||||
|
### Update channel list
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run channels:parse --- --config=./sites/rts.ch/rts.ch.config.js --output=./sites/rts.ch/rts.ch.channels.xml
|
||||||
|
```
|
||||||
|
|
||||||
|
### Test
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm test --- rts.ch
|
||||||
|
```
|
||||||
6
sites/rts.ch/rts.ch.channels.xml
Normal file
6
sites/rts.ch/rts.ch.channels.xml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<channels>
|
||||||
|
<channel site="rts.ch" site_id="143932a79bb5a123a646b68b1d1188d7ae493e5b" lang="fr" xmltv_id="RTS1.ch@SD">RTS 1</channel>
|
||||||
|
<channel site="rts.ch" site_id="d7dfff28deee44e1d3c49a3d37d36d492b29671b" lang="fr" xmltv_id="RTS2.ch@SD">RTS 2</channel>
|
||||||
|
<channel site="rts.ch" site_id="5d332a26e06d08eec8ad385d566187df72955623" lang="fr" xmltv_id="RTSInfo.ch@SD">RTS Info</channel>
|
||||||
|
</channels>
|
||||||
44
sites/rts.ch/rts.ch.config.js
Normal file
44
sites/rts.ch/rts.ch.config.js
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
const axios = require('axios')
|
||||||
|
const dayjs = require('dayjs')
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
site: 'rts.ch',
|
||||||
|
days: 2,
|
||||||
|
|
||||||
|
url({ date }) {
|
||||||
|
return `https://www.rts.ch/play/v3/api/rts/production/tv-program-guide?date=${date.format('YYYY-MM-DD')}`
|
||||||
|
},
|
||||||
|
|
||||||
|
parser({ content, channel }) {
|
||||||
|
try {
|
||||||
|
const { data } = JSON.parse(content)
|
||||||
|
|
||||||
|
const channelData = data.find(entry => entry.channel.id === channel.site_id)
|
||||||
|
if (!channelData || !Array.isArray(channelData.programList)) return []
|
||||||
|
|
||||||
|
return channelData.programList.map(program => ({
|
||||||
|
title: program.title || '',
|
||||||
|
subTitle: program.subtitle || undefined,
|
||||||
|
description: program.description || program.show?.description || undefined,
|
||||||
|
start: new Date(program.startTime).toISOString(),
|
||||||
|
stop: new Date(program.endTime).toISOString(),
|
||||||
|
icon: program.imageUrl ? { src: program.imageUrl } : undefined,
|
||||||
|
category: program.genre || undefined,
|
||||||
|
}))
|
||||||
|
} catch {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async channels() {
|
||||||
|
const today = dayjs().format('YYYY-MM-DD')
|
||||||
|
const { data: body } = await axios.get(
|
||||||
|
`https://www.rts.ch/play/v3/api/rts/production/tv-program-guide?date=${today}`
|
||||||
|
)
|
||||||
|
return body.data.map(entry => ({
|
||||||
|
site_id: entry.channel.id,
|
||||||
|
name: entry.channel.title,
|
||||||
|
lang: 'fr',
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
82
sites/rts.ch/rts.ch.test.js
Normal file
82
sites/rts.ch/rts.ch.test.js
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
const { parser, url } = require('./rts.ch.config.js')
|
||||||
|
const dayjs = require('dayjs')
|
||||||
|
const utc = require('dayjs/plugin/utc')
|
||||||
|
const customParseFormat = require('dayjs/plugin/customParseFormat')
|
||||||
|
dayjs.extend(customParseFormat)
|
||||||
|
dayjs.extend(utc)
|
||||||
|
|
||||||
|
const date = dayjs.utc('2026-03-20', 'YYYY-MM-DD').startOf('d')
|
||||||
|
const channel = { site_id: '143932a79bb5a123a646b68b1d1188d7ae493e5b', name: 'RTS 1', lang: 'fr' }
|
||||||
|
|
||||||
|
it('can generate valid url', () => {
|
||||||
|
expect(url({ channel, date })).toBe(
|
||||||
|
'https://www.rts.ch/play/v3/api/rts/production/tv-program-guide?date=2026-03-20'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can parse response', () => {
|
||||||
|
const content = JSON.stringify({
|
||||||
|
data: [
|
||||||
|
{
|
||||||
|
channel: { id: '143932a79bb5a123a646b68b1d1188d7ae493e5b', title: 'RTS 1' },
|
||||||
|
programList: [
|
||||||
|
{
|
||||||
|
title: '19h30',
|
||||||
|
startTime: '2026-03-20T19:30:00+01:00',
|
||||||
|
endTime: '2026-03-20T20:01:00+01:00',
|
||||||
|
imageUrl: 'https://kingfisher.rts.ch/res/img/cdns3/sherlock/urn:orphea-image:1035156',
|
||||||
|
genre: 'Actualité',
|
||||||
|
description: 'Le journal du soir.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Météo',
|
||||||
|
startTime: '2026-03-20T20:00:00+01:00',
|
||||||
|
endTime: '2026-03-20T20:03:00+01:00',
|
||||||
|
imageUrl: 'https://kingfisher.rts.ch/res/img/cdns3/sherlock/urn:orphea-image:400670',
|
||||||
|
genre: 'Actualité',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
})
|
||||||
|
|
||||||
|
const results = parser({ content, channel })
|
||||||
|
|
||||||
|
expect(results[0]).toMatchObject({
|
||||||
|
title: '19h30',
|
||||||
|
start: '2026-03-20T18:30:00.000Z',
|
||||||
|
stop: '2026-03-20T19:01:00.000Z',
|
||||||
|
description: 'Le journal du soir.',
|
||||||
|
category: 'Actualité',
|
||||||
|
icon: { src: 'https://kingfisher.rts.ch/res/img/cdns3/sherlock/urn:orphea-image:1035156' },
|
||||||
|
})
|
||||||
|
expect(results[1]).toMatchObject({
|
||||||
|
title: 'Météo',
|
||||||
|
start: '2026-03-20T19:00:00.000Z',
|
||||||
|
stop: '2026-03-20T19:03:00.000Z',
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can handle channel not found in response', () => {
|
||||||
|
const content = JSON.stringify({
|
||||||
|
data: [
|
||||||
|
{
|
||||||
|
channel: { id: 'some-other-channel-id', title: 'Other' },
|
||||||
|
programList: [{ title: 'Show', startTime: '2026-03-20T10:00:00+01:00', endTime: '2026-03-20T11:00:00+01:00' }],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
})
|
||||||
|
|
||||||
|
const results = parser({ content, channel })
|
||||||
|
expect(results).toMatchObject([])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can handle empty guide', () => {
|
||||||
|
const results = parser({ content: '', channel })
|
||||||
|
expect(results).toMatchObject([])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('can handle malformed JSON', () => {
|
||||||
|
const results = parser({ content: 'not-json', channel })
|
||||||
|
expect(results).toMatchObject([])
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user