Merge pull request #1759 from iptv-org/add-tving.com

Add guide from tving.com
This commit is contained in:
Aleksandr Statciuk 2023-01-23 03:18:22 +03:00 committed by GitHub
commit 43aa001fae
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 198 additions and 0 deletions

17
.github/workflows/tving.com.yml vendored Normal file
View file

@ -0,0 +1,17 @@
name: tving.com
on:
schedule:
- cron: '0 3 * * *'
workflow_dispatch:
workflow_run:
workflows: [_trigger]
types:
- completed
jobs:
load:
uses: ./.github/workflows/_load.yml
with:
site: ${{github.workflow}}
secrets:
APP_ID: ${{ secrets.APP_ID }}
APP_PRIVATE_KEY: ${{ secrets.APP_PRIVATE_KEY }}

File diff suppressed because one or more lines are too long

View file

@ -0,0 +1 @@
cb({"header":{"message":"OK","status":200},"body":{"result":[],"total_count":0,"has_more":"N"}});

View file

@ -0,0 +1,35 @@
<?xml version="1.0" encoding="UTF-8"?>
<site site="tving.com">
<channels>
<channel lang="ko" xmltv_id="ChannelA.kr" site_id="C01583">채널a</channel>
<channel lang="ko" xmltv_id="ChannelAPlus.kr" site_id="C17141">채널a plus</channel>
<channel lang="ko" xmltv_id="ChunghwaTV.kr" site_id="C00544">중화TV</channel>
<channel lang="ko" xmltv_id="HistoryKorea.kr" site_id="C17341">히스토리</channel>
<channel lang="ko" xmltv_id="JTBC.kr" site_id="C01582">jtbc</channel>
<channel lang="ko" xmltv_id="JTBC2.kr" site_id="C15741">jtbc2</channel>
<channel lang="ko" xmltv_id="JTBC4.kr" site_id="C22041">jtbc4</channel>
<channel lang="ko" xmltv_id="MBN.kr" site_id="C00708">MBN</channel>
<channel lang="ko" xmltv_id="MBNPlus.kr" site_id="C17142">MBN Plus</channel>
<channel lang="ko" xmltv_id="Mnet.kr" site_id="C00579">Mnet</channel>
<channel lang="ko" xmltv_id="OCN.kr" site_id="C07381">OCN</channel>
<channel lang="ko" xmltv_id="OCNMovies.kr" site_id="C04601">Ch. CGV</channel>
<channel lang="ko" xmltv_id="OCNThrills.kr" site_id="C07382">super Action</channel>
<channel lang="ko" xmltv_id="OGN.kr" site_id="C00590">OGN</channel>
<channel lang="ko" xmltv_id="Olive.kr" site_id="C00575">Olive</channel>
<channel lang="ko" xmltv_id="Onstyle.kr" site_id="C01142">Onstyle</channel>
<channel lang="ko" xmltv_id="OTVN.kr" site_id="C01143">OtvN</channel>
<channel lang="ko" xmltv_id="Tooniverse.kr" site_id="C06941">Tooniverse</channel>
<channel lang="ko" xmltv_id="TVChosun.kr" site_id="C01581">tv조선</channel>
<channel lang="ko" xmltv_id="TVChosun2.kr" site_id="C00585">c time</channel>
<channel lang="ko" xmltv_id="tvN.kr" site_id="C00551">tvN</channel>
<channel lang="ko" xmltv_id="XTVN.kr" site_id="C01141">Xtvn</channel>
<channel lang="ko" xmltv_id="YonhapNewsTV.kr" site_id="C01723">연합뉴스</channel>
<channel lang="ko" xmltv_id="YTN.kr" site_id="C00593">YTN</channel>
<channel lang="ko" xmltv_id="YTN2.kr" site_id="C01101">YTN life</channel>
<channel lang="ko" xmltv_id="YTNScience.kr" site_id="C15347">YTN science</channel>
<!-- <channel lang="ko" xmltv_id="" site_id="C00611">life time</channel> -->
<!-- <channel lang="ko" xmltv_id="" site_id="C00805">jtbc3</channel> -->
<!-- <channel lang="ko" xmltv_id="" site_id="C05661">디즈니</channel> -->
<!-- <channel lang="ko" xmltv_id="" site_id="C15152">DAI TV</channel> -->
</channels>
</site>

View file

@ -0,0 +1,94 @@
const axios = require('axios')
const cheerio = require('cheerio')
const dayjs = require('dayjs')
const utc = require('dayjs/plugin/utc')
const timezone = require('dayjs/plugin/timezone')
const customParseFormat = require('dayjs/plugin/customParseFormat')
dayjs.extend(utc)
dayjs.extend(timezone)
dayjs.extend(customParseFormat)
module.exports = {
site: 'tving.com',
days: 2,
url: function ({ channel, date }) {
return `https://api.tving.com/v2/media/schedules/${channel.site_id}/${date.format(
'YYYYMMDD'
)}?callback=cb&pageNo=1&pageSize=500&screenCode=CSSD0200&networkCode=CSND0900&osCode=CSOD0900&teleCode=CSCD0900&apiKey=4263d7d76161f4a19a9efe9ca7903ec4`
},
parser: function ({ content }) {
let programs = []
const items = parseItems(content)
items.forEach(item => {
programs.push({
title: item.program.name.ko,
description: item.program.synopsis.ko,
categories: parseCategories(item),
date: item.program.product_year,
directors: item.program.director,
actors: item.program.actor,
start: parseStart(item),
stop: parseStop(item),
icon: parseIcon(item)
})
})
return programs
},
async channels() {
let items = await axios
.get(`https://m.tving.com/guide/schedule.tving`)
.then(r => r.data)
.then(html => {
let $ = cheerio.load(html)
return $('ul.cb > li').toArray()
})
.catch(console.log)
return items.map(item => {
let $item = cheerio.load(item)
let [, site_id] = $item('a')
.attr('href')
.match(/\?id\=(.*)/) || [null, null]
let name = $item('img').attr('alt')
return {
lang: 'ko',
site_id,
name
}
})
}
}
function parseIcon(item) {
return item.program.image.length ? `https://image.tving.com${item.program.image[0].url}` : null
}
function parseStart(item) {
return dayjs.tz(item.broadcast_start_time.toString(), `YYYYMMDDHHmmss`, 'Asia/Seoul')
}
function parseStop(item) {
return dayjs.tz(item.broadcast_end_time.toString(), `YYYYMMDDHHmmss`, 'Asia/Seoul')
}
function parseCategories(item) {
const categories = []
if (item.category1_name) categories.push(item.category1_name.ko)
if (item.category2_name) categories.push(item.category2_name.ko)
return categories.filter(Boolean)
}
function parseItems(content) {
let data = (content.match(/cb\((.*)\)/) || [null, null])[1]
if (!data) return []
let json = JSON.parse(data)
if (!json || !json.body || !Array.isArray(json.body.result)) return []
return json.body.result
}

View file

@ -0,0 +1,50 @@
// npm run channels:parse -- --config=./sites/tving.com/tving.com.config.js --output=./sites/tving.com/tving.com.channels.xml
// npx epg-grabber --config=sites/tving.com/tving.com.config.js --channels=sites/tving.com/tving.com.channels.xml --output=guide.xml --days=2
const { parser, url, request } = require('./tving.com.config.js')
const fs = require('fs')
const path = require('path')
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('2023-01-23', 'YYYY-MM-DD').startOf('d')
const channel = {
site_id: 'C00551',
xmltv_id: 'tvN.kr'
}
it('can generate valid url', () => {
expect(url({ channel, date })).toBe(
`https://api.tving.com/v2/media/schedules/C00551/20230123?callback=cb&pageNo=1&pageSize=500&screenCode=CSSD0200&networkCode=CSND0900&osCode=CSOD0900&teleCode=CSCD0900&apiKey=4263d7d76161f4a19a9efe9ca7903ec4`
)
})
it('can parse response', () => {
const content = fs.readFileSync(path.resolve(__dirname, '__data__/content.txt'), 'utf8')
const results = parser({ content }).map(p => {
p.start = p.start.toJSON()
p.stop = p.stop.toJSON()
return p
})
expect(results[0]).toMatchObject({
title: '외계+인 1부',
description: '외계+인 1부',
icon: 'https://image.tving.com/upload/cms/caip/CAIP0200/P001661154.jpg',
date: 2022,
categories: [],
directors: ['최동훈'],
actors: ['김우빈', '류준열'],
start: '2023-01-22T13:40:00.000Z',
stop: '2023-01-22T15:00:00.000Z'
})
})
it('can handle empty guide', () => {
const content = fs.readFileSync(path.resolve(__dirname, '__data__/no_content.txt'), 'utf8')
expect(parser({ content })).toMatchObject([])
})