2024-01-22 10:36:06 +00:00
|
|
|
/*
|
2024-05-05 10:32:08 +00:00
|
|
|
* ALR - Any Linux Repository
|
|
|
|
* Copyright (C) 2024 Евгений Храмов
|
2024-01-22 10:36:06 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package overrides
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
|
2024-05-05 10:32:08 +00:00
|
|
|
"plemya-x.ru/alr/internal/cpu"
|
|
|
|
"plemya-x.ru/alr/internal/db"
|
|
|
|
"plemya-x.ru/alr/pkg/distro"
|
2024-01-22 10:36:06 +00:00
|
|
|
"golang.org/x/exp/slices"
|
|
|
|
"golang.org/x/text/language"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Opts struct {
|
|
|
|
Name string
|
|
|
|
Overrides bool
|
|
|
|
LikeDistros bool
|
|
|
|
Languages []string
|
|
|
|
LanguageTags []language.Tag
|
|
|
|
}
|
|
|
|
|
|
|
|
var DefaultOpts = &Opts{
|
|
|
|
Overrides: true,
|
|
|
|
LikeDistros: true,
|
|
|
|
Languages: []string{"en"},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resolve generates a slice of possible override names in the order that they should be checked
|
|
|
|
func Resolve(info *distro.OSRelease, opts *Opts) ([]string, error) {
|
|
|
|
if opts == nil {
|
|
|
|
opts = DefaultOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
if !opts.Overrides {
|
|
|
|
return []string{opts.Name}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
langs, err := parseLangs(opts.Languages, opts.LanguageTags)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
architectures, err := cpu.CompatibleArches(cpu.Arch())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
distros := []string{info.ID}
|
|
|
|
if opts.LikeDistros {
|
|
|
|
distros = append(distros, info.Like...)
|
|
|
|
}
|
|
|
|
|
|
|
|
var out []string
|
|
|
|
for _, lang := range langs {
|
|
|
|
for _, distro := range distros {
|
|
|
|
for _, arch := range architectures {
|
|
|
|
out = append(out, opts.Name+"_"+arch+"_"+distro+"_"+lang)
|
|
|
|
}
|
|
|
|
|
|
|
|
out = append(out, opts.Name+"_"+distro+"_"+lang)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, arch := range architectures {
|
|
|
|
out = append(out, opts.Name+"_"+arch+"_"+lang)
|
|
|
|
}
|
|
|
|
|
|
|
|
out = append(out, opts.Name+"_"+lang)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, distro := range distros {
|
|
|
|
for _, arch := range architectures {
|
|
|
|
out = append(out, opts.Name+"_"+arch+"_"+distro)
|
|
|
|
}
|
|
|
|
|
|
|
|
out = append(out, opts.Name+"_"+distro)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, arch := range architectures {
|
|
|
|
out = append(out, opts.Name+"_"+arch)
|
|
|
|
}
|
|
|
|
|
|
|
|
out = append(out, opts.Name)
|
|
|
|
|
|
|
|
for index, item := range out {
|
|
|
|
out[index] = strings.TrimPrefix(strings.ReplaceAll(item, "-", "_"), "_")
|
|
|
|
}
|
|
|
|
|
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Opts) WithName(name string) *Opts {
|
|
|
|
out := &Opts{}
|
|
|
|
*out = *o
|
|
|
|
|
|
|
|
out.Name = name
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Opts) WithOverrides(v bool) *Opts {
|
|
|
|
out := &Opts{}
|
|
|
|
*out = *o
|
|
|
|
|
|
|
|
out.Overrides = v
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Opts) WithLikeDistros(v bool) *Opts {
|
|
|
|
out := &Opts{}
|
|
|
|
*out = *o
|
|
|
|
|
|
|
|
out.LikeDistros = v
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Opts) WithLanguages(langs []string) *Opts {
|
|
|
|
out := &Opts{}
|
|
|
|
*out = *o
|
|
|
|
|
|
|
|
out.Languages = langs
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *Opts) WithLanguageTags(langs []string) *Opts {
|
|
|
|
out := &Opts{}
|
|
|
|
*out = *o
|
|
|
|
|
|
|
|
out.Languages = langs
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2024-05-05 10:32:08 +00:00
|
|
|
// ResolvedPackage is a ALR package after its overrides
|
2024-01-22 10:36:06 +00:00
|
|
|
// have been resolved
|
|
|
|
type ResolvedPackage struct {
|
|
|
|
Name string `sh:"name"`
|
|
|
|
Version string `sh:"version"`
|
|
|
|
Release int `sh:"release"`
|
|
|
|
Epoch uint `sh:"epoch"`
|
|
|
|
Description string `db:"description"`
|
|
|
|
Homepage string `db:"homepage"`
|
|
|
|
Maintainer string `db:"maintainer"`
|
|
|
|
Architectures []string `sh:"architectures"`
|
|
|
|
Licenses []string `sh:"license"`
|
|
|
|
Provides []string `sh:"provides"`
|
|
|
|
Conflicts []string `sh:"conflicts"`
|
|
|
|
Replaces []string `sh:"replaces"`
|
|
|
|
Depends []string `sh:"deps"`
|
|
|
|
BuildDepends []string `sh:"build_deps"`
|
|
|
|
OptDepends []string `sh:"opt_deps"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func ResolvePackage(pkg *db.Package, overrides []string) *ResolvedPackage {
|
|
|
|
out := &ResolvedPackage{}
|
|
|
|
outVal := reflect.ValueOf(out).Elem()
|
|
|
|
pkgVal := reflect.ValueOf(pkg).Elem()
|
|
|
|
|
|
|
|
for i := 0; i < outVal.NumField(); i++ {
|
|
|
|
fieldVal := outVal.Field(i)
|
|
|
|
fieldType := fieldVal.Type()
|
|
|
|
pkgFieldVal := pkgVal.FieldByName(outVal.Type().Field(i).Name)
|
|
|
|
pkgFieldType := pkgFieldVal.Type()
|
|
|
|
|
|
|
|
if strings.HasPrefix(pkgFieldType.String(), "db.JSON") {
|
|
|
|
pkgFieldVal = pkgFieldVal.FieldByName("Val")
|
|
|
|
pkgFieldType = pkgFieldVal.Type()
|
|
|
|
}
|
|
|
|
|
|
|
|
if pkgFieldType.AssignableTo(fieldType) {
|
|
|
|
fieldVal.Set(pkgFieldVal)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if pkgFieldVal.Kind() == reflect.Map && pkgFieldType.Elem().AssignableTo(fieldType) {
|
|
|
|
for _, override := range overrides {
|
|
|
|
overrideVal := pkgFieldVal.MapIndex(reflect.ValueOf(override))
|
|
|
|
if !overrideVal.IsValid() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
fieldVal.Set(overrideVal)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseLangs(langs []string, tags []language.Tag) ([]string, error) {
|
|
|
|
out := make([]string, len(tags)+len(langs))
|
|
|
|
for i, tag := range tags {
|
|
|
|
base, _ := tag.Base()
|
|
|
|
out[i] = base.String()
|
|
|
|
}
|
|
|
|
for i, lang := range langs {
|
|
|
|
tag, err := language.Parse(lang)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
base, _ := tag.Base()
|
|
|
|
out[len(tags)+i] = base.String()
|
|
|
|
}
|
|
|
|
slices.Sort(out)
|
|
|
|
out = slices.Compact(out)
|
|
|
|
return out, nil
|
|
|
|
}
|