9737c787f9
Инфраструктура M2M (живой обмен с НРД через ИШ): - обработка M2MTransferResponse: ERROR(M2Mxx) → заявка Отклонена, сохранение ответа; INFO → ждём Decision; идемпотентность поллера - fallback-корреляция ответов с нулевым GUID (M2M14/M2M17) по FIFO - сырой XML ответа НРД в карточке заявки (для пересылки в ТП) - тестовый пакет роботу приведён к эталону m2m_robot_samples (CostInfo=Yes, 4 бумаги, IsolationStatus, DocumentSeries=сценарий); override паспорта - редирект из теста сразу в карточку заявки Мастер установки ключа Валидаты на флешку (admin/setup/keywizard): - пошаговый: загрузка .7z+пароль → выбор флешки → запись → справочник сертификатов (CRL) → перезапуск+проверка ИШ → готово - привилегированный воркер (bj-keymedia) в host-namespace через файл-обмен, bj-server остаётся в песочнице - сохранение структуры профиля архива (spr<N>), перечисление съёмных USB Прочее: - пакет-доказательство для ТП НРД + форма регистрации участника M2M - эталонные образцы робота (DOC/m2m_robot_samples) Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
199 lines
8.3 KiB
Go
199 lines
8.3 KiB
Go
// Package release — формат манифеста релиза артефактории и его подпись
|
|
// Ed25519. Используется тремя сторонами:
|
|
//
|
|
// - cmd/bj-release — собирает манифест из артефактов и подписывает;
|
|
// - cmd/bj-artifactory — отдаёт манифест и файлы по HTTP;
|
|
// - bj-server (auto-update) — скачивает манифест, проверяет подпись,
|
|
// сравнивает версии и обновляет компоненты.
|
|
//
|
|
// Доверие строится на одном корневом Ed25519-ключе: приватный держит
|
|
// издатель (мы), публичный зашит в клиента (bj-server) и в установщик.
|
|
// Манифест подписывается целиком в каноничном виде — клиент проверяет
|
|
// подпись ДО того, как доверять версиям и хешам артефактов.
|
|
package release
|
|
|
|
import (
|
|
"crypto/ed25519"
|
|
"crypto/sha256"
|
|
"encoding/base64"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"time"
|
|
)
|
|
|
|
// CurrentSchema — версия формата манифеста (на случай эволюции).
|
|
const CurrentSchema = 1
|
|
|
|
// Artifact — один распространяемый файл (бинарь/скрипт/архив).
|
|
type Artifact struct {
|
|
Name string `json:"name"` // логическое имя: "bj-server", "install-validata.sh"
|
|
File string `json:"file"` // имя файла в хранилище
|
|
Version string `json:"version,omitempty"` // версия компонента (если своя)
|
|
SHA256 string `json:"sha256"` // hex-хеш содержимого
|
|
Size int64 `json:"size"` // размер в байтах
|
|
Exec bool `json:"exec,omitempty"` // ставить ли +x при установке
|
|
}
|
|
|
|
// Manifest — описание релиза. Подписывается целиком (каноничный JSON).
|
|
type Manifest struct {
|
|
Schema int `json:"schema"`
|
|
Version string `json:"version"` // версия релиза, напр. "1.2.0"
|
|
Channel string `json:"channel"` // "stable" | "beta"
|
|
ReleasedAt time.Time `json:"released_at"` // время сборки релиза
|
|
Notes string `json:"notes,omitempty"` // что нового (для UI)
|
|
MinVersion string `json:"min_version,omitempty"` // мин. версия для прямого апгрейда
|
|
Artifacts []Artifact `json:"artifacts"`
|
|
}
|
|
|
|
// SignedManifest — манифест + отделённая подпись. Именно это лежит в
|
|
// /v1/manifest.json и скачивается клиентом.
|
|
//
|
|
// Payload хранится как base64 от каноничного JSON Manifest — НЕ как
|
|
// вложенный JSON. Иначе json.MarshalIndent переформатировал бы содержимое
|
|
// и подписанные байты разошлись бы с прочитанными.
|
|
type SignedManifest struct {
|
|
Payload string `json:"payload"` // base64(каноничный JSON Manifest)
|
|
Signature string `json:"signature"` // base64(ed25519 over каноничным JSON)
|
|
KeyID string `json:"key_id"` // идентификатор публичного ключа
|
|
}
|
|
|
|
// Canonical сериализует манифест детерминированно (для подписи/проверки).
|
|
// json.Marshal в Go стабилен по порядку полей структуры — этого достаточно,
|
|
// т.к. подписываем и проверяем одни и те же байты Payload.
|
|
func (m *Manifest) Canonical() ([]byte, error) {
|
|
if m.Schema == 0 {
|
|
m.Schema = CurrentSchema
|
|
}
|
|
return json.Marshal(m)
|
|
}
|
|
|
|
// Sign подписывает манифест приватным ключом и возвращает SignedManifest.
|
|
func Sign(m *Manifest, priv ed25519.PrivateKey, keyID string) (*SignedManifest, error) {
|
|
payload, err := m.Canonical()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("release: canonical: %w", err)
|
|
}
|
|
sig := ed25519.Sign(priv, payload)
|
|
return &SignedManifest{
|
|
Payload: base64.StdEncoding.EncodeToString(payload),
|
|
Signature: base64.StdEncoding.EncodeToString(sig),
|
|
KeyID: keyID,
|
|
}, nil
|
|
}
|
|
|
|
// Verify проверяет подпись и возвращает разобранный Manifest. Это
|
|
// единственный способ получить доверенный Manifest на стороне клиента.
|
|
func Verify(sm *SignedManifest, pub ed25519.PublicKey) (*Manifest, error) {
|
|
sig, err := base64.StdEncoding.DecodeString(sm.Signature)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("release: decode signature: %w", err)
|
|
}
|
|
payload, err := base64.StdEncoding.DecodeString(sm.Payload)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("release: decode payload: %w", err)
|
|
}
|
|
if !ed25519.Verify(pub, payload, sig) {
|
|
return nil, errors.New("release: подпись манифеста недействительна")
|
|
}
|
|
var m Manifest
|
|
if err := json.Unmarshal(payload, &m); err != nil {
|
|
return nil, fmt.Errorf("release: unmarshal payload: %w", err)
|
|
}
|
|
if m.Schema != CurrentSchema {
|
|
return nil, fmt.Errorf("release: неподдерживаемая схема манифеста %d (ожидается %d)", m.Schema, CurrentSchema)
|
|
}
|
|
return &m, nil
|
|
}
|
|
|
|
// HashFile считает sha256 (hex) и размер файла — для заполнения Artifact.
|
|
func HashFile(path string) (sha string, size int64, err error) {
|
|
f, err := os.Open(path)
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
defer f.Close()
|
|
h := sha256.New()
|
|
n, err := io.Copy(h, f)
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
return hex.EncodeToString(h.Sum(nil)), n, nil
|
|
}
|
|
|
|
// VerifyArtifact проверяет, что файл по пути совпадает с Artifact (хеш+размер).
|
|
func VerifyArtifact(path string, a Artifact) error {
|
|
sha, size, err := HashFile(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if size != a.Size {
|
|
return fmt.Errorf("release: размер %s не совпал: %d != %d", a.Name, size, a.Size)
|
|
}
|
|
if sha != a.SHA256 {
|
|
return fmt.Errorf("release: sha256 %s не совпал", a.Name)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// --- Ключи ---
|
|
|
|
// LoadPrivateKey читает Ed25519 приватный ключ из файла (base64 seed, 32 байта).
|
|
func LoadPrivateKey(path string) (ed25519.PrivateKey, error) {
|
|
b, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
seed, err := base64.StdEncoding.DecodeString(stringsTrim(string(b)))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("release: decode seed: %w", err)
|
|
}
|
|
if len(seed) != ed25519.SeedSize {
|
|
return nil, fmt.Errorf("release: неверный размер seed %d (ожидается %d)", len(seed), ed25519.SeedSize)
|
|
}
|
|
return ed25519.NewKeyFromSeed(seed), nil
|
|
}
|
|
|
|
// LoadPublicKey читает Ed25519 публичный ключ из файла (base64, 32 байта).
|
|
func LoadPublicKey(path string) (ed25519.PublicKey, error) {
|
|
b, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
pub, err := base64.StdEncoding.DecodeString(stringsTrim(string(b)))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("release: decode pubkey: %w", err)
|
|
}
|
|
if len(pub) != ed25519.PublicKeySize {
|
|
return nil, fmt.Errorf("release: неверный размер pubkey %d", len(pub))
|
|
}
|
|
return ed25519.PublicKey(pub), nil
|
|
}
|
|
|
|
// ParsePublicKey разбирает публичный ключ из base64-строки (для зашитого в клиента).
|
|
func ParsePublicKey(b64 string) (ed25519.PublicKey, error) {
|
|
pub, err := base64.StdEncoding.DecodeString(stringsTrim(b64))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(pub) != ed25519.PublicKeySize {
|
|
return nil, fmt.Errorf("release: неверный размер pubkey %d", len(pub))
|
|
}
|
|
return ed25519.PublicKey(pub), nil
|
|
}
|
|
|
|
func stringsTrim(s string) string {
|
|
// убираем пробелы/переводы строк вокруг base64
|
|
start, end := 0, len(s)
|
|
for start < end && (s[start] == ' ' || s[start] == '\n' || s[start] == '\r' || s[start] == '\t') {
|
|
start++
|
|
}
|
|
for end > start && (s[end-1] == ' ' || s[end-1] == '\n' || s[end-1] == '\r' || s[end-1] == '\t') {
|
|
end--
|
|
}
|
|
return s[start:end]
|
|
}
|