Files
2026-06-14 19:37:09 +03:00

98 lines
3.5 KiB
Go

package layout
import (
"fmt"
"path/filepath"
"strings"
)
// sanitizeComponent чистит один компонент пути (имя папки/файла): убирает
// разделители, управляющие символы и неудобные для ФС/SMB знаки, схлопывает
// пробелы и срезает точки/пробелы по краям. Кириллица и пробелы внутри
// сохраняются. Результат гарантированно не содержит '/', '\\', ".." целиком
// и не пуст (иначе ошибка у вызывающего).
func sanitizeComponent(s string) string {
var b strings.Builder
for _, r := range s {
switch {
case r < 0x20 || r == 0x7f: // управляющие
b.WriteByte(' ')
case strings.ContainsRune(`/\:*?"<>|`, r): // разделители и недопустимые в SMB/NTFS
b.WriteByte(' ')
default:
b.WriteRune(r)
}
}
out := strings.Join(strings.Fields(b.String()), " ") // схлопнуть пробелы
out = strings.Trim(out, " .") // края: ни точек, ни пробелов
return out
}
// titleYear строит базу "Название (Год)" или "Название" при year == 0.
func titleYear(title string, year int) (string, error) {
t := sanitizeComponent(title)
if t == "" {
return "", fmt.Errorf("layout: empty title after sanitization (%q)", title)
}
if year > 0 {
return fmt.Sprintf("%s (%d)", t, year), nil
}
return t, nil
}
// folderName добавляет provider-тег к базе: "Название (Год) [tmdbid-123]".
func folderName(base, providerTag string) string {
tag := sanitizeComponent(providerTag)
if tag == "" {
return base
}
return fmt.Sprintf("%s [%s]", base, tag)
}
// seasonFolder — "Season 00" (спецвыпуски) / "Season 01" / ...
func seasonFolder(season int) string {
return fmt.Sprintf("Season %02d", season)
}
// episodeStem — "Название (Год) S01E02"; при двойной серии episodeEnd>episode
// даёт "S01E02-E03".
func episodeStem(base string, season, episode, episodeEnd int) string {
if episodeEnd > episode {
return fmt.Sprintf("%s S%02dE%02d-E%02d", base, season, episode, episodeEnd)
}
return fmt.Sprintf("%s S%02dE%02d", base, season, episode)
}
// subtitleSuffix — ".ru", ".ru.forced" и т.п. (флаги после языка).
func subtitleSuffix(lang string, flags []string) string {
var b strings.Builder
if l := sanitizeComponent(lang); l != "" {
b.WriteByte('.')
b.WriteString(strings.ToLower(l))
}
for _, f := range flags {
if f = sanitizeComponent(f); f != "" {
b.WriteByte('.')
b.WriteString(strings.ToLower(f))
}
}
return b.String()
}
// ext возвращает расширение файла источника в нижнем регистре (с точкой).
func ext(src string) string {
return strings.ToLower(filepath.Ext(src))
}
// underRoot сообщает, лежит ли clean-путь p строго под каталогом root
// (после filepath.Clean у обоих). Защита от traversal: даже если имя
// прошло санитизацию, итог обязан остаться внутри библиотеки.
func underRoot(root, p string) bool {
root = filepath.Clean(root)
p = filepath.Clean(p)
if p == root {
return false // сам корень — не цель для файла
}
return strings.HasPrefix(p, root+string(filepath.Separator))
}