Replace LF endings with CRLF

This commit is contained in:
freearhey
2025-07-31 22:29:01 +03:00
parent 17e3b4ddda
commit 29aa427923
379 changed files with 29332 additions and 29332 deletions

View File

@@ -1,94 +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: 'm.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),
image: parseImage(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 parseImage(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
}
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: 'm.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),
image: parseImage(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 parseImage(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

@@ -1,47 +1,47 @@
const { parser, url } = require('./m.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부',
image: '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([])
})
const { parser, url } = require('./m.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부',
image: '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([])
})