Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
d6b133a8e0
|
|||
|
c9486c54b2
|
|||
|
ac062aa1ba
|
|||
|
bcd45e34bf
|
|||
|
b3ee981bd1
|
|||
|
182b9a92ce
|
|||
|
fbc1870ce7
|
|||
|
edd18e92ed
|
|||
|
dc61d47b66
|
|||
|
10c3b8f5c1
|
|||
|
041b32e1df
|
|||
|
e98d923ce5
|
|||
|
01ddf25ed5
|
|||
|
4772f0179d
|
|||
|
c00dc8d33e
|
|||
|
57eb194efa
|
|||
|
4cbdd41b7c
|
|||
|
79891d178f
|
|||
|
89601096ba
|
|||
|
68329697ac
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -9,5 +9,6 @@ output/
|
|||||||
*.m3u8
|
*.m3u8
|
||||||
*.json
|
*.json
|
||||||
*.ini
|
*.ini
|
||||||
|
iptvc
|
||||||
|
|
||||||
!/**/*.gitkeep
|
!/**/*.gitkeep
|
||||||
|
|||||||
20
README.md
20
README.md
@@ -1,19 +1,20 @@
|
|||||||
# IPTV Checker (iptvc)
|
# IPTV Checker (iptvc)
|
||||||
|
|
||||||

|
[](https://git.axenov.dev/IPTV/iptvc/releases/latest)
|
||||||
|
|
||||||
Консольная программа для проверки IPTV-плейлистов в формате m3u или m3u8.
|
Консольная программа для проверки IPTV-плейлистов в формате m3u или m3u8.
|
||||||
|
|
||||||
> [!IMPORTANT]
|
> **Веб-сайт:** [m3u.su](https://m3u.su)
|
||||||
> Проект находится на ранней стадии разработки.
|
> Исходный код: [git.axenov.dev/IPTV/iptvc](https://git.axenov.dev/IPTV/iptvc)
|
||||||
> Реализован минимально необходимый функционал.
|
> Telegram-канал: [@iptv_aggregator](https://t.me/iptv_aggregator)
|
||||||
> Возможны ошибки, неточности и обратно несовместимые изменения.
|
> Обсуждение: [@iptv_aggregator_chat](https://t.me/iptv_aggregator_chat)
|
||||||
|
> Дополнительные сведения:
|
||||||
Для дополнительной документации можно обращаться в директорию [docs](./docs).
|
> * [./docs](./docs)
|
||||||
|
> * [git.axenov.dev/IPTV/.profile](https://git.axenov.dev/IPTV/.profile)
|
||||||
|
|
||||||
## Установка
|
## Установка
|
||||||
|
|
||||||
Достаточно скачать и распаковать архив с подходящим исполняемым файлом [со страницы релизов](https://git.axenov.dev/IPTV/iptvc/releases):
|
Достаточно скачать и распаковать архив с подходящим исполняемым файлом [со страницы последнего релиза](https://git.axenov.dev/IPTV/iptvc/releases/latest):
|
||||||
|
|
||||||
| ОС | Архив | Платформа |
|
| ОС | Архив | Платформа |
|
||||||
|---------|----------------------|-----------|
|
|---------|----------------------|-----------|
|
||||||
@@ -62,10 +63,13 @@
|
|||||||
2. Выполнить команду `./iptvc check` или `./iptvc check -i playlist.ini` для проверки всех плейлистов из файла `./playlists.ini`
|
2. Выполнить команду `./iptvc check` или `./iptvc check -i playlist.ini` для проверки всех плейлистов из файла `./playlists.ini`
|
||||||
3. Выполнить команду `./iptvc check -i test.ini -c ABC`, чтобы проверить только плейлист с кодом `ABC` из файла `./test.ini`
|
3. Выполнить команду `./iptvc check -i test.ini -c ABC`, чтобы проверить только плейлист с кодом `ABC` из файла `./test.ini`
|
||||||
|
|
||||||
|
Если `-i` не указан явно, то будет попытка прочитать файл `playlists.ini`, находящийся в одной директории с iptvc.
|
||||||
|
|
||||||
Аргумент `-i` можно указывать только однажды, но его можно комбинировать с `-f` и `-u`.
|
Аргумент `-i` можно указывать только однажды, но его можно комбинировать с `-f` и `-u`.
|
||||||
|
|
||||||
### Другие возможности команды `check`
|
### Другие возможности команды `check`
|
||||||
|
|
||||||
|
* `--random|-r X` -- проверить X случайных плейлистов из ini-файла
|
||||||
* `--json|-j` -- вывести результаты проверки в формате JSON
|
* `--json|-j` -- вывести результаты проверки в формате JSON
|
||||||
* `--quiet|-q` -- полностью подавить вывод лога (включая отладочную информацию)
|
* `--quiet|-q` -- полностью подавить вывод лога (включая отладочную информацию)
|
||||||
* `--verbose|-v` -- добавить в лог более подробную отладочную информацию (значительно увеличит количество строк!)
|
* `--verbose|-v` -- добавить в лог более подробную отладочную информацию (значительно увеличит количество строк!)
|
||||||
|
|||||||
@@ -10,10 +10,11 @@ import (
|
|||||||
"axenov/iptv-checker/app/cache"
|
"axenov/iptv-checker/app/cache"
|
||||||
"axenov/iptv-checker/app/config"
|
"axenov/iptv-checker/app/config"
|
||||||
"axenov/iptv-checker/app/logger"
|
"axenov/iptv-checker/app/logger"
|
||||||
|
|
||||||
"github.com/redis/go-redis/v9"
|
"github.com/redis/go-redis/v9"
|
||||||
)
|
)
|
||||||
|
|
||||||
const VERSION = "1.0.1"
|
const VERSION = "1.0.6"
|
||||||
|
|
||||||
// Arguments описывает аргументы командной строки
|
// Arguments описывает аргументы командной строки
|
||||||
type Arguments struct {
|
type Arguments struct {
|
||||||
|
|||||||
14
app/cache/cache.go
vendored
14
app/cache/cache.go
vendored
@@ -10,28 +10,30 @@ import (
|
|||||||
"axenov/iptv-checker/app/config"
|
"axenov/iptv-checker/app/config"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/redis/go-redis/v9"
|
|
||||||
"log"
|
"log"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/redis/go-redis/v9"
|
||||||
)
|
)
|
||||||
|
|
||||||
func Init(cfg *config.CacheConfig) *redis.Client {
|
func Init(cfg *config.CacheConfig) *redis.Client {
|
||||||
redis := redis.NewClient(&redis.Options{
|
redisUrl := fmt.Sprintf("%s:%s", cfg.Host, strconv.Itoa(int(cfg.Port)))
|
||||||
Addr: fmt.Sprintf("%s:%s", cfg.Host, strconv.Itoa(int(cfg.Port))),
|
redisClient := redis.NewClient(&redis.Options{
|
||||||
|
Addr: redisUrl,
|
||||||
DB: int(cfg.Db),
|
DB: int(cfg.Db),
|
||||||
PoolSize: 1000,
|
PoolSize: 1000,
|
||||||
ReadTimeout: -1,
|
ReadTimeout: -1,
|
||||||
WriteTimeout: -1,
|
WriteTimeout: -1,
|
||||||
})
|
})
|
||||||
client := redis.Conn()
|
client := redisClient.Conn()
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
err := client.Ping(ctx).Err()
|
err := client.Ping(ctx).Err()
|
||||||
if err == nil {
|
if err == nil {
|
||||||
log.Println("Connected to cache DB")
|
log.Println("Connected to cache DB:", redisUrl)
|
||||||
cfg.IsActive = true
|
cfg.IsActive = true
|
||||||
} else {
|
} else {
|
||||||
log.Println("Error while connecting to cache DB, program may work not as expected:", err)
|
log.Println("Error while connecting to cache DB, program may work not as expected:", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return redis
|
return redisClient
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,10 +18,10 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"maps"
|
"maps"
|
||||||
|
"math"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"runtime"
|
|
||||||
"slices"
|
"slices"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
@@ -165,20 +165,22 @@ func CheckPlaylists(lists []playlist.Playlist) (int, int) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func cachePlaylist(pls playlist.Playlist) {
|
func cachePlaylist(pls playlist.Playlist) {
|
||||||
if app.Config.Cache.IsActive {
|
if !app.Config.Cache.IsActive {
|
||||||
jsonBytes, err := json.Marshal(pls)
|
return
|
||||||
if err != nil {
|
|
||||||
log.Printf("Error while saving playlist to cache: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
ttl := time.Duration(app.Config.Cache.Ttl) * time.Second
|
|
||||||
written := app.Cache.Set(ctx, pls.Code, string(jsonBytes), ttl)
|
|
||||||
if written.Err() != nil {
|
|
||||||
log.Printf("Error while saving playlist to cache: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Println("Cached sucessfully")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
jsonBytes, err := json.Marshal(pls)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Error while saving playlist to cache: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ttl := time.Duration(app.Config.Cache.Ttl) * time.Second
|
||||||
|
written := app.Cache.Set(ctx, pls.Code, string(jsonBytes), ttl)
|
||||||
|
if written.Err() != nil {
|
||||||
|
log.Printf("Error while saving playlist to cache: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println("Cached sucessfully")
|
||||||
}
|
}
|
||||||
|
|
||||||
// CheckChannels проверяет каналы и возвращает их же с результатами проверки
|
// CheckChannels проверяет каналы и возвращает их же с результатами проверки
|
||||||
@@ -216,16 +218,16 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
|
|
||||||
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
||||||
req, err := http.NewRequest("GET", tvChannel.URL, nil)
|
req, err := http.NewRequest("GET", tvChannel.URL, nil)
|
||||||
tvChannel.CheckedAt = time.Now().Unix()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
data := errorData{tvChannel: tvChannel, err: err}
|
data := errorData{tvChannel: tvChannel, err: err}
|
||||||
chError <- data
|
chError <- data
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO user-agent
|
|
||||||
req.Header.Set("User-Agent", "Mozilla/5.0 WINK/1.31.1 (AndroidTV/9) HlsWinkPlayer")
|
req.Header.Set("User-Agent", "Mozilla/5.0 WINK/1.31.1 (AndroidTV/9) HlsWinkPlayer")
|
||||||
|
req.Header.Set("Range", "bytes=0-511") // 512 B, but sometimes servers ignore it
|
||||||
resp, err := httpClient.Do(req)
|
resp, err := httpClient.Do(req)
|
||||||
|
tvChannel.CheckedAt = time.Now().Unix()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
data := errorData{tvChannel: tvChannel, err: err}
|
data := errorData{tvChannel: tvChannel, err: err}
|
||||||
chError <- data
|
chError <- data
|
||||||
@@ -235,7 +237,8 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
tvChannel.Status = resp.StatusCode
|
tvChannel.Status = resp.StatusCode
|
||||||
tvChannel.IsOnline = tvChannel.Status < http.StatusBadRequest
|
tvChannel.IsOnline = tvChannel.Status < http.StatusBadRequest
|
||||||
tvChannel.ContentType = resp.Header.Get("Content-Type")
|
tvChannel.ContentType = resp.Header.Get("Content-Type")
|
||||||
bodyBytes, _ := io.ReadAll(resp.Body)
|
chunk := io.LimitReader(resp.Body, 512) // just for sure
|
||||||
|
bodyBytes, _ := io.ReadAll(chunk)
|
||||||
bodyString := string(bodyBytes)
|
bodyString := string(bodyBytes)
|
||||||
_ = resp.Body.Close()
|
_ = resp.Body.Close()
|
||||||
contentType := http.DetectContentType(bodyBytes)
|
contentType := http.DetectContentType(bodyBytes)
|
||||||
@@ -245,20 +248,17 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
|
|
||||||
isContentCorrect := isContentBinary ||
|
isContentCorrect := isContentBinary ||
|
||||||
strings.Contains(bodyString, "#EXTM3U") ||
|
strings.Contains(bodyString, "#EXTM3U") ||
|
||||||
strings.Contains(bodyString, "<SegmentTemplate")
|
strings.Contains(bodyString, "#EXT-X-") ||
|
||||||
|
strings.Contains(bodyString, "<MPD ") ||
|
||||||
|
strings.Contains(bodyString, "<SegmentTemplate ") ||
|
||||||
|
strings.Contains(bodyString, "<AdaptationSet ")
|
||||||
|
|
||||||
if tvChannel.Status >= http.StatusBadRequest || !isContentCorrect {
|
if tvChannel.Status >= http.StatusBadRequest && !isContentCorrect {
|
||||||
tvChannel.Error = bodyString
|
tvChannel.Error = bodyString
|
||||||
chOffline <- tvChannel
|
chOffline <- tvChannel
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if isContentBinary {
|
|
||||||
tvChannel.Content = "binary"
|
|
||||||
} else {
|
|
||||||
tvChannel.Content = bodyString
|
|
||||||
}
|
|
||||||
|
|
||||||
chOnline <- tvChannel
|
chOnline <- tvChannel
|
||||||
return
|
return
|
||||||
}(tvChannel)
|
}(tvChannel)
|
||||||
@@ -268,7 +268,6 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
select {
|
select {
|
||||||
case tvChannel := <-chOnline:
|
case tvChannel := <-chOnline:
|
||||||
tvChannel.IsOnline = true
|
tvChannel.IsOnline = true
|
||||||
pls.OnlineCount++
|
|
||||||
pls.Channels[tvChannel.Id] = tvChannel
|
pls.Channels[tvChannel.Id] = tvChannel
|
||||||
if app.Args.Verbose {
|
if app.Args.Verbose {
|
||||||
log.Printf("[%.3d/%.3d] ONLINE '%s'\n", idx, count, tvChannel.Title)
|
log.Printf("[%.3d/%.3d] ONLINE '%s'\n", idx, count, tvChannel.Title)
|
||||||
@@ -277,7 +276,6 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
log.Printf("> MimeType: %s\n", tvChannel.ContentType)
|
log.Printf("> MimeType: %s\n", tvChannel.ContentType)
|
||||||
}
|
}
|
||||||
case tvChannel := <-chOffline:
|
case tvChannel := <-chOffline:
|
||||||
pls.OfflineCount++
|
|
||||||
pls.Channels[tvChannel.Id] = tvChannel
|
pls.Channels[tvChannel.Id] = tvChannel
|
||||||
if app.Args.Verbose {
|
if app.Args.Verbose {
|
||||||
log.Printf("[%.3d/%.3d] OFFLINE '%s'\n", idx, count, tvChannel.Title)
|
log.Printf("[%.3d/%.3d] OFFLINE '%s'\n", idx, count, tvChannel.Title)
|
||||||
@@ -286,7 +284,6 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
log.Printf("> Status: %d\n", tvChannel.Status)
|
log.Printf("> Status: %d\n", tvChannel.Status)
|
||||||
}
|
}
|
||||||
case data := <-chError:
|
case data := <-chError:
|
||||||
pls.OfflineCount++
|
|
||||||
pls.Channels[data.tvChannel.Id] = data.tvChannel
|
pls.Channels[data.tvChannel.Id] = data.tvChannel
|
||||||
if app.Args.Verbose {
|
if app.Args.Verbose {
|
||||||
log.Printf("[%.3d/%.3d] ERROR '%s'\n", idx, count, data.tvChannel.Title)
|
log.Printf("[%.3d/%.3d] ERROR '%s'\n", idx, count, data.tvChannel.Title)
|
||||||
@@ -303,6 +300,14 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
close(chError)
|
close(chError)
|
||||||
pls.CheckedAt = time.Now().Unix()
|
pls.CheckedAt = time.Now().Unix()
|
||||||
|
|
||||||
|
for _, tvChannel := range pls.Channels {
|
||||||
|
if tvChannel.IsOnline {
|
||||||
|
pls.OnlineCount++
|
||||||
|
} else {
|
||||||
|
pls.OfflineCount++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
log.Printf(
|
log.Printf(
|
||||||
"Checked successfully! online=%d onlinePercent=%.2f%% offline=%d offlinePercent=%.2f%% elapsedTime=%.2fs",
|
"Checked successfully! online=%d onlinePercent=%.2f%% offline=%d offlinePercent=%.2f%% elapsedTime=%.2fs",
|
||||||
pls.OnlineCount,
|
pls.OnlineCount,
|
||||||
@@ -317,26 +322,22 @@ func CheckChannels(pls playlist.Playlist) playlist.Playlist {
|
|||||||
|
|
||||||
// calcParameters вычисляет оптимальное количество горутин и таймаут запроса
|
// calcParameters вычисляет оптимальное количество горутин и таймаут запроса
|
||||||
func calcParameters(count int) (time.Duration, int) {
|
func calcParameters(count int) (time.Duration, int) {
|
||||||
var routines int
|
routines := count
|
||||||
var percentage float32
|
if routines > 3000 {
|
||||||
|
routines = 3000
|
||||||
if count <= 100 {
|
|
||||||
routines = count
|
|
||||||
} else {
|
|
||||||
percentage = float32(runtime.NumCPU()) * 0.075
|
|
||||||
for percentage >= 1 {
|
|
||||||
percentage *= 0.5
|
|
||||||
}
|
|
||||||
routines = int(float32(count) * percentage)
|
|
||||||
}
|
|
||||||
if routines > 500 {
|
|
||||||
routines = 500
|
|
||||||
}
|
}
|
||||||
if routines < 1 {
|
if routines < 1 {
|
||||||
routines = 1
|
routines = 1
|
||||||
}
|
}
|
||||||
|
|
||||||
timeout := 10 / float32(count) * 150
|
var digits = 1
|
||||||
|
x := count
|
||||||
|
for x >= 10 {
|
||||||
|
digits++
|
||||||
|
x /= 10
|
||||||
|
}
|
||||||
|
|
||||||
|
timeout := 10 - int(math.Ceil(float64(digits)*1.5))
|
||||||
if timeout > 10 {
|
if timeout > 10 {
|
||||||
timeout = 10
|
timeout = 10
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Group - структура для хранения информации о группе каналов
|
// Group - структура для хранения информации о группе каналов
|
||||||
@@ -31,7 +32,6 @@ type Channel struct {
|
|||||||
Status int `json:"status"` // Код статуса HTTP
|
Status int `json:"status"` // Код статуса HTTP
|
||||||
IsOnline bool `json:"isOnline"` // Признак доступности канала (при Status < 400)
|
IsOnline bool `json:"isOnline"` // Признак доступности канала (при Status < 400)
|
||||||
Error string `json:"error"` // Текст ошибки (при Status >= 400)
|
Error string `json:"error"` // Текст ошибки (при Status >= 400)
|
||||||
Content string `json:"content"` // Тело ответа (формат m3u, либо маскированные бинарные данные, либо пусто)
|
|
||||||
ContentType string `json:"contentType"` // MIME-тип тела ответа
|
ContentType string `json:"contentType"` // MIME-тип тела ответа
|
||||||
Tags []string `json:"tags"` // Список тегов канала
|
Tags []string `json:"tags"` // Список тегов канала
|
||||||
CheckedAt int64 `json:"checkedAt"` // Время проверки в формате UNIX timestamp
|
CheckedAt int64 `json:"checkedAt"` // Время проверки в формате UNIX timestamp
|
||||||
@@ -106,17 +106,28 @@ func parseAttributes(line string) map[string]string {
|
|||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseName парсит название канала из строки тега #EXTINF
|
// parseTitle парсит название канала из строки тега #EXTINF
|
||||||
func parseName(line string) string {
|
func parseTitle(line string) string {
|
||||||
//TODO https://git.axenov.dev/IPTV/iptvc/issues/7
|
// сначала пытаемся по-доброму: в строке есть тег, могут быть атрибуты,
|
||||||
parts := strings.Split(line, ",")
|
// есть запятая-разделитель, после неё -- название канала (с запятыми или без)
|
||||||
if len(parts) == 2 {
|
|
||||||
return strings.Trim(parts[1], " ")
|
|
||||||
}
|
|
||||||
|
|
||||||
regex := regexp.MustCompile(`['"]?\s*,\s*(.+)`)
|
regex := regexp.MustCompile(`['"]?\s*,\s*(.+)`)
|
||||||
regexMatches := regex.FindAllStringSubmatch(line, -1)
|
regexMatches := regex.FindAllStringSubmatch(line, -1)
|
||||||
return regexMatches[0][1]
|
if len(regexMatches) > 0 && len(regexMatches[0]) >= 2 {
|
||||||
|
return strings.TrimSpace(regexMatches[0][1])
|
||||||
|
}
|
||||||
|
|
||||||
|
// теперь пытаемся хоть как-то: в строке есть тег, могут быть атрибуты,
|
||||||
|
// НЕТ запятой-разделителя и название канала (с запятыми или без)
|
||||||
|
lastQuotePos := strings.LastIndexAny(line, `,"'`)
|
||||||
|
if lastQuotePos != -1 && lastQuotePos < len(line)-1 {
|
||||||
|
afterLastQuote := line[lastQuotePos+1:]
|
||||||
|
name := strings.TrimSpace(afterLastQuote)
|
||||||
|
if name != "" {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return line // ну штош
|
||||||
}
|
}
|
||||||
|
|
||||||
// Download загружает плейлист по URL-адресу
|
// Download загружает плейлист по URL-адресу
|
||||||
@@ -124,6 +135,7 @@ func (pls *Playlist) Download() error {
|
|||||||
content, err := utils.Fetch(pls.Url)
|
content, err := utils.Fetch(pls.Url)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
pls.Content = err.Error()
|
pls.Content = err.Error()
|
||||||
|
pls.CheckedAt = time.Now().Unix()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,6 +148,7 @@ func (pls *Playlist) ReadFromFs() error {
|
|||||||
content, err := os.ReadFile(pls.Url)
|
content, err := os.ReadFile(pls.Url)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
pls.Content = err.Error()
|
pls.Content = err.Error()
|
||||||
|
pls.CheckedAt = time.Now().Unix()
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -161,14 +174,14 @@ func (pls *Playlist) Parse() Playlist {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if strings.HasPrefix(line, "#EXTM3U") {
|
if strings.HasPrefix(line, "#EXTM3U") {
|
||||||
pls.Attributes = parseAttributes(content)
|
pls.Attributes = parseAttributes(line)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if strings.HasPrefix(line, "#EXTINF") {
|
if strings.HasPrefix(line, "#EXTINF") {
|
||||||
isChannel = true
|
isChannel = true
|
||||||
tmpChannel.Attributes = parseAttributes(line)
|
tmpChannel.Attributes = parseAttributes(line)
|
||||||
tmpChannel.Title = parseName(line)
|
tmpChannel.Title = parseTitle(line)
|
||||||
|
|
||||||
if tmpChannel.Title == "" {
|
if tmpChannel.Title == "" {
|
||||||
if tvgid, ok := tmpChannel.Attributes["tvg-id"]; ok {
|
if tvgid, ok := tmpChannel.Attributes["tvg-id"]; ok {
|
||||||
|
|||||||
@@ -18,9 +18,10 @@ import (
|
|||||||
|
|
||||||
// TagBlock описывает объект с набором тегов, который подходит для каналов по регулярному выражению
|
// TagBlock описывает объект с набором тегов, который подходит для каналов по регулярному выражению
|
||||||
type TagBlock struct {
|
type TagBlock struct {
|
||||||
TvgId string `json:"tvg-id"`
|
TvgId string `json:"tvg-id"`
|
||||||
Title string `json:"title"`
|
TvgName string `json:"tvg-name"`
|
||||||
Tags []string `json:"tags"`
|
Title string `json:"title"`
|
||||||
|
Tags []string `json:"tags"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetTags возвращает теги, соответствующие каналу
|
// GetTags возвращает теги, соответствующие каналу
|
||||||
@@ -42,6 +43,18 @@ func (block *TagBlock) GetTags(ch playlist.Channel) []string {
|
|||||||
if checkString == "" {
|
if checkString == "" {
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
} else if block.TvgName != "" {
|
||||||
|
regex, err = regexp.Compile(block.TvgName)
|
||||||
|
if err != nil {
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
if _, ok := ch.Attributes["tvg-name"]; !ok {
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
checkString = ch.Attributes["tvg-name"]
|
||||||
|
if checkString == "" {
|
||||||
|
return result
|
||||||
|
}
|
||||||
} else if block.Title != "" {
|
} else if block.Title != "" {
|
||||||
regex, err = regexp.Compile(block.Title)
|
regex, err = regexp.Compile(block.Title)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -51,13 +51,9 @@ func Fetch(url string) ([]byte, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
req.Header.Set(
|
req.Header.Set("User-Agent", "Mozilla/5.0 WINK/1.31.1 (AndroidTV/9) HlsWinkPlayer")
|
||||||
"User-Agent",
|
|
||||||
"Mozilla/5.0 WINK/1.31.1 (AndroidTV/9) HlsWinkPlayer",
|
|
||||||
)
|
|
||||||
|
|
||||||
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
||||||
httpClient := http.Client{Timeout: 5 * time.Second}
|
httpClient := http.Client{Timeout: 10 * time.Second}
|
||||||
resp, err := httpClient.Do(req)
|
resp, err := httpClient.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
|||||||
@@ -11,9 +11,10 @@ import (
|
|||||||
"axenov/iptv-checker/app/checker"
|
"axenov/iptv-checker/app/checker"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/spf13/cobra"
|
|
||||||
"log"
|
"log"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
||||||
// checkCmd represents the file command
|
// checkCmd represents the file command
|
||||||
|
|||||||
@@ -16,8 +16,8 @@ import (
|
|||||||
// rootCmd represents the base command when called without any subcommands
|
// rootCmd represents the base command when called without any subcommands
|
||||||
var rootCmd = &cobra.Command{
|
var rootCmd = &cobra.Command{
|
||||||
Use: "iptvc",
|
Use: "iptvc",
|
||||||
Short: "Simple utility to check iptv playlists. Part of iptv.axenov.dev project.",
|
Short: "Simple utility to check iptv playlists. Part of m3u.su project.",
|
||||||
Long: `Simple utility to check iptv playlists. Part of iptv.axenov.dev project.
|
Long: `Simple utility to check iptv playlists. Part of m3u.su project.
|
||||||
Copyright (c) 2025, Антон Аксенов, MIT license.`,
|
Copyright (c) 2025, Антон Аксенов, MIT license.`,
|
||||||
// Uncomment the following line if your bare application
|
// Uncomment the following line if your bare application
|
||||||
// has an action associated with it:
|
// has an action associated with it:
|
||||||
|
|||||||
Reference in New Issue
Block a user