feat(admin): блок «Новости» + doc-watcher + авто-уведомления о сертификатах УЦ
Новый раздел /admin/news — лента событий системы (окна техработ НРД, обновления документации, переустановка сертификатов УЦ). Каждая новость со временем, типом (maintenance/feature/doc-update/system/ manual), опциональным окном действительности (ValidFrom..ValidTo) и ссылкой на источник. Лента не очищается — служит журналом для аудита. На дашборде /admin/ — компактный блок «📢 Новости»: показывает максимум 3 актуальных события (активных сейчас или с окном, начинающимся в ближайшие 7 дней; в остатке — самые свежие). Окна техработ при наступлении становятся жёлтыми (border-left, ValidFrom..ValidTo). В ленте можно добавлять новости вручную (форма на /admin/news), скрывать (soft-delete через Dismissed). Дедуп по ID. Doc-watcher: горутина в bj-server, раз в сутки качает страницы НРД (дефолтные источники — moex-most, программы НРД, криптосервис), парсит HTML на ссылки .pdf, скачивает новые версии в DOC/ (со старыми переименовывая в .YYYY-MM-DD.pdf.bak для аудита), и публикует новость «Обновлена документация: <file>». Sha256-дедуп — пере-импорта неизменённого PDF не будет. Cacerts.go: FetchCACertificates теперь принимает *RuntimeConfig и при успешной переустановке сертификата эмитирует NewsItem «Обновлён сертификат УЦ: <CN>». Если сертификат истекает в ближайшие 14 дней — отдельная новость-предупреждение. Это закрывает запрос «получает в авто режиме и предупреждает об этом» из обсуждения. SeedDefaultNews публикует при старте bj-server две известные новости: - TEST3 недоступен 18.05.2026 — 22.05.2026 (НРД письмо НРД-И-2026-8452) - Робот-автотест MOEX МОСТ доступен на TEST3 с 12.05.2026 Скачаны три свежие инструкции с nsd.ru/services/novye-servisy/moex-most-dlya-m2m/: - DOC/instruktsiya-po-testirovaniyu-s-robotom.pdf (новая, 12.05.2026) - DOC/instruktsiya-dlya-osuschestvleniya-obmena-soobscheniyami-...-fizicheskim-litsom-samomu-sebe.pdf (новая, 12.05.2026) - DOC/servis-most-m2m.pdf (актуальная общая инструкция) Mastered tasks: #46, #47, #48.
This commit is contained in:
@@ -25,10 +25,43 @@ type Settings struct {
|
||||
NSD NSDSettings `json:"nsd"`
|
||||
LK LKSettings `json:"lk"`
|
||||
CACerts CACertsSettings `json:"ca_certs"`
|
||||
News NewsSettings `json:"news"`
|
||||
LastTest *TestRunResult `json:"last_test,omitempty"`
|
||||
UpdatedAt time.Time `json:"updated_at"`
|
||||
}
|
||||
|
||||
// NewsSettings — лента новостей (события системы, окна техработ, обновления
|
||||
// документации НРД). События добавляются вручную через UI или автоматически
|
||||
// doc-watcher'ом и cron-задачами. Каждое событие может быть скрыто (Dismissed)
|
||||
// оператором, но не удалено — лента служит «журналом» для аудита.
|
||||
type NewsSettings struct {
|
||||
Items []NewsItem `json:"items"`
|
||||
DocSources []DocSource `json:"doc_sources"` // URL'ы для авто-проверки (NSD pages)
|
||||
LastDocCheck time.Time `json:"last_doc_check"`
|
||||
DocCheckResult string `json:"doc_check_result"`
|
||||
}
|
||||
|
||||
// NewsItem — одно событие в ленте.
|
||||
type NewsItem struct {
|
||||
ID string `json:"id"` // уникальный идентификатор для dismiss
|
||||
At time.Time `json:"at"`
|
||||
Kind string `json:"kind"` // "maintenance" | "feature" | "doc-update" | "manual" | "system"
|
||||
Title string `json:"title"`
|
||||
Body string `json:"body"`
|
||||
URL string `json:"url,omitempty"` // ссылка на источник
|
||||
ValidFrom time.Time `json:"valid_from,omitempty"` // для maintenance окон
|
||||
ValidTo time.Time `json:"valid_to,omitempty"`
|
||||
Dismissed bool `json:"dismissed"`
|
||||
}
|
||||
|
||||
// DocSource — страница НРД, которую doc-watcher периодически проверяет.
|
||||
type DocSource struct {
|
||||
URL string `json:"url"`
|
||||
Name string `json:"name"` // человекочитаемое имя
|
||||
LastChecked time.Time `json:"last_checked"`
|
||||
KnownPDFs map[string]string `json:"known_pdfs"` // url → sha256
|
||||
}
|
||||
|
||||
// CACertsSettings — URL'ы для авто-загрузки сертификатов УЦ НРД и нашего
|
||||
// УЦ. Список редактируется пользователем; раз в сутки фоновая горутина
|
||||
// перекачивает каждый URL и переустанавливает сертификат, если он
|
||||
@@ -160,6 +193,55 @@ func (r *RuntimeConfig) UpdateCACerts(s CACertsSettings) error {
|
||||
return r.save()
|
||||
}
|
||||
|
||||
// UpdateNews заменяет всю ленту новостей.
|
||||
func (r *RuntimeConfig) UpdateNews(s NewsSettings) error {
|
||||
r.mu.Lock()
|
||||
r.data.News = s
|
||||
r.data.UpdatedAt = time.Now()
|
||||
r.mu.Unlock()
|
||||
return r.save()
|
||||
}
|
||||
|
||||
// AddNews добавляет новость в начало ленты (newest first). Если в ленте уже
|
||||
// есть новость с таким же ID — она обновляется (вместо дубликата).
|
||||
func (r *RuntimeConfig) AddNews(item NewsItem) error {
|
||||
r.mu.Lock()
|
||||
if item.ID == "" {
|
||||
item.ID = item.At.Format("20060102-150405") + "-" + item.Kind
|
||||
}
|
||||
if item.At.IsZero() {
|
||||
item.At = time.Now()
|
||||
}
|
||||
// Дедуп по ID.
|
||||
replaced := false
|
||||
for i, ex := range r.data.News.Items {
|
||||
if ex.ID == item.ID {
|
||||
r.data.News.Items[i] = item
|
||||
replaced = true
|
||||
break
|
||||
}
|
||||
}
|
||||
if !replaced {
|
||||
r.data.News.Items = append([]NewsItem{item}, r.data.News.Items...)
|
||||
}
|
||||
r.data.UpdatedAt = time.Now()
|
||||
r.mu.Unlock()
|
||||
return r.save()
|
||||
}
|
||||
|
||||
// DismissNews помечает новость скрытой по ID (не удаляет — для аудита).
|
||||
func (r *RuntimeConfig) DismissNews(id string) error {
|
||||
r.mu.Lock()
|
||||
for i := range r.data.News.Items {
|
||||
if r.data.News.Items[i].ID == id {
|
||||
r.data.News.Items[i].Dismissed = true
|
||||
}
|
||||
}
|
||||
r.data.UpdatedAt = time.Now()
|
||||
r.mu.Unlock()
|
||||
return r.save()
|
||||
}
|
||||
|
||||
func (r *RuntimeConfig) UpdateLK(s LKSettings) error {
|
||||
r.mu.Lock()
|
||||
r.data.LK = s
|
||||
|
||||
Reference in New Issue
Block a user