8 Commits

Author SHA1 Message Date
a3c33d7ec1 Версия v1.1.1
All checks were successful
Release / release (push) Successful in 4m53s
2025-11-22 19:28:00 +08:00
d7f28413b2 Скорректировано поведение --repeat
При значении 0 количество итераций будет бесконечным
2025-11-22 19:27:33 +08:00
d4260f85ec Версия v1.1.0
All checks were successful
Release / release (push) Successful in 5m51s
2025-11-22 18:33:47 +08:00
6ea3683350 Аргументы --repeat и --every
Some checks failed
Release / release (push) Has been cancelled
2025-11-22 18:32:27 +08:00
bc03abeb9d Мелочи по сборке и README 2025-11-22 18:05:38 +08:00
14c251f3e4 Построение докер-образов в gitea 2025-11-22 01:17:38 +08:00
317ebfdf5f Dockerfile 2025-11-22 01:17:19 +08:00
68bb6199b9 Синтаксис Makefile + возможность передать ARCH 2025-11-21 00:15:06 +08:00
8 changed files with 196 additions and 90 deletions

View File

@@ -1,28 +0,0 @@
name: release
on:
push:
tags:
- 'v*'
jobs:
release:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
name: Checkout
with:
fetch-depth: 0
- name: Setup go
uses: https://github.com/actions/setup-go@v4
with:
go-version: '>=1.23.6'
- name: Compile
run: make release
- name: Make release
id: use-go-action
uses: https://gitea.com/actions/release-action@main
with:
files: |-
bin/*.zip
api_key: '${{secrets.RELEASE_TOKEN}}'

View File

@@ -0,0 +1,65 @@
# https://docs.gitea.com/usage/actions/overview
# https://docs.github.com/ru/actions/reference/workflows-and-actions/contexts
name: Release
on:
push:
tags:
- 'v*'
jobs:
release:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
with: # https://github.com/actions/checkout
fetch-depth: 0
- name: Set up go
uses: actions/setup-go@v4
with: # https://github.com/actions/setup-go
go-version: '>=1.24.2'
- name: Build release files (amd64)
run: make release ARCH=amd64
- name: Build release files (arm64)
run: make release ARCH=arm64
- name: Create new release
id: use-go-action
uses: https://gitea.com/actions/gitea-release-action@main
with: # https://gitea.com/actions/gitea-release-action
server_url: https://git.axenov.dev
token: '${{secrets.RELEASE_TOKEN}}'
files: |-
bin/*.zip
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
with: # https://github.com/docker/setup-buildx-action
build-args: "IPTVC_VERSION=${{ github.ref_name }}"
buildkitd-config-inline: |
# https://github.com/moby/buildkit/blob/master/docs/buildkitd.toml.md
[ registry."docker.io" ]
mirrors = ["https://dockerhub.timeweb.cloud", "https://dh-mirror.gitverse.ru"]
http = true
insecure = true
- name: Log in to Gitea Container Registry
uses: docker/login-action@v3
with: # https://github.com/docker/login-action
registry: git.axenov.dev
username: ${{ secrets.USERNAME }}
password: ${{ secrets.RELEASE_TOKEN }}
- name: Build and push Docker images
uses: docker/build-push-action@v5
with: # https://github.com/docker/build-push-action
context: .
push: true
tags: |
git.axenov.dev/iptv/iptvc:${{ github.ref_name }}
git.axenov.dev/iptv/iptvc:latest

24
Dockerfile Normal file
View File

@@ -0,0 +1,24 @@
FROM golang:1.25-alpine AS iptv-img-builder
ARG GOOS
ARG GOARCH
ARG IPTVC_VERSION
ENV CGO_ENABLED=0
ENV GOOS=${GOOS:-linux}
ENV GOARCH=${GOARCH:-amd64}
COPY go.mod go.sum ./
RUN go mod download
COPY . .
RUN go build \
-trimpath \
-ldflags="-s -w -X main.version=${IPTVC_VERSION}" \
-o /app/iptvc \
.
FROM alpine:3.22.2 AS iptv-img-checker
LABEL org.opencontainers.image.authors="Anthony Axenov <anthonyaxenov@gmail.com>"
WORKDIR /app
RUN apk add --no-cache ca-certificates tzdata
RUN addgroup -S iptvc-user && adduser -S -G iptvc-user -H -s /sbin/nologin iptvc-user
COPY --from=iptv-img-builder --chown=iptvc-user:iptvc-user /app/iptvc /app/iptvc
USER iptvc-user
ENTRYPOINT ["/app/iptvc"]

View File

@@ -1,46 +1,46 @@
.DEFAULT_GOAL=help .DEFAULT_GOAL=help
BINARY_NAME=iptvc BINARY_NAME := iptvc
ARCH=amd64 GOARCH ?= amd64
LINUX_PATH="bin/linux_${ARCH}" LINUX_PATH := "bin/linux_$(GOARCH)"
WINDOWS_PATH="bin/windows_${ARCH}" WINDOWS_PATH := "bin/windows_$(GOARCH)"
DARWIN_PATH="bin/darwin_${ARCH}" DARWIN_PATH := "bin/darwin_$(GOARCH)"
LINUX_FILE="${LINUX_PATH}/${BINARY_NAME}" LINUX_FILE := "$(LINUX_PATH)/$(BINARY_NAME)"
WINDOWS_FILE="${WINDOWS_PATH}/${BINARY_NAME}.exe" WINDOWS_FILE := "$(WINDOWS_PATH)/$(BINARY_NAME).exe"
DARWIN_FILE="${DARWIN_PATH}/${BINARY_NAME}" DARWIN_FILE := "$(DARWIN_PATH)/$(BINARY_NAME)"
## clean: Remove all compiled binaries ## clean: Remove all compiled binaries
clean: clean:
@go clean @go clean
@rm -rf bin/ @rm -rf bin/
## linux: Build new binaries for linux (x64) ## linux: Build new binaries for linux
linux: linux:
@rm -rf ${LINUX_PATH} @rm -rf $(LINUX_PATH)
@GOARCH=${ARCH} GOOS=linux go build -o ${LINUX_FILE} . && echo "Compiled: ${LINUX_FILE}" @GOARCH=$(GOARCH) GOOS=linux go build -o $(LINUX_FILE) . && echo "Compiled: $(LINUX_FILE)"
## win: Build new binaries for windows (x64) ## win: Build new binaries for windows
win: win:
@rm -rf ${WINDOWS_PATH} @rm -rf $(WINDOWS_PATH)
@GOARCH=${ARCH} GOOS=windows go build -o ${WINDOWS_FILE} . && echo "Compiled: ${WINDOWS_FILE}" @GOARCH=$(GOARCH) GOOS=windows go build -o $(WINDOWS_FILE) . && echo "Compiled: $(WINDOWS_FILE)"
## darwin: Build new binaries for darwin (x64) ## darwin: Build new binaries for darwin
darwin: darwin:
@rm -rf ${DARWIN_PATH} @rm -rf $(DARWIN_PATH)
@GOARCH=${ARCH} GOOS=darwin go build -o ${DARWIN_FILE} . && echo "Compiled: ${DARWIN_FILE}" @GOARCH=$(GOARCH) GOOS=darwin go build -o $(DARWIN_FILE) . && echo "Compiled: $(DARWIN_FILE)"
## all: Build new binaries for linux, windows and darwin (x64) ## all: Build new binaries for linux, windows and darwin
all: clean linux win darwin all: clean linux win darwin
## release: Build all binaries and zip them ## release: Build all binaries and zip them
release: clean darwin linux win release: linux win darwin
@zip -j ${LINUX_PATH}.zip ${LINUX_FILE} @zip -j $(LINUX_PATH).zip $(LINUX_FILE)
@zip -j ${DARWIN_PATH}.zip ${DARWIN_FILE} @zip -j $(DARWIN_PATH).zip $(DARWIN_FILE)
@zip -j ${WINDOWS_PATH}.zip ${WINDOWS_FILE} @zip -j $(WINDOWS_PATH).zip $(WINDOWS_FILE)
## help: Show this message and exit ## help: Show this message and exit
help: Makefile help: Makefile
@echo "Choose a command run:" @echo "Available recipes:"
@sed -n 's/^##//p' $< | column -t -s ':' | sed -e 's/^/ /' @sed -n 's/^##//p' $< | column -t -s ':' | sed -e 's/^/ /'

View File

@@ -5,22 +5,21 @@
Консольная программа для проверки IPTV-плейлистов в формате m3u или m3u8. Консольная программа для проверки IPTV-плейлистов в формате m3u или m3u8.
> **Веб-сайт:** [m3u.su](https://m3u.su) > **Веб-сайт:** [m3u.su](https://m3u.su)
> Исходный код: [git.axenov.dev/IPTV/iptvc](https://git.axenov.dev/IPTV/iptvc) > **Документация:** [m3u.su/docs](https://m3u.su/docs)
> Исходный код: [git.axenov.dev/IPTV](https://git.axenov.dev/IPTV)
> Telegram-канал: [@iptv_aggregator](https://t.me/iptv_aggregator) > Telegram-канал: [@iptv_aggregator](https://t.me/iptv_aggregator)
> Обсуждение: [@iptv_aggregator_chat](https://t.me/iptv_aggregator_chat) > Обсуждение: [@iptv_aggregator_chat](https://t.me/iptv_aggregator_chat)
> Дополнительные сведения: > Бот: [@iptv_aggregator_bot](https://t.me/iptv_aggregator_bot)
> * [./docs](./docs)
> * [git.axenov.dev/IPTV/.profile](https://git.axenov.dev/IPTV/.profile)
## Установка ## Установка
Достаточно скачать и распаковать архив с подходящим исполняемым файлом [со страницы последнего релиза](https://git.axenov.dev/IPTV/iptvc/releases/latest): Достаточно скачать и распаковать архив с подходящим исполняемым файлом [со страницы последнего релиза](https://git.axenov.dev/IPTV/iptvc/releases/latest):
| ОС | Архив | Платформа | | ОС | Скачать для `amd64` | Скачать для `arm64` |
|---------|----------------------|-----------| | ------- | ------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------- |
| Linux | `linux_amd64.zip` | x64 | | Linux | [linux_amd64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/linux_amd64.zip) | [linux_arm64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/linux_arm64.zip) |
| MacOS | `darwin_amd64.zip` | x64 | | MacOS | [darwin_amd64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/darwin_amd64.zip) | [darwin_arm64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/darwin_arm64.zip) |
| Windows | `windows_amd64.zip` | x64 | | Windows | [windows_amd64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/windows_amd64.zip) | [windows_arm64.zip](https://git.axenov.dev/IPTV/iptvc/releases/download/latest/windows_arm64.zip) |
## Компиляция ## Компиляция
@@ -29,7 +28,7 @@
1. Склонировать репозиторий 1. Склонировать репозиторий
2. Находясь в корне репозитория, следует выполнить `make` или `make help` для получения справки. 2. Находясь в корне репозитория, следует выполнить `make` или `make help` для получения справки.
3. Другой способ -- выполнить `go run .` для быстрого запуска. 3. Другой способ выполнить `go run .` для быстрого запуска.
## Быстрый старт ## Быстрый старт
@@ -69,11 +68,11 @@
### Другие возможности команды `check` ### Другие возможности команды `check`
* `--random|-r X` -- проверить X случайных плейлистов из ini-файла * `--random|-r X` проверить X случайных плейлистов из ini-файла
* `--json|-j` -- вывести результаты проверки в формате JSON * `--json|-j` вывести результаты проверки в формате JSON
* `--quiet|-q` -- полностью подавить вывод лога (включая отладочную информацию) * `--quiet|-q` полностью подавить вывод лога (включая отладочную информацию)
* `--verbose|-v` -- добавить в лог более подробную отладочную информацию (значительно увеличит количество строк!) * `--verbose|-v` добавить в лог более подробную отладочную информацию (значительно увеличит количество строк!)
* `--tags|-t` -- файл с перечислением тегов (подробности см. [здесь](https://git.axenov.dev/IPTV/playlists#файл-channelsjson)) * `--tags|-t` файл с перечислением тегов (подробности см. [здесь](https://git.axenov.dev/IPTV/playlists#файл-channelsjson))
Например, можно получить только json с результатами, передать его в `jq` и, отфильтровав результат, вывести названия оффлайн каналов: Например, можно получить только json с результатами, передать его в `jq` и, отфильтровав результат, вывести названия оффлайн каналов:
@@ -126,8 +125,8 @@ pls='https://example.com/list2.m3u'
### Параметры проверки ### Параметры проверки
Выше в п.7 видно некоторые служебные данные: Выше в п.7 видно некоторые служебные данные:
* `timeout` -- таймаут каждого запроса в секундах (макс. время ожидания ответа канала); * `timeout` таймаут каждого запроса в секундах (макс. время ожидания ответа канала);
* `routines` -- количество одновременных проверок. * `routines` количество одновременных проверок.
Эти параметры рассчитываются динамически для каждого плейлиста в отдельности, исходя из количества каналов в каждом (`count`). Эти параметры рассчитываются динамически для каждого плейлиста в отдельности, исходя из количества каналов в каждом (`count`).
См. [app/checker/checker.go](app/checker/checker.go) для подробностей. См. [app/checker/checker.go](app/checker/checker.go) для подробностей.
@@ -151,9 +150,9 @@ pls='https://example.com/list2.m3u'
### Коды возврата ### Коды возврата
* 0 -- успех * 0 успех
* 1 -- общая ошибка, см. вывод * 1 общая ошибка, см. вывод
* 2 -- команде `check` не переданы параметры `--file`, `--url` и `--code` * 2 команде `check` не переданы параметры `--file`, `--url` и `--code`
## Лицензия ## Лицензия

View File

@@ -14,13 +14,15 @@ import (
"github.com/redis/go-redis/v9" "github.com/redis/go-redis/v9"
) )
const VERSION = "1.0.6" const VERSION = "1.1.1"
// Arguments описывает аргументы командной строки // Arguments описывает аргументы командной строки
type Arguments struct { type Arguments struct {
IniPath string IniPath string
TagsPath string TagsPath string
RandomCount uint RandomCount uint
RepeatCount uint
RepeatEverySec uint
NeedJson bool NeedJson bool
NeedQuiet bool NeedQuiet bool
Verbose bool Verbose bool

17
build-docker-image.sh Executable file
View File

@@ -0,0 +1,17 @@
#!/usr/bin/env bash
[[ "$1" ]] && DOCKER_TAG="$1" || DOCKER_TAG="latest"
[[ "$1" ]] && GIT_TAG="$1" || GIT_TAG=$(git describe --tags --always)
GIT_HASH=$(git rev-parse --short HEAD)
IPTVC_VERSION="${GIT_TAG}-${GIT_HASH}"
git checkout "${GIT_TAG}" 2>/dev/null
docker build \
--build-arg IPTVC_VERSION="${IPTVC_VERSION}" \
--build-arg GOOS="${GOOS:-linux}" \
--build-arg GOARCH="${GOARCH:-amd64}" \
--tag iptvc:"${DOCKER_TAG}" \
--tag git.axenov.dev/iptv/iptvc:"${DOCKER_TAG}" \
.
docker push git.axenov.dev/iptv/iptvc:"${DOCKER_TAG}"

View File

@@ -29,6 +29,21 @@ var checkCmd = &cobra.Command{
codes, _ := cmd.Flags().GetStringSlice("code") codes, _ := cmd.Flags().GetStringSlice("code")
lists := checker.PrepareListsToCheck(files, urls, codes) lists := checker.PrepareListsToCheck(files, urls, codes)
waitSeconds := app.Args.RepeatEverySec
if waitSeconds <= 0 {
waitSeconds = 5
}
currentIteration := 1
for {
if app.Args.RepeatCount != 1 {
log.Printf(
"@ New iteration current=%d count=%d\n",
currentIteration,
app.Args.RepeatCount,
)
}
startTime := time.Now() startTime := time.Now()
onlineCount, offlineCount := checker.CheckPlaylists(lists) onlineCount, offlineCount := checker.CheckPlaylists(lists)
@@ -44,6 +59,16 @@ var checkCmd = &cobra.Command{
marshal, _ := json.Marshal(lists) marshal, _ := json.Marshal(lists)
fmt.Println(string(marshal)) fmt.Println(string(marshal))
} }
if app.Args.RepeatCount != 0 {
if uint(currentIteration) == app.Args.RepeatCount {
break
}
currentIteration++
}
log.Printf("Waiting for new iteration... seconds=%d\n", app.Args.RepeatEverySec)
time.Sleep(time.Duration(app.Args.RepeatEverySec) * time.Second)
}
}, },
} }
@@ -51,6 +76,8 @@ func init() {
checkCmd.Flags().StringVarP(&app.Args.TagsPath, "tags", "t", "./channels.json", "path to a local tagfile") checkCmd.Flags().StringVarP(&app.Args.TagsPath, "tags", "t", "./channels.json", "path to a local tagfile")
checkCmd.Flags().StringVarP(&app.Args.IniPath, "ini", "i", "./playlists.ini", "path to a local ini-file") checkCmd.Flags().StringVarP(&app.Args.IniPath, "ini", "i", "./playlists.ini", "path to a local ini-file")
checkCmd.Flags().UintVarP(&app.Args.RandomCount, "random", "r", 0, "take this count of random playlists to check from ini-file") checkCmd.Flags().UintVarP(&app.Args.RandomCount, "random", "r", 0, "take this count of random playlists to check from ini-file")
checkCmd.Flags().UintVarP(&app.Args.RepeatCount, "repeat", "", 1, "repeat same check X times")
checkCmd.Flags().UintVarP(&app.Args.RepeatEverySec, "every", "", 5, "wait N seconds after every check")
checkCmd.Flags().BoolVarP(&app.Args.NeedJson, "json", "j", false, "print results in JSON format in the end") checkCmd.Flags().BoolVarP(&app.Args.NeedJson, "json", "j", false, "print results in JSON format in the end")
checkCmd.Flags().BoolVarP(&app.Args.NeedQuiet, "quiet", "q", false, "suppress logs (does not affect on -j)") checkCmd.Flags().BoolVarP(&app.Args.NeedQuiet, "quiet", "q", false, "suppress logs (does not affect on -j)")
checkCmd.Flags().StringSliceP("file", "f", []string{}, "path to a local playlist file (m3u/m3u8)") checkCmd.Flags().StringSliceP("file", "f", []string{}, "path to a local playlist file (m3u/m3u8)")