mirror of
https://github.com/iptv-org/epg
synced 2026-05-11 03:47:02 -04:00
CRLF
This commit is contained in:
File diff suppressed because it is too large
Load Diff
@@ -1,18 +1,18 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<channels>
|
<channels>
|
||||||
<channel site="france.tv" site_id="arte" lang="fr" xmltv_id="Arte.de@France">Arté</channel>
|
<channel site="france.tv" site_id="arte" lang="fr" xmltv_id="Arte.de@France">Arté</channel>
|
||||||
<channel site="france.tv" site_id="documentaire" lang="fr" xmltv_id="">France Télévisions Docs</channel>
|
<channel site="france.tv" site_id="documentaire" lang="fr" xmltv_id="">France Télévisions Docs</channel>
|
||||||
<channel site="france.tv" site_id="france-2" lang="fr" xmltv_id="France2.fr">France 2</channel>
|
<channel site="france.tv" site_id="france-2" lang="fr" xmltv_id="France2.fr">France 2</channel>
|
||||||
<channel site="france.tv" site_id="france-3" lang="fr" xmltv_id="France3.fr">France 3</channel>
|
<channel site="france.tv" site_id="france-3" lang="fr" xmltv_id="France3.fr">France 3</channel>
|
||||||
<channel site="france.tv" site_id="france-4" lang="fr" xmltv_id="France4.fr">France 4</channel>
|
<channel site="france.tv" site_id="france-4" lang="fr" xmltv_id="France4.fr">France 4</channel>
|
||||||
<channel site="france.tv" site_id="france-5" lang="fr" xmltv_id="France5.fr">France 5</channel>
|
<channel site="france.tv" site_id="france-5" lang="fr" xmltv_id="France5.fr">France 5</channel>
|
||||||
<channel site="france.tv" site_id="france-24" lang="fr" xmltv_id="France24.fr">France 24</channel>
|
<channel site="france.tv" site_id="france-24" lang="fr" xmltv_id="France24.fr">France 24</channel>
|
||||||
<channel site="france.tv" site_id="franceinfo" lang="fr" xmltv_id="FranceInfo.fr">Franceinfo</channel>
|
<channel site="france.tv" site_id="franceinfo" lang="fr" xmltv_id="FranceInfo.fr">Franceinfo</channel>
|
||||||
<channel site="france.tv" site_id="francetv" lang="fr" xmltv_id="">France TV</channel>
|
<channel site="france.tv" site_id="francetv" lang="fr" xmltv_id="">France TV</channel>
|
||||||
<channel site="france.tv" site_id="ina" lang="fr" xmltv_id="">INA (Institut National de l&#x27;Audiovisuel)</channel>
|
<channel site="france.tv" site_id="ina" lang="fr" xmltv_id="">INA (Institut National de l&#x27;Audiovisuel)</channel>
|
||||||
<channel site="france.tv" site_id="lcp-public-senat" lang="fr" xmltv_id="LCPPublicSenat.fr">LCP Public Sénat</channel>
|
<channel site="france.tv" site_id="lcp-public-senat" lang="fr" xmltv_id="LCPPublicSenat.fr">LCP Public Sénat</channel>
|
||||||
<channel site="france.tv" site_id="mieux" lang="fr" xmltv_id="">Mieux</channel>
|
<channel site="france.tv" site_id="mieux" lang="fr" xmltv_id="">Mieux</channel>
|
||||||
<channel site="france.tv" site_id="serie" lang="fr" xmltv_id="">France Télévisions Séries</channel>
|
<channel site="france.tv" site_id="serie" lang="fr" xmltv_id="">France Télévisions Séries</channel>
|
||||||
<channel site="france.tv" site_id="sport" lang="fr" xmltv_id="">France Télévisions Sport</channel>
|
<channel site="france.tv" site_id="sport" lang="fr" xmltv_id="">France Télévisions Sport</channel>
|
||||||
<channel site="france.tv" site_id="tv5-monde" lang="fr" xmltv_id="">TV5 Monde Plus</channel>
|
<channel site="france.tv" site_id="tv5-monde" lang="fr" xmltv_id="">TV5 Monde Plus</channel>
|
||||||
</channels>
|
</channels>
|
||||||
|
|||||||
@@ -1,159 +1,159 @@
|
|||||||
const dayjs = require('dayjs')
|
const dayjs = require('dayjs')
|
||||||
const axios = require('axios')
|
const axios = require('axios')
|
||||||
const utc = require('dayjs/plugin/utc')
|
const utc = require('dayjs/plugin/utc')
|
||||||
const timezone = require('dayjs/plugin/timezone')
|
const timezone = require('dayjs/plugin/timezone')
|
||||||
const customParseFormat = require('dayjs/plugin/customParseFormat')
|
const customParseFormat = require('dayjs/plugin/customParseFormat')
|
||||||
|
|
||||||
dayjs.extend(utc)
|
dayjs.extend(utc)
|
||||||
dayjs.extend(timezone)
|
dayjs.extend(timezone)
|
||||||
dayjs.extend(customParseFormat)
|
dayjs.extend(customParseFormat)
|
||||||
dayjs.tz.setDefault('Europe/Paris')
|
dayjs.tz.setDefault('Europe/Paris')
|
||||||
|
|
||||||
// Because France is excellent at pointing hours, their programs ALL start at 5/6 am,
|
// Because France is excellent at pointing hours, their programs ALL start at 5/6 am,
|
||||||
// so we need to keep track of the earlier day's program to get the midnight programming. How... odd.
|
// so we need to keep track of the earlier day's program to get the midnight programming. How... odd.
|
||||||
module.exports = {
|
module.exports = {
|
||||||
site: 'france.tv',
|
site: 'france.tv',
|
||||||
days: 2,
|
days: 2,
|
||||||
url: function ({ channel, date }) {
|
url: function ({ channel, date }) {
|
||||||
return `https://www.france.tv/api/epg/videos/?date=${date.format('YYYY-MM-DD')}&channel=${channel.site_id}`
|
return `https://www.france.tv/api/epg/videos/?date=${date.format('YYYY-MM-DD')}&channel=${channel.site_id}`
|
||||||
},
|
},
|
||||||
parser: async function ({ channel, content, date }) {
|
parser: async function ({ channel, content, date }) {
|
||||||
const programs = []
|
const programs = []
|
||||||
let items = []
|
let items = []
|
||||||
|
|
||||||
const dayBefore = date.subtract(1, 'd').format('YYYY-MM-DD')
|
const dayBefore = date.subtract(1, 'd').format('YYYY-MM-DD')
|
||||||
const linkDayBefore = `https://www.france.tv/api/epg/videos/?date=${dayBefore}&channel=${channel.site_id}`
|
const linkDayBefore = `https://www.france.tv/api/epg/videos/?date=${dayBefore}&channel=${channel.site_id}`
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const responseDayBefore = await axios.get(linkDayBefore)
|
const responseDayBefore = await axios.get(linkDayBefore)
|
||||||
const programmingDayBefore = responseDayBefore.data || []
|
const programmingDayBefore = responseDayBefore.data || []
|
||||||
|
|
||||||
// The broadcast day starts at ~6 AM. Programs with hour < 6 in the day-before API
|
// The broadcast day starts at ~6 AM. Programs with hour < 6 in the day-before API
|
||||||
// are actually early morning programs (00:00-05:59) of our target date.
|
// are actually early morning programs (00:00-05:59) of our target date.
|
||||||
if (Array.isArray(programmingDayBefore)) {
|
if (Array.isArray(programmingDayBefore)) {
|
||||||
programmingDayBefore.forEach(item => {
|
programmingDayBefore.forEach(item => {
|
||||||
const time = item?.content?.broadcastBeginDate
|
const time = item?.content?.broadcastBeginDate
|
||||||
if (!time) return
|
if (!time) return
|
||||||
const hour = parseInt(time.split('h')[0])
|
const hour = parseInt(time.split('h')[0])
|
||||||
|
|
||||||
if (hour < 6) {
|
if (hour < 6) {
|
||||||
items.push(item)
|
items.push(item)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
// Day before data unavailable, continue with current day only
|
// Day before data unavailable, continue with current day only
|
||||||
}
|
}
|
||||||
|
|
||||||
// From the current day's API, only include programs starting from 6h onwards.
|
// From the current day's API, only include programs starting from 6h onwards.
|
||||||
// Programs with hour < 6 belong to the next calendar day's schedule.
|
// Programs with hour < 6 belong to the next calendar day's schedule.
|
||||||
try {
|
try {
|
||||||
const currentDayItems = JSON.parse(content) || []
|
const currentDayItems = JSON.parse(content) || []
|
||||||
if (Array.isArray(currentDayItems)) {
|
if (Array.isArray(currentDayItems)) {
|
||||||
currentDayItems.forEach(item => {
|
currentDayItems.forEach(item => {
|
||||||
const time = item?.content?.broadcastBeginDate
|
const time = item?.content?.broadcastBeginDate
|
||||||
if (!time) return
|
if (!time) return
|
||||||
const hour = parseInt(time.split('h')[0])
|
const hour = parseInt(time.split('h')[0])
|
||||||
|
|
||||||
if (hour >= 6) {
|
if (hour >= 6) {
|
||||||
items.push(item)
|
items.push(item)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
return programs
|
return programs
|
||||||
}
|
}
|
||||||
|
|
||||||
items.forEach(item => {
|
items.forEach(item => {
|
||||||
const { start, stop } = parseDuration(date, item)
|
const { start, stop } = parseDuration(date, item)
|
||||||
if (!start.isValid() || !stop.isValid()) return
|
if (!start.isValid() || !stop.isValid()) return
|
||||||
// Can contain Season and Episode in title, but not always. If title is missing, skip the program
|
// Can contain Season and Episode in title, but not always. If title is missing, skip the program
|
||||||
if (!item?.content?.title) return
|
if (!item?.content?.title) return
|
||||||
|
|
||||||
let title = item.content.title
|
let title = item.content.title
|
||||||
let season = null
|
let season = null
|
||||||
let episode = null
|
let episode = null
|
||||||
|
|
||||||
const seMatch = title.match(/\s*-?\s*S(\d+)\s+E(\d+)\s*-?\s*/)
|
const seMatch = title.match(/\s*-?\s*S(\d+)\s+E(\d+)\s*-?\s*/)
|
||||||
if (seMatch) {
|
if (seMatch) {
|
||||||
season = parseInt(seMatch[1])
|
season = parseInt(seMatch[1])
|
||||||
episode = parseInt(seMatch[2])
|
episode = parseInt(seMatch[2])
|
||||||
title = title.replace(seMatch[0], ' ').replace(/^\s+/, '').replace(/\s+$/, '').trim()
|
title = title.replace(seMatch[0], ' ').replace(/^\s+/, '').replace(/\s+$/, '').trim()
|
||||||
}
|
}
|
||||||
|
|
||||||
const fullTitle = (item.content.titleLeading ? item.content.titleLeading + (title ? ' - ' : '') : '') + title
|
const fullTitle = (item.content.titleLeading ? item.content.titleLeading + (title ? ' - ' : '') : '') + title
|
||||||
|
|
||||||
programs.push({
|
programs.push({
|
||||||
title: fullTitle,
|
title: fullTitle,
|
||||||
description: item.content.description,
|
description: item.content.description,
|
||||||
image: getImageUrl(item),
|
image: getImageUrl(item),
|
||||||
icon: getImageUrl(item),
|
icon: getImageUrl(item),
|
||||||
start,
|
start,
|
||||||
stop,
|
stop,
|
||||||
season: season,
|
season: season,
|
||||||
episode: episode,
|
episode: episode,
|
||||||
rating: item.content.csa
|
rating: item.content.csa
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
return programs
|
return programs
|
||||||
},
|
},
|
||||||
async channels() {
|
async channels() {
|
||||||
try {
|
try {
|
||||||
const response = await axios.get('https://www.france.tv/chaines/')
|
const response = await axios.get('https://www.france.tv/chaines/')
|
||||||
const data = response.data || ''
|
const data = response.data || ''
|
||||||
const channels = []
|
const channels = []
|
||||||
|
|
||||||
const channelRegex =
|
const channelRegex =
|
||||||
/<button[^>]+aria-controls="[^"]*content-([a-z0-9-]+)"[\s\S]*?<title>([^<]+)<\/title>/gi
|
/<button[^>]+aria-controls="[^"]*content-([a-z0-9-]+)"[\s\S]*?<title>([^<]+)<\/title>/gi
|
||||||
|
|
||||||
let match
|
let match
|
||||||
while ((match = channelRegex.exec(data)) !== null) {
|
while ((match = channelRegex.exec(data)) !== null) {
|
||||||
channels.push({
|
channels.push({
|
||||||
lang: 'fr',
|
lang: 'fr',
|
||||||
site_id: match[1],
|
site_id: match[1],
|
||||||
name: match[2].trim()
|
name: match[2].trim()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return [...new Map(channels.map(channel => [channel.site_id, channel])).values()]
|
return [...new Map(channels.map(channel => [channel.site_id, channel])).values()]
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to fetch channels list:', error.message)
|
console.error('Failed to fetch channels list:', error.message)
|
||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function parseDuration(date, item) {
|
function parseDuration(date, item) {
|
||||||
const current_date = date.format('YYYY-MM-DD')
|
const current_date = date.format('YYYY-MM-DD')
|
||||||
const time = item.content?.broadcastBeginDate
|
const time = item.content?.broadcastBeginDate
|
||||||
const duration = item.content?.duration // e.g. "11 min 45 s", "1 h 30 min", "30 min"
|
const duration = item.content?.duration // e.g. "11 min 45 s", "1 h 30 min", "30 min"
|
||||||
|
|
||||||
if (!time) return { start: dayjs(null), stop: dayjs(null) }
|
if (!time) return { start: dayjs(null), stop: dayjs(null) }
|
||||||
|
|
||||||
const timeParts = time.split('h')
|
const timeParts = time.split('h')
|
||||||
|
|
||||||
let durationInSeconds = 0
|
let durationInSeconds = 0
|
||||||
if (duration) {
|
if (duration) {
|
||||||
const durationParts = duration.split(' ')
|
const durationParts = duration.split(' ')
|
||||||
for (let i = 0; i < durationParts.length; i++) {
|
for (let i = 0; i < durationParts.length; i++) {
|
||||||
const part = durationParts[i]
|
const part = durationParts[i]
|
||||||
if (part === 'h' && i > 0) {
|
if (part === 'h' && i > 0) {
|
||||||
durationInSeconds += parseInt(durationParts[i - 1]) * 3600
|
durationInSeconds += parseInt(durationParts[i - 1]) * 3600
|
||||||
} else if (part === 'min' && i > 0) {
|
} else if (part === 'min' && i > 0) {
|
||||||
durationInSeconds += parseInt(durationParts[i - 1]) * 60
|
durationInSeconds += parseInt(durationParts[i - 1]) * 60
|
||||||
} else if (part === 's' && i > 0) {
|
} else if (part === 's' && i > 0) {
|
||||||
durationInSeconds += parseInt(durationParts[i - 1])
|
durationInSeconds += parseInt(durationParts[i - 1])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const start = dayjs.utc(`${current_date} ${timeParts[0]}:${timeParts[1]}`, 'YYYY-MM-DD HH:mm')
|
const start = dayjs.utc(`${current_date} ${timeParts[0]}:${timeParts[1]}`, 'YYYY-MM-DD HH:mm')
|
||||||
const stop = start.add(durationInSeconds, 'second')
|
const stop = start.add(durationInSeconds, 'second')
|
||||||
return { start, stop }
|
return { start, stop }
|
||||||
}
|
}
|
||||||
|
|
||||||
function getImageUrl(item) {
|
function getImageUrl(item) {
|
||||||
const url = item.content?.thumbnail?.x1
|
const url = item.content?.thumbnail?.x1
|
||||||
return url
|
return url
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,54 +1,54 @@
|
|||||||
const { parser, url } = require('./france.tv.config.js')
|
const { parser, url } = require('./france.tv.config.js')
|
||||||
const axios = require('axios')
|
const axios = require('axios')
|
||||||
const fs = require('fs')
|
const fs = require('fs')
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const dayjs = require('dayjs')
|
const dayjs = require('dayjs')
|
||||||
const utc = require('dayjs/plugin/utc')
|
const utc = require('dayjs/plugin/utc')
|
||||||
const customParseFormat = require('dayjs/plugin/customParseFormat')
|
const customParseFormat = require('dayjs/plugin/customParseFormat')
|
||||||
dayjs.extend(customParseFormat)
|
dayjs.extend(customParseFormat)
|
||||||
dayjs.extend(utc)
|
dayjs.extend(utc)
|
||||||
|
|
||||||
jest.mock('axios')
|
jest.mock('axios')
|
||||||
|
|
||||||
const date = dayjs.utc('2026-02-19', 'YYYY-MM-DD').startOf('d')
|
const date = dayjs.utc('2026-02-19', 'YYYY-MM-DD').startOf('d')
|
||||||
const channel = {
|
const channel = {
|
||||||
site_id: 'france-2',
|
site_id: 'france-2',
|
||||||
xmltv_id: 'France2.fr@HD'
|
xmltv_id: 'France2.fr@HD'
|
||||||
}
|
}
|
||||||
|
|
||||||
it('can generate valid url', () => {
|
it('can generate valid url', () => {
|
||||||
expect(url({ channel, date })).toBe('https://www.france.tv/api/epg/videos/?date=2026-02-19&channel=france-2')
|
expect(url({ channel, date })).toBe('https://www.france.tv/api/epg/videos/?date=2026-02-19&channel=france-2')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('can parse response', async () => {
|
it('can parse response', async () => {
|
||||||
axios.get.mockResolvedValue({ data: [] })
|
axios.get.mockResolvedValue({ data: [] })
|
||||||
const content = fs.readFileSync(path.resolve(__dirname, '__data__/content.json'))
|
const content = fs.readFileSync(path.resolve(__dirname, '__data__/content.json'))
|
||||||
const results = (await parser({ content, date, channel })).map(p => {
|
const results = (await parser({ content, date, channel })).map(p => {
|
||||||
p.start = p.start.toJSON()
|
p.start = p.start.toJSON()
|
||||||
p.stop = p.stop.toJSON()
|
p.stop = p.stop.toJSON()
|
||||||
return p
|
return p
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(results.length).toBe(18)
|
expect(results.length).toBe(18)
|
||||||
expect(results[0]).toMatchObject({
|
expect(results[0]).toMatchObject({
|
||||||
title: 'Le 6h info - Émission du jeudi 19 février 2026',
|
title: 'Le 6h info - Émission du jeudi 19 février 2026',
|
||||||
description: "Un rendez-vous réveil-matin, avec un point sur l'actualité assorti de différentes rubriques qui permettent d'en explorer certains aspects plus en profondeur.",
|
description: "Un rendez-vous réveil-matin, avec un point sur l'actualité assorti de différentes rubriques qui permettent d'en explorer certains aspects plus en profondeur.",
|
||||||
image: 'https://medias.france.tv/S9p5NdAs4OR2UbyC1NIQWsYV-K4/240x0/filters:quality(85):format(webp)/b/f/3/e85c2e8fed4a4955965dfff63c3843fb.jpg',
|
image: 'https://medias.france.tv/S9p5NdAs4OR2UbyC1NIQWsYV-K4/240x0/filters:quality(85):format(webp)/b/f/3/e85c2e8fed4a4955965dfff63c3843fb.jpg',
|
||||||
start: '2026-02-19T06:00:00.000Z',
|
start: '2026-02-19T06:00:00.000Z',
|
||||||
stop: '2026-02-19T06:30:00.000Z'
|
stop: '2026-02-19T06:30:00.000Z'
|
||||||
})
|
})
|
||||||
expect(results[17]).toMatchObject({
|
expect(results[17]).toMatchObject({
|
||||||
title: 'JO Club - Émission du jeudi 19 février 2026',
|
title: 'JO Club - Émission du jeudi 19 février 2026',
|
||||||
description: "Tous les soirs, tout au long de ces Jeux olympiques d'hiver de Milan-Cortina, Laurent Luyat revient, avec les journalistes et consultants de France Télévisions, sur les épreuves de la journée. Il accueille les athlètes et les médaillés du jour. La journée a été marquée par du combiné nordique, avec l'épreuve par équipes messieurs, les demi-final...",
|
description: "Tous les soirs, tout au long de ces Jeux olympiques d'hiver de Milan-Cortina, Laurent Luyat revient, avec les journalistes et consultants de France Télévisions, sur les épreuves de la journée. Il accueille les athlètes et les médaillés du jour. La journée a été marquée par du combiné nordique, avec l'épreuve par équipes messieurs, les demi-final...",
|
||||||
image: 'https://medias.france.tv/xuxaBPNFyhMiVB5eeYrZV_1nPj4/240x0/filters:quality(85):format(webp)/v/p/h/phpmhbhpv.jpg',
|
image: 'https://medias.france.tv/xuxaBPNFyhMiVB5eeYrZV_1nPj4/240x0/filters:quality(85):format(webp)/v/p/h/phpmhbhpv.jpg',
|
||||||
start: '2026-02-19T23:00:00.000Z',
|
start: '2026-02-19T23:00:00.000Z',
|
||||||
stop: '2026-02-20T00:00:00.000Z'
|
stop: '2026-02-20T00:00:00.000Z'
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('can handle empty guide', async () => {
|
it('can handle empty guide', async () => {
|
||||||
axios.get.mockResolvedValue({ data: [] })
|
axios.get.mockResolvedValue({ data: [] })
|
||||||
const results = await parser({ content: [], date, channel })
|
const results = await parser({ content: [], date, channel })
|
||||||
|
|
||||||
expect(results).toMatchObject([])
|
expect(results).toMatchObject([])
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,21 +1,21 @@
|
|||||||
# france.tv
|
# france.tv
|
||||||
|
|
||||||
https://www.france.tv/
|
https://www.france.tv/
|
||||||
|
|
||||||
### Download the guide
|
### Download the guide
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
npm run grab --- --site=france.tv
|
npm run grab --- --site=france.tv
|
||||||
```
|
```
|
||||||
|
|
||||||
### Update channel list
|
### Update channel list
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
npm run channels:parse --- --config=./sites/france.tv/france.tv.config.js --output=./sites/france.tv/france.tv.channels.xml
|
npm run channels:parse --- --config=./sites/france.tv/france.tv.config.js --output=./sites/france.tv/france.tv.channels.xml
|
||||||
```
|
```
|
||||||
|
|
||||||
### Test
|
### Test
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
npm test --- france.tv
|
npm test --- france.tv
|
||||||
```
|
```
|
||||||
|
|||||||
Reference in New Issue
Block a user