Compare commits
	
		
			56 Commits
		
	
	
		
			v0.0.5
			...
			7770675a8d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7770675a8d | |||
| bd79d56776 | |||
| fff8b777fe | |||
| 6bee268ea9 | |||
| 4b53e819d8 | |||
| 6c0e8aeb3f | |||
| cbc6b9f452 | |||
| c705c25613 | |||
| 8f4b021a93 | |||
| 5e7d4033e4 | |||
| 4ac2432770 | |||
| 3c37310f0d | |||
| d300ab197b | |||
| eb2cc3c1e6 | |||
| 7a3acfe5c1 | |||
| 9cf8af08ab | |||
| 86940e8962 | |||
| db244204c7 | |||
| 9cb0a5e9ad | |||
| 1a57ccdb83 | |||
| 615cd83fb7 | |||
| 27e2f54653 | |||
| af57165c89 | |||
| 3770c82240 | |||
| 2dff463303 | |||
| 9085e38454 | |||
| a7d016abc9 | |||
| 4a5cca2d0f | |||
| 71000fd3cd | |||
| 71968bbe13 | |||
| 29c1a31066 | |||
| 8f94b61a0e | |||
| ae8e2d2807 | |||
| 0fa288b8a2 | |||
| dcac0b9ee5 | |||
| 4e6e1f524a | |||
| 47a9b9a96c | |||
| 9bb14312bd | |||
| 88b8d2fbf3 | |||
| 04523775f1 | |||
| adc4a42800 | |||
| 81651af20d | |||
| f04ebbaf14 | |||
| be1a137eab | |||
| 719a5b7fe7 | |||
| e05bb07f23 | |||
| ec053f7e6a | |||
| ad1696d507 | |||
| a57602a278 | |||
| 606cd5473a | |||
| d2bcb4e345 | |||
| 1fcb88976c | |||
| 55feea9b25 | |||
| 0d1db212e1 | |||
| 99ec48c4c6 | |||
| d201aae6e0 | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -6,3 +6,7 @@ | |||||||
| .fleet | .fleet | ||||||
| .idea | .idea | ||||||
| .gigaide | .gigaide | ||||||
|  |  | ||||||
|  | *.out | ||||||
|  |  | ||||||
|  | e2e-tests/alr | ||||||
| @@ -44,3 +44,7 @@ issues: | |||||||
|     - path: _test\.go |     - path: _test\.go | ||||||
|       linters: |       linters: | ||||||
|         - errcheck |         - errcheck | ||||||
|  |     # TODO: remove | ||||||
|  |     - linters: | ||||||
|  |         - staticcheck | ||||||
|  |       text: "SA1019: interp.ExecHandler" | ||||||
							
								
								
									
										42
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | # ALR - Any Linux Repository | ||||||
|  | # Copyright (C) 2025 Евгений Храмов | ||||||
|  | # | ||||||
|  | # 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/>. | ||||||
|  |  | ||||||
|  | repos: | ||||||
|  |   - repo: local | ||||||
|  |     hooks: | ||||||
|  |       - id: test-coverage | ||||||
|  |         name: Run test coverage | ||||||
|  |         entry: make test-coverage | ||||||
|  |         language: system | ||||||
|  |         pass_filenames: false | ||||||
|  |  | ||||||
|  |       - id: fmt | ||||||
|  |         name: Format code | ||||||
|  |         entry: make fmt | ||||||
|  |         language: system | ||||||
|  |         pass_filenames: false | ||||||
|  |  | ||||||
|  |       - id: update-license | ||||||
|  |         name: Update license | ||||||
|  |         entry: make update-license | ||||||
|  |         language: system | ||||||
|  |         pass_filenames: false | ||||||
|  |  | ||||||
|  |       - id: i18n | ||||||
|  |         name: Update i18n | ||||||
|  |         entry: make i18n | ||||||
|  |         language: system | ||||||
|  |         pass_filenames: false | ||||||
							
								
								
									
										14
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								Makefile
									
									
									
									
									
								
							| @@ -12,7 +12,7 @@ INSTALLED_BASH_COMPLETION := $(DESTDIR)$(PREFIX)/share/bash-completion/completio | |||||||
| INSTALLED_ZSH_COMPLETION := $(DESTDIR)$(PREFIX)/share/zsh/site-functions/_$(NAME) | INSTALLED_ZSH_COMPLETION := $(DESTDIR)$(PREFIX)/share/zsh/site-functions/_$(NAME) | ||||||
|  |  | ||||||
| ADD_LICENSE_BIN := go run github.com/google/addlicense@4caba19b7ed7818bb86bc4cd20411a246aa4a524 | ADD_LICENSE_BIN := go run github.com/google/addlicense@4caba19b7ed7818bb86bc4cd20411a246aa4a524 | ||||||
| GOLANGCI_LINT_BIN := go run github.com/golangci/golangci-lint/cmd/golangci-lint@v1.62.2 | GOLANGCI_LINT_BIN := go run github.com/golangci/golangci-lint/cmd/golangci-lint@v1.63.4 | ||||||
| XGOTEXT_BIN := go run github.com/Tom5521/xgotext@v1.2.0 | XGOTEXT_BIN := go run github.com/Tom5521/xgotext@v1.2.0 | ||||||
|  |  | ||||||
| .PHONY: build install clean clear uninstall check-no-root | .PHONY: build install clean clear uninstall check-no-root | ||||||
| @@ -21,7 +21,7 @@ build: check-no-root $(BIN) | |||||||
|  |  | ||||||
| export CGO_ENABLED := 0 | export CGO_ENABLED := 0 | ||||||
| $(BIN): | $(BIN): | ||||||
| 	go build -ldflags="-X 'gitea.plemya-x.ru/xpamych/ALR/internal/config.Version=$(GIT_VERSION)'" -o $@ | 	go build -ldflags="-X 'gitea.plemya-x.ru/Plemya-x/ALR/internal/config.Version=$(GIT_VERSION)'" -o $@ | ||||||
|  |  | ||||||
| check-no-root: | check-no-root: | ||||||
| 	@if [[ "$$(whoami)" == 'root' ]]; then \ | 	@if [[ "$$(whoami)" == 'root' ]]; then \ | ||||||
| @@ -66,3 +66,13 @@ i18n: | |||||||
| 	$(XGOTEXT_BIN)  --output ./internal/translations/default.pot | 	$(XGOTEXT_BIN)  --output ./internal/translations/default.pot | ||||||
| 	msguniq --use-first -o ./internal/translations/default.pot ./internal/translations/default.pot | 	msguniq --use-first -o ./internal/translations/default.pot ./internal/translations/default.pot | ||||||
| 	msgmerge --backup=off -U ./internal/translations/po/ru/default.po ./internal/translations/default.pot | 	msgmerge --backup=off -U ./internal/translations/po/ru/default.po ./internal/translations/default.pot | ||||||
|  | 	bash scripts/i18n-badge.sh | ||||||
|  |  | ||||||
|  | test-coverage: | ||||||
|  | 	go test ./... -v -coverpkg=./... -coverprofile=coverage.out | ||||||
|  | 	bash scripts/coverage-badge.sh | ||||||
|  |  | ||||||
|  | e2e-test: clean build | ||||||
|  | 	rm -f ./e2e-tests/alr | ||||||
|  | 	cp alr e2e-tests | ||||||
|  | 	go test -tags=e2e ./... | ||||||
							
								
								
									
										24
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								README.md
									
									
									
									
									
								
							| @@ -3,11 +3,13 @@ | |||||||
| </p> | </p> | ||||||
| <b></b> | <b></b> | ||||||
|  |  | ||||||
|  | [](https://goreportcard.com/report/gitea.plemya-x.ru/Plemya-x/ALR)   | ||||||
|  |  | ||||||
| # ALR (Any Linux Repository) | # ALR (Any Linux Repository) | ||||||
|  |  | ||||||
| ALR - это независимая от дистрибутива система сборки для Linux, аналогичная [AUR](https://wiki.archlinux.org/title/Arch_User_Repository). В настоящее время она находится в стадии бета-тестирования. Исправлено большинство основных ошибок и добавлено большинство важных функций. alr готов к общему использованию, но все еще может время от времени ломаться или заменяться. | ALR - это независимая от дистрибутива система сборки для Linux (форк [LURE](https://github.com/lure-sh/lure), аналогичная [AUR](https://wiki.archlinux.org/title/Arch_User_Repository). В настоящее время она находится в стадии бета-тестирования. Исправлено большинство основных ошибок и добавлено большинство важных функций. ALR готов к общему использованию, но все еще может время от времени ломаться или изменяться. | ||||||
|  |  | ||||||
| ALR написан на чистом Go и после сборки не имеет зависимостей. Единственное, для повышения привилегий ALR требуется команда, такая как `sudo`, `doas` и т.д., а также поддерживаемый менеджер пакетов. В настоящее время ALR поддерживает `apt`, `pacman`, `apk`, `dnf`, `yum`, and `zypper`. Если в вашей системе существует поддерживаемый менеджер пакетов, он будет обнаружен и использован автоматически. | ALR написан на чистом Go и после сборки не имеет зависимостей. Для повышения привилегий ALR требуется команда, такая как `sudo`, `doas` и т.д., а также поддерживаемый менеджер пакетов. В настоящее время ALR поддерживает `apt`, `apt-get` `pacman`, `apk`, `dnf`, `yum`, and `zypper`. Если в вашей системе используется поддерживаемый менеджер пакетов, то он будет обнаружен и использован автоматически. | ||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
| @@ -21,14 +23,14 @@ ALR написан на чистом Go и после сборки не имее | |||||||
| curl -fsSL plemya-x.ru/alr/install.sh | bash | curl -fsSL plemya-x.ru/alr/install.sh | bash | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| **ВАЖНО**: При этом скрипт будет загружен и запущен с <https://gitea.plemya-x.ru/Plemya-x/ALR/src/branch/master/scripts/install.sh>. Пожалуйста, просматривайте любые скрипты, которые вы скачиваете из Интернета (включая этот), прежде чем запускать их. | **ВАЖНО**: При этом скрипт будет загружен и запущен с <https://plemya-x.ru/alr/install.sh>. Пожалуйста, просматривайте любые скрипты, которые вы скачиваете из Интернета (включая этот), прежде чем запускать их. | ||||||
|  |  | ||||||
| ### Сборка из исходного кода | ### Сборка из исходного кода | ||||||
|  |  | ||||||
| Чтобы собрать ALR из исходного кода, вам понадобится версия Go 1.18 или новее. Как только Go будет установлен, клонируйте это репозиторий и запустите: | Чтобы собрать ALR из исходного кода, вам понадобится версия Go 1.18 или новее. Как только Go будет установлен, клонируйте это репозиторий и запустите: | ||||||
|  |  | ||||||
| ```shell | ```shell | ||||||
| make build | make build -B | ||||||
| sudo make install | sudo make install | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| @@ -42,20 +44,23 @@ ALR был создан потому, что упаковка программн | |||||||
|  |  | ||||||
| ## Документация | ## Документация | ||||||
|  |  | ||||||
| Документация по всем этим вопросам находится в [Wiki](https://gitea.plemya-x.ru/xpamych/ALR/wiki/Home). | Документация находится в [Wiki](https://disc.plemya-x.ru/c/alr/wiki-alr). | ||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
| ## Репозитории | ## Репозитории | ||||||
|  |  | ||||||
| Репозитории alr - это git-хранилища, которые содержат каталог для каждого пакета с файлом `alr.sh` внутри. Файл `alr.sh` содержит все инструкции по сборке пакета и информацию о нем. Скрипты `alr.sh` аналогичны скриптам Aur PKGBUILD. Репозиторий [по-умолчанию](https://gitea.plemya-x.ru/xpamych/xpamych-alr-repo.git). | Репозитории alr - это git-хранилища, которые содержат каталог для каждого пакета с файлом `alr.sh` внутри. Файл `alr.sh` содержит все инструкции по сборке пакета и информацию о нем. Скрипты `alr.sh` аналогичны скриптам Aur PKGBUILD.  | ||||||
|  |  | ||||||
|  | Например, репозиторий [Plemya-x/alr-repo](https://gitea.plemya-x.ru/Plemya-x/alr-repo.git) можно подключить так: | ||||||
|  | ``` | ||||||
|  | alr addrepo --name alr-repo --url https://gitea.plemya-x.ru/Plemya-x/alr-repo.git | ||||||
|  | ``` | ||||||
|  |  | ||||||
| --- | --- | ||||||
| ## Соцсети | ## Соцсети | ||||||
| VK - https://vk.com/plemya_kh | VK - https://vk.com/plemya_kh | ||||||
|  |  | ||||||
| Discord - https://discord.com/channels/817759634105827358/1261631565084233749 |  | ||||||
|  |  | ||||||
| Telegram - https://t.me/plemyakh | Telegram - https://t.me/plemyakh | ||||||
|  |  | ||||||
| ## Спасибы | ## Спасибы | ||||||
| @@ -68,3 +73,6 @@ Telegram - https://t.me/plemyakh | |||||||
| - <https://github.com/goreleaser/nfpm> | - <https://github.com/goreleaser/nfpm> | ||||||
| - <https://github.com/charmbracelet/bubbletea> | - <https://github.com/charmbracelet/bubbletea> | ||||||
| - <https://gitlab.com/cznic/sqlite> | - <https://gitlab.com/cznic/sqlite> | ||||||
|  |  | ||||||
|  | Благодарим за активное участие в развитии проекта: | ||||||
|  | - Maks1mS <maxim@slipenko.com> | ||||||
							
								
								
									
										17
									
								
								assets/coverage-badge.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								assets/coverage-badge.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="109" height="20"> | ||||||
|  |     <linearGradient id="smooth" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/> | ||||||
|  |     <stop offset="1" stop-opacity=".1"/></linearGradient> | ||||||
|  |     <mask id="round"> | ||||||
|  |         <rect width="109" height="20" rx="3" fill="#fff"/> | ||||||
|  |     </mask> | ||||||
|  |     <g mask="url(#round)"><rect width="65" height="20" fill="#555"/> | ||||||
|  |         <rect x="65" width="44" height="20" fill="#e05d44"/> | ||||||
|  |         <rect width="109" height="20" fill="url(#smooth)"/> | ||||||
|  |     </g> | ||||||
|  |     <g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"> | ||||||
|  |         <text x="33.5" y="15" fill="#010101" fill-opacity=".3">coverage</text> | ||||||
|  |         <text x="33.5" y="14">coverage</text> | ||||||
|  |         <text x="86" y="15" fill="#010101" fill-opacity=".3">19.4%</text> | ||||||
|  |         <text x="86" y="14">19.4%</text> | ||||||
|  |     </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 926 B | 
							
								
								
									
										18
									
								
								assets/i18n-ru-badge.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								assets/i18n-ru-badge.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="129" height="20"> | ||||||
|  |     <linearGradient id="smooth" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/> | ||||||
|  |     <stop offset="1" stop-opacity=".1"/></linearGradient> | ||||||
|  |     <mask id="round"> | ||||||
|  |         <rect width="129" height="20" rx="3" fill="#fff"/> | ||||||
|  |     </mask> | ||||||
|  |     <g mask="url(#round)"> | ||||||
|  |         <rect width="75" height="20" fill="#555"/> | ||||||
|  |         <rect x="75" width="64" height="20" fill="#4c1"/> | ||||||
|  |         <rect width="129" height="20" fill="url(#smooth)"/> | ||||||
|  |     </g> | ||||||
|  |     <g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"> | ||||||
|  |         <text x="37" y="15" fill="#010101" fill-opacity=".3">ru translate</text> | ||||||
|  |         <text x="37" y="14">ru translate</text> | ||||||
|  |         <text x="100" y="15" fill="#010101" fill-opacity=".3">97.00%</text> | ||||||
|  |         <text x="100" y="14">97.00%</text> | ||||||
|  |     </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 940 B | 
							
								
								
									
										92
									
								
								build.go
									
									
									
									
									
								
							
							
						
						
									
										92
									
								
								build.go
									
									
									
									
									
								
							| @@ -23,14 +23,17 @@ import ( | |||||||
| 	"log/slog" | 	"log/slog" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
| 	"github.com/leonelquinteros/gotext" | 	"github.com/leonelquinteros/gotext" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
|  | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/osutils" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/osutils" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| ) | ) | ||||||
| @@ -46,6 +49,11 @@ func BuildCmd() *cli.Command { | |||||||
| 				Value:   "alr.sh", | 				Value:   "alr.sh", | ||||||
| 				Usage:   gotext.Get("Path to the build script"), | 				Usage:   gotext.Get("Path to the build script"), | ||||||
| 			}, | 			}, | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "subpackage", | ||||||
|  | 				Aliases: []string{"sb"}, | ||||||
|  | 				Usage:   gotext.Get("Specify subpackage in script (for multi package script only)"), | ||||||
|  | 			}, | ||||||
| 			&cli.StringFlag{ | 			&cli.StringFlag{ | ||||||
| 				Name:    "package", | 				Name:    "package", | ||||||
| 				Aliases: []string{"p"}, | 				Aliases: []string{"p"}, | ||||||
| @@ -59,31 +67,65 @@ func BuildCmd() *cli.Command { | |||||||
| 		}, | 		}, | ||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
|  | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			var script string | 			var script string | ||||||
|  | 			var packages []string | ||||||
|  | 			repository := "default" | ||||||
|  |  | ||||||
| 			// Проверяем, установлен ли флаг script (-s) | 			repoDir := cfg.GetPaths().RepoDir | ||||||
| 			if c.IsSet("script") { |  | ||||||
|  | 			switch { | ||||||
|  | 			case c.IsSet("script"): | ||||||
| 				script = c.String("script") | 				script = c.String("script") | ||||||
| 			} else if c.IsSet("package") { | 				packages = append(packages, c.String("script-package")) | ||||||
| 				// Если флаг script не установлен, проверяем флаг package (-p) | 			case c.IsSet("package"): | ||||||
|  | 				// TODO: handle multiple packages | ||||||
| 				packageInput := c.String("package") | 				packageInput := c.String("package") | ||||||
| 				// Определяем, содержит ли packageInput символ '/' |  | ||||||
| 				if filepath.Dir(packageInput) == "." { | 				arr := strings.Split(packageInput, "/") | ||||||
| 					// Не указана директория репозитория, используем 'default' как префикс | 				var packageSearch string | ||||||
| 					script = filepath.Join(config.GetPaths(ctx).RepoDir, "default", packageInput, "alr.sh") | 				if len(arr) == 2 { | ||||||
|  | 					packageSearch = arr[1] | ||||||
| 				} else { | 				} else { | ||||||
| 					// Используем путь с указанным репозиторием | 					packageSearch = arr[0] | ||||||
| 					script = filepath.Join(config.GetPaths(ctx).RepoDir, packageInput, "alr.sh") |  | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
|  | 				pkgs, _, _ := rs.FindPkgs(ctx, []string{packageSearch}) | ||||||
|  | 				pkg, ok := pkgs[packageSearch] | ||||||
|  | 				if len(pkg) < 1 || !ok { | ||||||
|  | 					slog.Error(gotext.Get("Package not found")) | ||||||
|  | 					os.Exit(1) | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				repository = pkg[0].Repository | ||||||
|  |  | ||||||
|  | 				if pkg[0].BasePkgName != "" { | ||||||
|  | 					script = filepath.Join(repoDir, repository, pkg[0].BasePkgName, "alr.sh") | ||||||
|  | 					packages = append(packages, pkg[0].Name) | ||||||
| 				} else { | 				} else { | ||||||
| 				// Ни флаги script, ни package не установлены, используем дефолтный скрипт | 					script = filepath.Join(repoDir, repository, pkg[0].Name, "alr.sh") | ||||||
| 				script = filepath.Join(config.GetPaths(ctx).RepoDir, "alr.sh") | 				} | ||||||
|  | 			default: | ||||||
|  | 				script = filepath.Join(repoDir, "alr.sh") | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			// Проверка автоматического пулла репозиториев | 			// Проверка автоматического пулла репозиториев | ||||||
| 			if config.GetInstance(ctx).AutoPull(ctx) { | 			if cfg.AutoPull() { | ||||||
| 				err := repos.Pull(ctx, config.Config(ctx).Repos) | 				err := rs.Pull(ctx, cfg.Repos()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| @@ -97,13 +139,29 @@ func BuildCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			// Сборка пакета | 			info, err := distro.ParseOSRelease(ctx) | ||||||
| 			pkgPaths, _, err := build.BuildPackage(ctx, types.BuildOpts{ | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error parsing os release"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			builder := build.NewBuilder( | ||||||
|  | 				ctx, | ||||||
|  | 				types.BuildOpts{ | ||||||
|  | 					Packages:    packages, | ||||||
|  | 					Repository:  repository, | ||||||
| 					Script:      script, | 					Script:      script, | ||||||
| 					Manager:     mgr, | 					Manager:     mgr, | ||||||
| 					Clean:       c.Bool("clean"), | 					Clean:       c.Bool("clean"), | ||||||
| 					Interactive: c.Bool("interactive"), | 					Interactive: c.Bool("interactive"), | ||||||
| 			}) | 				}, | ||||||
|  | 				rs, | ||||||
|  | 				info, | ||||||
|  | 				cfg, | ||||||
|  | 			) | ||||||
|  |  | ||||||
|  | 			// Сборка пакета | ||||||
|  | 			pkgPaths, _, err := builder.BuildPackage(ctx) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error building package"), "err", err) | 				slog.Error(gotext.Get("Error building package"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								e2e-tests/addrepo_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								e2e-tests/addrepo_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestE2EAlrAddRepo(t *testing.T) { | ||||||
|  | 	dockerMultipleRun( | ||||||
|  | 		t, | ||||||
|  | 		"add-repo-remove-repo", | ||||||
|  | 		COMMON_SYSTEMS, | ||||||
|  | 		func(t *testing.T, r e2e.Runnable) { | ||||||
|  | 			err := r.Exec(e2e.NewCommand( | ||||||
|  | 				"alr", | ||||||
|  | 				"addrepo", | ||||||
|  | 				"--name", | ||||||
|  | 				"alr-repo", | ||||||
|  | 				"--url", | ||||||
|  | 				"https://gitea.plemya-x.ru/Plemya-x/alr-repo.git", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = r.Exec(e2e.NewCommand( | ||||||
|  | 				"bash", | ||||||
|  | 				"-c", | ||||||
|  | 				"cat $HOME/.config/alr/alr.toml", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = r.Exec(e2e.NewCommand( | ||||||
|  | 				"alr", | ||||||
|  | 				"removerepo", | ||||||
|  | 				"--name", | ||||||
|  | 				"alr-repo", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			var buf bytes.Buffer | ||||||
|  | 			err = r.Exec(e2e.NewCommand( | ||||||
|  | 				"bash", | ||||||
|  | 				"-c", | ||||||
|  | 				"cat $HOME/.config/alr/alr.toml", | ||||||
|  | 			), e2e.WithExecOptionStdout(&buf)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			assert.Contains(t, buf.String(), "rootCmd") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
							
								
								
									
										180
									
								
								e2e-tests/common_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										180
									
								
								e2e-tests/common_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,180 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"crypto/sha256" | ||||||
|  | 	"encoding/hex" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
|  | 	"log" | ||||||
|  | 	"os" | ||||||
|  | 	"os/exec" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  |  | ||||||
|  | 	expect "github.com/tailscale/goexpect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // DebugWriter оборачивает io.Writer и логирует все записываемые данные. | ||||||
|  | type DebugWriter struct { | ||||||
|  | 	prefix string | ||||||
|  | 	writer io.Writer | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *DebugWriter) Write(p []byte) (n int, err error) { | ||||||
|  | 	log.Printf("%s: Writing data: %q", d.prefix, p) // Логируем данные | ||||||
|  | 	return d.writer.Write(p) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // DebugReader оборачивает io.Reader и логирует все читаемые данные. | ||||||
|  | type DebugReader struct { | ||||||
|  | 	prefix string | ||||||
|  | 	reader io.Reader | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (d *DebugReader) Read(p []byte) (n int, err error) { | ||||||
|  | 	n, err = d.reader.Read(p) | ||||||
|  | 	if n > 0 { | ||||||
|  | 		log.Printf("%s: Read data: %q", d.prefix, p[:n]) // Логируем данные | ||||||
|  | 	} | ||||||
|  | 	return n, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func e2eSpawn(runnable e2e.Runnable, command e2e.Command, timeout time.Duration, opts ...expect.Option) (expect.Expecter, <-chan error, error, *io.PipeWriter) { | ||||||
|  | 	resCh := make(chan error) | ||||||
|  |  | ||||||
|  | 	// Создаем pipe для stdin и stdout | ||||||
|  | 	stdinReader, stdinWriter := io.Pipe() | ||||||
|  | 	stdoutReader, stdoutWriter := io.Pipe() | ||||||
|  |  | ||||||
|  | 	debugStdinReader := &DebugReader{prefix: "STDIN", reader: stdinReader} | ||||||
|  | 	debugStdoutWriter := &DebugWriter{prefix: "STDOUT", writer: stdoutWriter} | ||||||
|  |  | ||||||
|  | 	go func() { | ||||||
|  | 		err := runnable.Exec( | ||||||
|  | 			command, | ||||||
|  | 			e2e.WithExecOptionStdout(debugStdoutWriter), | ||||||
|  | 			e2e.WithExecOptionStdin(debugStdinReader), | ||||||
|  | 			e2e.WithExecOptionStderr(debugStdoutWriter), | ||||||
|  | 		) | ||||||
|  |  | ||||||
|  | 		resCh <- err | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	exp, chnErr, err := expect.SpawnGeneric(&expect.GenOptions{ | ||||||
|  | 		In:  stdinWriter, | ||||||
|  | 		Out: stdoutReader, | ||||||
|  | 		Wait: func() error { | ||||||
|  | 			return <-resCh | ||||||
|  | 		}, | ||||||
|  | 		Close: func() error { | ||||||
|  | 			stdinWriter.Close() | ||||||
|  | 			stdoutReader.Close() | ||||||
|  | 			return nil | ||||||
|  | 		}, | ||||||
|  | 		Check: func() bool { return true }, | ||||||
|  | 	}, timeout, expect.Verbose(true), expect.VerboseWriter(os.Stdout)) | ||||||
|  |  | ||||||
|  | 	return exp, chnErr, err, stdinWriter | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var ALL_SYSTEMS []string = []string{ | ||||||
|  | 	"ubuntu-24.04", | ||||||
|  | 	// "alt-sisyphus", | ||||||
|  | 	// "archlinux", | ||||||
|  | 	// "alpine", | ||||||
|  | 	// "opensuse-leap", | ||||||
|  | 	// "redos-8", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var COMMON_SYSTEMS []string = []string{ | ||||||
|  | 	"ubuntu-24.04", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func init() { | ||||||
|  | 	for _, id := range ALL_SYSTEMS { | ||||||
|  | 		buildAlrTestImage(id) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func buildAlrTestImage(id string) { | ||||||
|  | 	cmd := exec.Command( | ||||||
|  | 		"docker", | ||||||
|  | 		"build", | ||||||
|  | 		"-t", fmt.Sprintf("alr-testimage-%s", id), | ||||||
|  | 		"-f", fmt.Sprintf("images/Dockerfile.%s", id), | ||||||
|  | 		".", | ||||||
|  | 	) | ||||||
|  | 	cmd.Stdout = os.Stdout | ||||||
|  | 	cmd.Stderr = os.Stderr | ||||||
|  | 	err := cmd.Run() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println("Error:", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func dockerMultipleRun(t *testing.T, name string, ids []string, f func(t *testing.T, runnable e2e.Runnable)) { | ||||||
|  | 	t.Run(name, func(t *testing.T) { | ||||||
|  | 		for _, id := range ids { | ||||||
|  | 			t.Run(id, func(t *testing.T) { | ||||||
|  | 				t.Parallel() | ||||||
|  | 				dockerName := fmt.Sprintf("alr-test-%s-%s", name, id) | ||||||
|  | 				hash := sha256.New() | ||||||
|  | 				hash.Write([]byte(dockerName)) | ||||||
|  | 				hashSum := hash.Sum(nil) | ||||||
|  | 				hashString := hex.EncodeToString(hashSum) | ||||||
|  | 				truncatedHash := hashString[:8] | ||||||
|  | 				e, err := e2e.New(e2e.WithVerbose(), e2e.WithName(fmt.Sprintf("alr-%s", truncatedHash))) | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  | 				t.Cleanup(e.Close) | ||||||
|  | 				imageId := fmt.Sprintf("alr-testimage-%s", id) | ||||||
|  | 				runnable := e.Runnable(dockerName).Init( | ||||||
|  | 					e2e.StartOptions{ | ||||||
|  | 						Image: imageId, | ||||||
|  | 						Volumes: []string{ | ||||||
|  | 							"./alr:/usr/bin/alr", | ||||||
|  | 						}, | ||||||
|  | 						Privileged: true, | ||||||
|  | 					}, | ||||||
|  | 				) | ||||||
|  | 				assert.NoError(t, e2e.StartAndWaitReady(runnable)) | ||||||
|  | 				f(t, runnable) | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func runTestCommands(t *testing.T, r e2e.Runnable, timeout time.Duration, expects []expect.Batcher) { | ||||||
|  | 	exp, _, err, _ := e2eSpawn( | ||||||
|  | 		r, | ||||||
|  | 		e2e.NewCommand("/bin/bash"), 25*time.Second, | ||||||
|  | 		expect.Verbose(true), | ||||||
|  | 	) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	_, err = exp.ExpectBatch( | ||||||
|  | 		expects, | ||||||
|  | 		timeout, | ||||||
|  | 	) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								e2e-tests/fix_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								e2e-tests/fix_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  | 	expect "github.com/tailscale/goexpect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestE2EAlrFix(t *testing.T) { | ||||||
|  | 	dockerMultipleRun( | ||||||
|  | 		t, | ||||||
|  | 		"run-fix", | ||||||
|  | 		COMMON_SYSTEMS, | ||||||
|  | 		func(t *testing.T, r e2e.Runnable) { | ||||||
|  | 			runTestCommands(t, r, time.Second*30, []expect.Batcher{ | ||||||
|  | 				&expect.BSnd{S: "alr fix\n"}, | ||||||
|  | 				&expect.BExp{R: `--> Done`}, | ||||||
|  | 				&expect.BSnd{S: "echo $?\n"}, | ||||||
|  | 				&expect.BExp{R: `^0\n$`}, | ||||||
|  | 			}) | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								e2e-tests/images/Dockerfile.alpine
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								e2e-tests/images/Dockerfile.alpine
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | FROM alpine:latest | ||||||
|  | RUN adduser -s /bin/bash alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										5
									
								
								e2e-tests/images/Dockerfile.alt-sisyphus
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								e2e-tests/images/Dockerfile.alt-sisyphus
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | FROM registry.altlinux.org/sisyphus/alt:latest | ||||||
|  | RUN apt-get update && apt-get install -y ca-certificates | ||||||
|  | RUN useradd -m -s /bin/bash alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										4
									
								
								e2e-tests/images/Dockerfile.archlinux
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								e2e-tests/images/Dockerfile.archlinux
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | FROM archlinux:latest | ||||||
|  | RUN useradd -m -s /bin/bash alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										4
									
								
								e2e-tests/images/Dockerfile.opensuse-leap
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								e2e-tests/images/Dockerfile.opensuse-leap
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | FROM opensuse/leap:latest | ||||||
|  | RUN useradd -m -s /bin/bash alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										4
									
								
								e2e-tests/images/Dockerfile.redos-8
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								e2e-tests/images/Dockerfile.redos-8
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | FROM registry.red-soft.ru/ubi8/ubi:latest | ||||||
|  | RUN useradd -m -s /bin/bash alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										7
									
								
								e2e-tests/images/Dockerfile.ubuntu-24.04
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								e2e-tests/images/Dockerfile.ubuntu-24.04
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | FROM ubuntu:24.10 | ||||||
|  | RUN apt-get update && apt-get install -y --no-install-recommends ca-certificates sudo | ||||||
|  | RUN useradd -m -s /bin/bash alr-user && \ | ||||||
|  |     echo "alr-user ALL=(ALL) NOPASSWD:ALL" >> /etc/sudoers.d/alr-user && \ | ||||||
|  |     chmod 0440 /etc/sudoers.d/alr-user | ||||||
|  | USER alr-user | ||||||
|  | ENTRYPOINT ["tail", "-f", "/dev/null"] | ||||||
							
								
								
									
										40
									
								
								e2e-tests/issue_32_interactive_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								e2e-tests/issue_32_interactive_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/alecthomas/assert/v2" | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestE2EIssue32Interactive(t *testing.T) { | ||||||
|  | 	dockerMultipleRun( | ||||||
|  | 		t, | ||||||
|  | 		"issue-32-interactive", | ||||||
|  | 		COMMON_SYSTEMS, | ||||||
|  | 		func(t *testing.T, r e2e.Runnable) { | ||||||
|  | 			err := r.Exec(e2e.NewCommand( | ||||||
|  | 				"alr", "--interactive=false", "remove", "ca-certificates", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								e2e-tests/issue_50_install_multiple_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								e2e-tests/issue_50_install_multiple_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/alecthomas/assert/v2" | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestE2EIssue50InstallMultiple(t *testing.T) { | ||||||
|  | 	dockerMultipleRun( | ||||||
|  | 		t, | ||||||
|  | 		"issue-50-install-multiple", | ||||||
|  | 		COMMON_SYSTEMS, | ||||||
|  | 		func(t *testing.T, r e2e.Runnable) { | ||||||
|  | 			err := r.Exec(e2e.NewCommand( | ||||||
|  | 				"alr", | ||||||
|  | 				"addrepo", | ||||||
|  | 				"--name", | ||||||
|  | 				"alr-repo", | ||||||
|  | 				"--url", | ||||||
|  | 				"https://gitea.plemya-x.ru/Maks1mS/repo-for-tests.git", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = r.Exec(e2e.NewCommand( | ||||||
|  | 				"alr", "in", "foo-pkg", "bar-pkg", | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = r.Exec(e2e.NewCommand("cat", "/opt/foo")) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			err = r.Exec(e2e.NewCommand("cat", "/opt/bar")) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
| @@ -14,39 +14,39 @@ | |||||||
| // You should have received a copy of the GNU General Public License | // You should have received a copy of the GNU General Public License | ||||||
| // along with this program.  If not, see <http://www.gnu.org/licenses/>. | // along with this program.  If not, see <http://www.gnu.org/licenses/>. | ||||||
| 
 | 
 | ||||||
| package config | //go:build e2e | ||||||
|  | 
 | ||||||
|  | package e2etests_test | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"testing" | ||||||
| 	"sync" |  | ||||||
| 
 | 
 | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"github.com/alecthomas/assert/v2" | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // Config returns a ALR configuration struct. | func TestE2EIssue53LcAllCInfo(t *testing.T) { | ||||||
| // The first time it's called, it'll load the config from a file. | 	dockerMultipleRun( | ||||||
| // Subsequent calls will just return the same value. | 		t, | ||||||
| // | 		"issue-53-lc-all-c-info", | ||||||
| // Deprecated: use struct method | 		COMMON_SYSTEMS, | ||||||
| func Config(ctx context.Context) *types.Config { | 		func(t *testing.T, r e2e.Runnable) { | ||||||
| 	return GetInstance(ctx).cfg | 			err := r.Exec(e2e.NewCommand( | ||||||
| } | 				"alr", | ||||||
| 
 | 				"addrepo", | ||||||
| // ======================= | 				"--name", | ||||||
| // FOR LEGACY ONLY | 				"alr-repo", | ||||||
| // ======================= | 				"--url", | ||||||
| 
 | 				"https://gitea.plemya-x.ru/Plemya-x/alr-repo.git", | ||||||
| var ( | 			)) | ||||||
| 	alrConfig     *ALRConfig | 			assert.NoError(t, err) | ||||||
| 	alrConfigOnce sync.Once | 
 | ||||||
| ) | 			err = r.Exec(e2e.NewCommand( | ||||||
| 
 | 				"bash", | ||||||
| // Deprecated: For legacy only | 				"-c", | ||||||
| func GetInstance(ctx context.Context) *ALRConfig { | 				"LANG=C alr info alr-bin", | ||||||
| 	alrConfigOnce.Do(func() { | 			)) | ||||||
| 		alrConfig = New() | 			assert.NoError(t, err) | ||||||
| 		alrConfig.Load(ctx) | 		}, | ||||||
| 	}) | 	) | ||||||
| 
 |  | ||||||
| 	return alrConfig |  | ||||||
| } | } | ||||||
							
								
								
									
										44
									
								
								e2e-tests/version_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								e2e-tests/version_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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/>. | ||||||
|  |  | ||||||
|  | //go:build e2e | ||||||
|  |  | ||||||
|  | package e2etests_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/efficientgo/e2e" | ||||||
|  |  | ||||||
|  | 	expect "github.com/tailscale/goexpect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestE2EAlrVersion(t *testing.T) { | ||||||
|  | 	dockerMultipleRun( | ||||||
|  | 		t, | ||||||
|  | 		"check-version", | ||||||
|  | 		COMMON_SYSTEMS, | ||||||
|  | 		func(t *testing.T, r e2e.Runnable) { | ||||||
|  | 			runTestCommands(t, r, time.Second*10, []expect.Batcher{ | ||||||
|  | 				&expect.BSnd{S: "alr version\n"}, | ||||||
|  | 				&expect.BExp{R: `^v\d+\.\d+\.\d+(?:-\d+-g[a-f0-9]+)?\n$`}, | ||||||
|  | 				&expect.BSnd{S: "echo $?\n"}, | ||||||
|  | 				&expect.BExp{R: `^0\n$`}, | ||||||
|  | 			}) | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
							
								
								
									
										29
									
								
								fix.go
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								fix.go
									
									
									
									
									
								
							| @@ -27,7 +27,7 @@ import ( | |||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -37,13 +37,18 @@ func FixCmd() *cli.Command { | |||||||
| 		Usage: gotext.Get("Attempt to fix problems with ALR"), | 		Usage: gotext.Get("Attempt to fix problems with ALR"), | ||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
|  | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			db.Close() | 			paths := cfg.GetPaths() | ||||||
| 			paths := config.GetPaths(ctx) |  | ||||||
|  |  | ||||||
| 			slog.Info(gotext.Get("Removing cache directory")) | 			slog.Info(gotext.Get("Removing cache directory")) | ||||||
|  |  | ||||||
| 			err := os.RemoveAll(paths.CacheDir) | 			err = os.RemoveAll(paths.CacheDir) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Unable to remove cache directory"), "err", err) | 				slog.Error(gotext.Get("Unable to remove cache directory"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| @@ -57,7 +62,21 @@ func FixCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			err = repos.Pull(ctx, config.Config(ctx).Repos) | 			cfg = config.New() | ||||||
|  | 			err = cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = rs.Pull(ctx, cfg.Repos()) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								go.mod
									
									
									
									
									
								
							| @@ -5,15 +5,20 @@ go 1.22 | |||||||
| toolchain go1.23.5 | toolchain go1.23.5 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
|  | 	gitea.plemya-x.ru/Plemya-x/fakeroot v0.0.1 | ||||||
| 	github.com/AlecAivazis/survey/v2 v2.3.7 | 	github.com/AlecAivazis/survey/v2 v2.3.7 | ||||||
| 	github.com/PuerkitoBio/purell v1.2.0 | 	github.com/PuerkitoBio/purell v1.2.0 | ||||||
|  | 	github.com/alecthomas/assert/v2 v2.2.1 | ||||||
| 	github.com/alecthomas/chroma/v2 v2.9.1 | 	github.com/alecthomas/chroma/v2 v2.9.1 | ||||||
|  | 	github.com/caarlos0/env v3.5.0+incompatible | ||||||
| 	github.com/charmbracelet/bubbles v0.20.0 | 	github.com/charmbracelet/bubbles v0.20.0 | ||||||
| 	github.com/charmbracelet/bubbletea v1.2.4 | 	github.com/charmbracelet/bubbletea v1.2.4 | ||||||
| 	github.com/charmbracelet/lipgloss v1.0.0 | 	github.com/charmbracelet/lipgloss v1.0.0 | ||||||
| 	github.com/charmbracelet/log v0.4.0 | 	github.com/charmbracelet/log v0.4.0 | ||||||
|  | 	github.com/efficientgo/e2e v0.14.1-0.20240418111536-97db25a0c6c0 | ||||||
| 	github.com/go-git/go-billy/v5 v5.5.0 | 	github.com/go-git/go-billy/v5 v5.5.0 | ||||||
| 	github.com/go-git/go-git/v5 v5.12.0 | 	github.com/go-git/go-git/v5 v5.12.0 | ||||||
|  | 	github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 | ||||||
| 	github.com/goreleaser/nfpm/v2 v2.41.0 | 	github.com/goreleaser/nfpm/v2 v2.41.0 | ||||||
| 	github.com/jeandeaual/go-locale v0.0.0-20241217141322-fcc2cadd6f08 | 	github.com/jeandeaual/go-locale v0.0.0-20241217141322-fcc2cadd6f08 | ||||||
| 	github.com/jmoiron/sqlx v1.3.5 | 	github.com/jmoiron/sqlx v1.3.5 | ||||||
| @@ -23,8 +28,8 @@ require ( | |||||||
| 	github.com/mitchellh/mapstructure v1.5.0 | 	github.com/mitchellh/mapstructure v1.5.0 | ||||||
| 	github.com/muesli/reflow v0.3.0 | 	github.com/muesli/reflow v0.3.0 | ||||||
| 	github.com/pelletier/go-toml/v2 v2.1.0 | 	github.com/pelletier/go-toml/v2 v2.1.0 | ||||||
| 	github.com/schollz/progressbar/v3 v3.18.0 |  | ||||||
| 	github.com/stretchr/testify v1.10.0 | 	github.com/stretchr/testify v1.10.0 | ||||||
|  | 	github.com/tailscale/goexpect v0.0.0-20210902213824-6e8c725cea41 | ||||||
| 	github.com/urfave/cli/v2 v2.25.7 | 	github.com/urfave/cli/v2 v2.25.7 | ||||||
| 	github.com/vmihailenco/msgpack/v5 v5.3.5 | 	github.com/vmihailenco/msgpack/v5 v5.3.5 | ||||||
| 	go.elara.ws/vercmp v0.0.0-20230622214216-0b2b067575c4 | 	go.elara.ws/vercmp v0.0.0-20230622214216-0b2b067575c4 | ||||||
| @@ -35,7 +40,6 @@ require ( | |||||||
| 	gopkg.in/yaml.v3 v3.0.1 | 	gopkg.in/yaml.v3 v3.0.1 | ||||||
| 	modernc.org/sqlite v1.25.0 | 	modernc.org/sqlite v1.25.0 | ||||||
| 	mvdan.cc/sh/v3 v3.10.0 | 	mvdan.cc/sh/v3 v3.10.0 | ||||||
| 	plemya-x.ru/fakeroot v0.0.0-20240601131003-c638a3543283 |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| @@ -46,6 +50,7 @@ require ( | |||||||
| 	github.com/Masterminds/sprig/v3 v3.2.3 // indirect | 	github.com/Masterminds/sprig/v3 v3.2.3 // indirect | ||||||
| 	github.com/Microsoft/go-winio v0.6.1 // indirect | 	github.com/Microsoft/go-winio v0.6.1 // indirect | ||||||
| 	github.com/ProtonMail/go-crypto v1.0.0 // indirect | 	github.com/ProtonMail/go-crypto v1.0.0 // indirect | ||||||
|  | 	github.com/alecthomas/repr v0.2.0 // indirect | ||||||
| 	github.com/andybalholm/brotli v1.0.4 // indirect | 	github.com/andybalholm/brotli v1.0.4 // indirect | ||||||
| 	github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect | 	github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect | ||||||
| 	github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb // indirect | 	github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb // indirect | ||||||
| @@ -58,13 +63,13 @@ require ( | |||||||
| 	github.com/charmbracelet/x/term v0.2.1 // indirect | 	github.com/charmbracelet/x/term v0.2.1 // indirect | ||||||
| 	github.com/cloudflare/circl v1.3.8 // indirect | 	github.com/cloudflare/circl v1.3.8 // indirect | ||||||
| 	github.com/connesc/cipherio v0.2.1 // indirect | 	github.com/connesc/cipherio v0.2.1 // indirect | ||||||
| 	github.com/containerd/console v1.0.4-0.20230313162750-1ae8d489ac81 // indirect |  | ||||||
| 	github.com/cpuguy83/go-md2man/v2 v2.0.4 // indirect | 	github.com/cpuguy83/go-md2man/v2 v2.0.4 // indirect | ||||||
| 	github.com/cyphar/filepath-securejoin v0.2.4 // indirect | 	github.com/cyphar/filepath-securejoin v0.2.4 // indirect | ||||||
| 	github.com/davecgh/go-spew v1.1.1 // indirect | 	github.com/davecgh/go-spew v1.1.1 // indirect | ||||||
| 	github.com/dlclark/regexp2 v1.10.0 // indirect | 	github.com/dlclark/regexp2 v1.10.0 // indirect | ||||||
| 	github.com/dsnet/compress v0.0.1 // indirect | 	github.com/dsnet/compress v0.0.1 // indirect | ||||||
| 	github.com/dustin/go-humanize v1.0.1 // indirect | 	github.com/dustin/go-humanize v1.0.1 // indirect | ||||||
|  | 	github.com/efficientgo/core v1.0.0-rc.0 // indirect | ||||||
| 	github.com/emirpasic/gods v1.18.1 // indirect | 	github.com/emirpasic/gods v1.18.1 // indirect | ||||||
| 	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect | 	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect | ||||||
| 	github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 // indirect | 	github.com/go-git/gcfg v1.5.1-0.20230307220236-3a3c6141e376 // indirect | ||||||
| @@ -72,13 +77,14 @@ require ( | |||||||
| 	github.com/gobwas/glob v0.2.3 // indirect | 	github.com/gobwas/glob v0.2.3 // indirect | ||||||
| 	github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect | 	github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect | ||||||
| 	github.com/golang/snappy v0.0.4 // indirect | 	github.com/golang/snappy v0.0.4 // indirect | ||||||
|  | 	github.com/google/goterm v0.0.0-20190703233501-fc88cf888a3f // indirect | ||||||
| 	github.com/google/rpmpack v0.6.1-0.20240329070804-c2247cbb881a // indirect | 	github.com/google/rpmpack v0.6.1-0.20240329070804-c2247cbb881a // indirect | ||||||
| 	github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect |  | ||||||
| 	github.com/google/uuid v1.4.0 // indirect | 	github.com/google/uuid v1.4.0 // indirect | ||||||
| 	github.com/goreleaser/chglog v0.6.1 // indirect | 	github.com/goreleaser/chglog v0.6.1 // indirect | ||||||
| 	github.com/goreleaser/fileglob v1.3.0 // indirect | 	github.com/goreleaser/fileglob v1.3.0 // indirect | ||||||
| 	github.com/hashicorp/errwrap v1.0.0 // indirect | 	github.com/hashicorp/errwrap v1.0.0 // indirect | ||||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||||
|  | 	github.com/hexops/gotextdiff v1.0.3 // indirect | ||||||
| 	github.com/huandu/xstrings v1.3.3 // indirect | 	github.com/huandu/xstrings v1.3.3 // indirect | ||||||
| 	github.com/imdario/mergo v0.3.16 // indirect | 	github.com/imdario/mergo v0.3.16 // indirect | ||||||
| 	github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect | 	github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect | ||||||
| @@ -91,7 +97,6 @@ require ( | |||||||
| 	github.com/mattn/go-localereader v0.0.1 // indirect | 	github.com/mattn/go-localereader v0.0.1 // indirect | ||||||
| 	github.com/mattn/go-runewidth v0.0.16 // indirect | 	github.com/mattn/go-runewidth v0.0.16 // indirect | ||||||
| 	github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b // indirect | 	github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b // indirect | ||||||
| 	github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db // indirect |  | ||||||
| 	github.com/mitchellh/copystructure v1.2.0 // indirect | 	github.com/mitchellh/copystructure v1.2.0 // indirect | ||||||
| 	github.com/mitchellh/reflectwalk v1.0.2 // indirect | 	github.com/mitchellh/reflectwalk v1.0.2 // indirect | ||||||
| 	github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect | 	github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect | ||||||
|   | |||||||
							
								
								
									
										66
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										66
									
								
								go.sum
									
									
									
									
									
								
							| @@ -17,6 +17,8 @@ cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0Zeo | |||||||
| dario.cat/mergo v1.0.1 h1:Ra4+bf83h2ztPIQYNP99R6m+Y7KfnARDfID+a+vLl4s= | dario.cat/mergo v1.0.1 h1:Ra4+bf83h2ztPIQYNP99R6m+Y7KfnARDfID+a+vLl4s= | ||||||
| dario.cat/mergo v1.0.1/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk= | dario.cat/mergo v1.0.1/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk= | ||||||
| dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | ||||||
|  | gitea.plemya-x.ru/Plemya-x/fakeroot v0.0.1 h1:c7F4OsyQbiVpSOrYGMrNsRL37BwoOfrgoKxAwULBKZo= | ||||||
|  | gitea.plemya-x.ru/Plemya-x/fakeroot v0.0.1/go.mod h1:iKQM6uttMJgE5CFrPw6SQqAV7TKtlJNICRAie/dTciw= | ||||||
| github.com/AlecAivazis/survey/v2 v2.3.7 h1:6I/u8FvytdGsgonrYsVn2t8t4QiRnh6QSTqkkhIiSjQ= | github.com/AlecAivazis/survey/v2 v2.3.7 h1:6I/u8FvytdGsgonrYsVn2t8t4QiRnh6QSTqkkhIiSjQ= | ||||||
| github.com/AlecAivazis/survey/v2 v2.3.7/go.mod h1:xUTIdE4KCOIjsBAE1JYsUPoCqYdZ1reCfTwbto0Fduo= | github.com/AlecAivazis/survey/v2 v2.3.7/go.mod h1:xUTIdE4KCOIjsBAE1JYsUPoCqYdZ1reCfTwbto0Fduo= | ||||||
| github.com/AlekSi/pointer v1.2.0 h1:glcy/gc4h8HnG2Z3ZECSzZ1IX1x2JxRVuDzaJwQE0+w= | github.com/AlekSi/pointer v1.2.0 h1:glcy/gc4h8HnG2Z3ZECSzZ1IX1x2JxRVuDzaJwQE0+w= | ||||||
| @@ -59,6 +61,8 @@ github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5 h1:0CwZNZbxp69SHPd | |||||||
| github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs= | github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs= | ||||||
| github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k= | github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k= | ||||||
| github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8= | github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8= | ||||||
|  | github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= | ||||||
|  | github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= | ||||||
| github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb h1:m935MPodAbYS46DG4pJSv7WO+VECIWUQ7OJYSoTrMh4= | github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb h1:m935MPodAbYS46DG4pJSv7WO+VECIWUQ7OJYSoTrMh4= | ||||||
| github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb/go.mod h1:PkYb9DJNAwrSvRx5DYA+gUcOIgTGVMNkfSCbZM8cWpI= | github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb/go.mod h1:PkYb9DJNAwrSvRx5DYA+gUcOIgTGVMNkfSCbZM8cWpI= | ||||||
| github.com/bodgit/plumbing v1.2.0 h1:gg4haxoKphLjml+tgnecR4yLBV5zo4HAZGCtAh3xCzM= | github.com/bodgit/plumbing v1.2.0 h1:gg4haxoKphLjml+tgnecR4yLBV5zo4HAZGCtAh3xCzM= | ||||||
| @@ -68,41 +72,29 @@ github.com/bodgit/sevenzip v1.3.0/go.mod h1:omwNcgZTEooWM8gA/IJ2Nk/+ZQ94+GsytRzO | |||||||
| github.com/bodgit/windows v1.0.0 h1:rLQ/XjsleZvx4fR1tB/UxQrK+SJ2OFHzfPjLWWOhDIA= | github.com/bodgit/windows v1.0.0 h1:rLQ/XjsleZvx4fR1tB/UxQrK+SJ2OFHzfPjLWWOhDIA= | ||||||
| github.com/bodgit/windows v1.0.0/go.mod h1:a6JLwrB4KrTR5hBpp8FI9/9W9jJfeQ2h4XDXU74ZCdM= | github.com/bodgit/windows v1.0.0/go.mod h1:a6JLwrB4KrTR5hBpp8FI9/9W9jJfeQ2h4XDXU74ZCdM= | ||||||
| github.com/bwesterb/go-ristretto v1.2.3/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7NFEuV9ekS419A0= | github.com/bwesterb/go-ristretto v1.2.3/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7NFEuV9ekS419A0= | ||||||
|  | github.com/caarlos0/env v3.5.0+incompatible h1:Yy0UN8o9Wtr/jGHZDpCBLpNrzcFLLM2yixi/rBrKyJs= | ||||||
|  | github.com/caarlos0/env v3.5.0+incompatible/go.mod h1:tdCsowwCzMLdkqRYDlHpZCp2UooDD3MspDBjZ2AD02Y= | ||||||
| github.com/caarlos0/testfs v0.4.4 h1:3PHvzHi5Lt+g332CiShwS8ogTgS3HjrmzZxCm6JCDr8= | github.com/caarlos0/testfs v0.4.4 h1:3PHvzHi5Lt+g332CiShwS8ogTgS3HjrmzZxCm6JCDr8= | ||||||
| github.com/caarlos0/testfs v0.4.4/go.mod h1:bRN55zgG4XCUVVHZCeU+/Tz1Q6AxEJOEJTliBy+1DMk= | github.com/caarlos0/testfs v0.4.4/go.mod h1:bRN55zgG4XCUVVHZCeU+/Tz1Q6AxEJOEJTliBy+1DMk= | ||||||
| github.com/cavaliergopher/cpio v1.0.1 h1:KQFSeKmZhv0cr+kawA3a0xTQCU4QxXF1vhU7P7av2KM= | github.com/cavaliergopher/cpio v1.0.1 h1:KQFSeKmZhv0cr+kawA3a0xTQCU4QxXF1vhU7P7av2KM= | ||||||
| github.com/cavaliergopher/cpio v1.0.1/go.mod h1:pBdaqQjnvXxdS/6CvNDwIANIFSP0xRKI16PX4xejRQc= | github.com/cavaliergopher/cpio v1.0.1/go.mod h1:pBdaqQjnvXxdS/6CvNDwIANIFSP0xRKI16PX4xejRQc= | ||||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||||
| github.com/charmbracelet/bubbles v0.16.1 h1:6uzpAAaT9ZqKssntbvZMlksWHruQLNxg49H5WdeuYSY= | github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | ||||||
| github.com/charmbracelet/bubbles v0.16.1/go.mod h1:2QCp9LFlEsBQMvIYERr7Ww2H2bA7xen1idUDIzm/+Xc= | github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||||
| github.com/charmbracelet/bubbles v0.20.0 h1:jSZu6qD8cRQ6k9OMfR1WlM+ruM8fkPWkHvQWD9LIutE= | github.com/charmbracelet/bubbles v0.20.0 h1:jSZu6qD8cRQ6k9OMfR1WlM+ruM8fkPWkHvQWD9LIutE= | ||||||
| github.com/charmbracelet/bubbles v0.20.0/go.mod h1:39slydyswPy+uVOHZ5x/GjwVAFkCsV8IIVy+4MhzwwU= | github.com/charmbracelet/bubbles v0.20.0/go.mod h1:39slydyswPy+uVOHZ5x/GjwVAFkCsV8IIVy+4MhzwwU= | ||||||
| github.com/charmbracelet/bubbletea v0.24.2 h1:uaQIKx9Ai6Gdh5zpTbGiWpytMU+CfsPp06RaW2cx/SY= |  | ||||||
| github.com/charmbracelet/bubbletea v0.24.2/go.mod h1:XdrNrV4J8GiyshTtx3DNuYkR1FDaJmO3l2nejekbsgg= |  | ||||||
| github.com/charmbracelet/bubbletea v1.1.0 h1:FjAl9eAL3HBCHenhz/ZPjkKdScmaS5SK69JAK2YJK9c= |  | ||||||
| github.com/charmbracelet/bubbletea v1.1.0/go.mod h1:9Ogk0HrdbHolIKHdjfFpyXJmiCzGwy+FesYkZr7hYU4= |  | ||||||
| github.com/charmbracelet/bubbletea v1.2.4 h1:KN8aCViA0eps9SCOThb2/XPIlea3ANJLUkv3KnQRNCE= | github.com/charmbracelet/bubbletea v1.2.4 h1:KN8aCViA0eps9SCOThb2/XPIlea3ANJLUkv3KnQRNCE= | ||||||
| github.com/charmbracelet/bubbletea v1.2.4/go.mod h1:Qr6fVQw+wX7JkWWkVyXYk/ZUQ92a6XNekLXa3rR18MM= | github.com/charmbracelet/bubbletea v1.2.4/go.mod h1:Qr6fVQw+wX7JkWWkVyXYk/ZUQ92a6XNekLXa3rR18MM= | ||||||
| github.com/charmbracelet/harmonica v0.2.0 h1:8NxJWRWg/bzKqqEaaeFNipOu77YR5t8aSwG4pgaUBiQ= | github.com/charmbracelet/harmonica v0.2.0 h1:8NxJWRWg/bzKqqEaaeFNipOu77YR5t8aSwG4pgaUBiQ= | ||||||
| github.com/charmbracelet/harmonica v0.2.0/go.mod h1:KSri/1RMQOZLbw7AHqgcBycp8pgJnQMYYT8QZRqZ1Ao= | github.com/charmbracelet/harmonica v0.2.0/go.mod h1:KSri/1RMQOZLbw7AHqgcBycp8pgJnQMYYT8QZRqZ1Ao= | ||||||
| github.com/charmbracelet/lipgloss v0.10.0 h1:KWeXFSexGcfahHX+54URiZGkBFazf70JNMtwg/AFW3s= |  | ||||||
| github.com/charmbracelet/lipgloss v0.10.0/go.mod h1:Wig9DSfvANsxqkRsqj6x87irdy123SR4dOXlKa91ciE= |  | ||||||
| github.com/charmbracelet/lipgloss v0.13.0 h1:4X3PPeoWEDCMvzDvGmTajSyYPcZM4+y8sCA/SsA3cjw= |  | ||||||
| github.com/charmbracelet/lipgloss v0.13.0/go.mod h1:nw4zy0SBX/F/eAO1cWdcvy6qnkDUxr8Lw7dvFrAIbbY= |  | ||||||
| github.com/charmbracelet/lipgloss v1.0.0 h1:O7VkGDvqEdGi93X+DeqsQ7PKHDgtQfF8j8/O2qFMQNg= | github.com/charmbracelet/lipgloss v1.0.0 h1:O7VkGDvqEdGi93X+DeqsQ7PKHDgtQfF8j8/O2qFMQNg= | ||||||
| github.com/charmbracelet/lipgloss v1.0.0/go.mod h1:U5fy9Z+C38obMs+T+tJqst9VGzlOYGj4ri9reL3qUlo= | github.com/charmbracelet/lipgloss v1.0.0/go.mod h1:U5fy9Z+C38obMs+T+tJqst9VGzlOYGj4ri9reL3qUlo= | ||||||
| github.com/charmbracelet/log v0.4.0 h1:G9bQAcx8rWA2T3pWvx7YtPTPwgqpk7D68BX21IRW8ZM= | github.com/charmbracelet/log v0.4.0 h1:G9bQAcx8rWA2T3pWvx7YtPTPwgqpk7D68BX21IRW8ZM= | ||||||
| github.com/charmbracelet/log v0.4.0/go.mod h1:63bXt/djrizTec0l11H20t8FDSvA4CRZJ1KH22MdptM= | github.com/charmbracelet/log v0.4.0/go.mod h1:63bXt/djrizTec0l11H20t8FDSvA4CRZJ1KH22MdptM= | ||||||
| github.com/charmbracelet/x/ansi v0.2.3 h1:VfFN0NUpcjBRd4DnKfRaIRo53KRgey/nhOoEqosGDEY= |  | ||||||
| github.com/charmbracelet/x/ansi v0.2.3/go.mod h1:dk73KoMTT5AX5BsX0KrqhsTqAnhZZoCBjs7dGWp4Ktw= |  | ||||||
| github.com/charmbracelet/x/ansi v0.4.5 h1:LqK4vwBNaXw2AyGIICa5/29Sbdq58GbGdFngSexTdRM= | github.com/charmbracelet/x/ansi v0.4.5 h1:LqK4vwBNaXw2AyGIICa5/29Sbdq58GbGdFngSexTdRM= | ||||||
| github.com/charmbracelet/x/ansi v0.4.5/go.mod h1:dk73KoMTT5AX5BsX0KrqhsTqAnhZZoCBjs7dGWp4Ktw= | github.com/charmbracelet/x/ansi v0.4.5/go.mod h1:dk73KoMTT5AX5BsX0KrqhsTqAnhZZoCBjs7dGWp4Ktw= | ||||||
| github.com/charmbracelet/x/term v0.2.0 h1:cNB9Ot9q8I711MyZ7myUR5HFWL/lc3OpU8jZ4hwm0x0= |  | ||||||
| github.com/charmbracelet/x/term v0.2.0/go.mod h1:GVxgxAbjUrmpvIINHIQnJJKpMlHiZ4cktEQCN6GWyF0= |  | ||||||
| github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ= | github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ= | ||||||
| github.com/charmbracelet/x/term v0.2.1/go.mod h1:oQ4enTYFV7QN4m0i9mzHrViD7TQKvNEEkHUMCmsxdUg= | github.com/charmbracelet/x/term v0.2.1/go.mod h1:oQ4enTYFV7QN4m0i9mzHrViD7TQKvNEEkHUMCmsxdUg= | ||||||
| github.com/chengxilo/virtualterm v1.0.4 h1:Z6IpERbRVlfB8WkOmtbHiDbBANU7cimRIof7mk9/PwM= |  | ||||||
| github.com/chengxilo/virtualterm v1.0.4/go.mod h1:DyxxBZz/x1iqJjFxTFcr6/x+jSpqN0iwWCOK1q10rlY= |  | ||||||
| github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= | github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= | ||||||
| github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= | github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= | ||||||
| github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= | github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= | ||||||
| @@ -112,8 +104,6 @@ github.com/cloudflare/circl v1.3.8 h1:j+V8jJt09PoeMFIu2uh5JUyEaIHTXVOHslFoLNAKqw | |||||||
| github.com/cloudflare/circl v1.3.8/go.mod h1:PDRU+oXvdD7KCtgKxW95M5Z8BpSCJXQORiZFnBQS5QU= | github.com/cloudflare/circl v1.3.8/go.mod h1:PDRU+oXvdD7KCtgKxW95M5Z8BpSCJXQORiZFnBQS5QU= | ||||||
| github.com/connesc/cipherio v0.2.1 h1:FGtpTPMbKNNWByNrr9aEBtaJtXjqOzkIXNYJp6OEycw= | github.com/connesc/cipherio v0.2.1 h1:FGtpTPMbKNNWByNrr9aEBtaJtXjqOzkIXNYJp6OEycw= | ||||||
| github.com/connesc/cipherio v0.2.1/go.mod h1:ukY0MWJDFnJEbXMQtOcn2VmTpRfzcTz4OoVrWGGJZcA= | github.com/connesc/cipherio v0.2.1/go.mod h1:ukY0MWJDFnJEbXMQtOcn2VmTpRfzcTz4OoVrWGGJZcA= | ||||||
| github.com/containerd/console v1.0.4-0.20230313162750-1ae8d489ac81 h1:q2hJAaP1k2wIvVRd/hEHD7lacgqrCPS+k8g1MndzfWY= |  | ||||||
| github.com/containerd/console v1.0.4-0.20230313162750-1ae8d489ac81/go.mod h1:YynlIjWYF8myEu6sdkwKIvGQq+cOckRm6So2avqoYAk= |  | ||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.4 h1:wfIWP927BUkWJb2NmU/kNDYIBTh/ziUX91+lVfRxZq4= | github.com/cpuguy83/go-md2man/v2 v2.0.4 h1:wfIWP927BUkWJb2NmU/kNDYIBTh/ziUX91+lVfRxZq4= | ||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.4/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= | github.com/cpuguy83/go-md2man/v2 v2.0.4/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= | ||||||
| github.com/creack/pty v1.1.17/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= | github.com/creack/pty v1.1.17/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= | ||||||
| @@ -131,6 +121,10 @@ github.com/dsnet/compress v0.0.1/go.mod h1:Aw8dCMJ7RioblQeTqt88akK31OvO8Dhf5Jflh | |||||||
| github.com/dsnet/golib v0.0.0-20171103203638-1ea166775780/go.mod h1:Lj+Z9rebOhdfkVLjJ8T6VcRQv3SXugXy999NBtR9aFY= | github.com/dsnet/golib v0.0.0-20171103203638-1ea166775780/go.mod h1:Lj+Z9rebOhdfkVLjJ8T6VcRQv3SXugXy999NBtR9aFY= | ||||||
| github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | ||||||
| github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | ||||||
|  | github.com/efficientgo/core v1.0.0-rc.0 h1:jJoA0N+C4/knWYVZ6GrdHOtDyrg8Y/TR4vFpTaqTsqs= | ||||||
|  | github.com/efficientgo/core v1.0.0-rc.0/go.mod h1:kQa0V74HNYMfuJH6jiPiwNdpWXl4xd/K4tzlrcvYDQI= | ||||||
|  | github.com/efficientgo/e2e v0.14.1-0.20240418111536-97db25a0c6c0 h1:C/FNIs+MtAJgQYLJ9FX/ACFYyDRuLYoXTmueErrOJyA= | ||||||
|  | github.com/efficientgo/e2e v0.14.1-0.20240418111536-97db25a0c6c0/go.mod h1:plsKU0YHE9uX+7utvr7SiDtVBSHJyEfHRO4UnUgDmts= | ||||||
| github.com/elazarl/goproxy v0.0.0-20230808193330-2592e75ae04a h1:mATvB/9r/3gvcejNsXKSkQ6lcIaNec2nyfOdlTBR2lU= | github.com/elazarl/goproxy v0.0.0-20230808193330-2592e75ae04a h1:mATvB/9r/3gvcejNsXKSkQ6lcIaNec2nyfOdlTBR2lU= | ||||||
| github.com/elazarl/goproxy v0.0.0-20230808193330-2592e75ae04a/go.mod h1:Ro8st/ElPeALwNFlcTpWmkr6IoMFfkjXAvTHpevnDsM= | github.com/elazarl/goproxy v0.0.0-20230808193330-2592e75ae04a/go.mod h1:Ro8st/ElPeALwNFlcTpWmkr6IoMFfkjXAvTHpevnDsM= | ||||||
| github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc= | github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc= | ||||||
| @@ -177,6 +171,8 @@ github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5y | |||||||
| github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
| github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | ||||||
|  | github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||||
|  | github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||||
| github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | ||||||
| github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||||
| github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||||
| @@ -187,6 +183,8 @@ github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMyw | |||||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||||
|  | github.com/google/goterm v0.0.0-20190703233501-fc88cf888a3f h1:5CjVwnuUcp5adK4gmY6i72gpVFVnZDP2h5TmPScB6u4= | ||||||
|  | github.com/google/goterm v0.0.0-20190703233501-fc88cf888a3f/go.mod h1:nOFQdrUlIlx6M6ODdSpBj1NVA+VgLC6kmw60mkw34H4= | ||||||
| github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= | github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= | ||||||
| github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||||
| github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||||
| @@ -233,6 +231,8 @@ github.com/jeandeaual/go-locale v0.0.0-20241217141322-fcc2cadd6f08 h1:wMeVzrPO3m | |||||||
| github.com/jeandeaual/go-locale v0.0.0-20241217141322-fcc2cadd6f08/go.mod h1:ZDXo8KHryOWSIqnsb/CiDq7hQUYryCgdVnxbj8tDG7o= | github.com/jeandeaual/go-locale v0.0.0-20241217141322-fcc2cadd6f08/go.mod h1:ZDXo8KHryOWSIqnsb/CiDq7hQUYryCgdVnxbj8tDG7o= | ||||||
| github.com/jmoiron/sqlx v1.3.5 h1:vFFPA71p1o5gAeqtEAwLU4dnX2napprKtHr7PYIcN3g= | github.com/jmoiron/sqlx v1.3.5 h1:vFFPA71p1o5gAeqtEAwLU4dnX2napprKtHr7PYIcN3g= | ||||||
| github.com/jmoiron/sqlx v1.3.5/go.mod h1:nRVWtLre0KfCLJvgxzCsLVMogSvQ1zNJtpYr2Ccp0mQ= | github.com/jmoiron/sqlx v1.3.5/go.mod h1:nRVWtLre0KfCLJvgxzCsLVMogSvQ1zNJtpYr2Ccp0mQ= | ||||||
|  | github.com/jpillora/backoff v1.0.0 h1:uvFg412JmmHBHw7iwprIxkPMI+sGQ4kzOWsMeHnm2EA= | ||||||
|  | github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= | ||||||
| github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= | github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= | ||||||
| github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= | github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= | ||||||
| github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= | github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= | ||||||
| @@ -278,12 +278,12 @@ github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh | |||||||
| github.com/mattn/go-sqlite3 v1.14.6/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU= | github.com/mattn/go-sqlite3 v1.14.6/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU= | ||||||
| github.com/mattn/go-sqlite3 v1.14.16 h1:yOQRA0RpS5PFz/oikGwBEqvAWhWg5ufRz4ETLjwpU1Y= | github.com/mattn/go-sqlite3 v1.14.16 h1:yOQRA0RpS5PFz/oikGwBEqvAWhWg5ufRz4ETLjwpU1Y= | ||||||
| github.com/mattn/go-sqlite3 v1.14.16/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= | github.com/mattn/go-sqlite3 v1.14.16/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= | ||||||
|  | github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= | ||||||
|  | github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | ||||||
| github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b h1:j7+1HpAFS1zy5+Q4qx1fWh90gTKwiN4QCGoY9TWyyO4= | github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b h1:j7+1HpAFS1zy5+Q4qx1fWh90gTKwiN4QCGoY9TWyyO4= | ||||||
| github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE= | github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE= | ||||||
| github.com/mholt/archiver/v4 v4.0.0-alpha.8 h1:tRGQuDVPh66WCOelqe6LIGh0gwmfwxUrSSDunscGsRM= | github.com/mholt/archiver/v4 v4.0.0-alpha.8 h1:tRGQuDVPh66WCOelqe6LIGh0gwmfwxUrSSDunscGsRM= | ||||||
| github.com/mholt/archiver/v4 v4.0.0-alpha.8/go.mod h1:5f7FUYGXdJWUjESffJaYR4R60VhnHxb2X3T1teMyv5A= | github.com/mholt/archiver/v4 v4.0.0-alpha.8/go.mod h1:5f7FUYGXdJWUjESffJaYR4R60VhnHxb2X3T1teMyv5A= | ||||||
| github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db h1:62I3jR2EmQ4l5rM/4FEfDWcRD+abF5XlKShorW5LRoQ= |  | ||||||
| github.com/mitchellh/colorstring v0.0.0-20190213212951-d06e56a500db/go.mod h1:l0dey0ia/Uv7NcFFVbCLtqEBQbrT4OCwCSKTEv6enCw= |  | ||||||
| github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= | github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= | ||||||
| github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= | github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= | ||||||
| github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= | github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= | ||||||
| @@ -292,8 +292,6 @@ github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RR | |||||||
| github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | ||||||
| github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= | github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= | ||||||
| github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= | ||||||
| github.com/muesli/ansi v0.0.0-20211018074035-2e021307bc4b h1:1XF24mVaiu7u+CFywTdcDo2ie1pzzhwjt6RHqzpMU34= |  | ||||||
| github.com/muesli/ansi v0.0.0-20211018074035-2e021307bc4b/go.mod h1:fQuZ0gauxyBcmsdE3ZT4NasjaRdxmbCS0jRHsrWu3Ho= |  | ||||||
| github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI= | github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI= | ||||||
| github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo= | github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo= | ||||||
| github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA= | github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA= | ||||||
| @@ -302,6 +300,8 @@ github.com/muesli/reflow v0.3.0 h1:IFsN6K9NfGtjeggFP+68I4chLZV2yIKsXJFNZ+eWh6s= | |||||||
| github.com/muesli/reflow v0.3.0/go.mod h1:pbwTDkVPibjO2kyvBQRBxTWEEGDGq0FlB1BIKtnHY/8= | github.com/muesli/reflow v0.3.0/go.mod h1:pbwTDkVPibjO2kyvBQRBxTWEEGDGq0FlB1BIKtnHY/8= | ||||||
| github.com/muesli/termenv v0.15.2 h1:GohcuySI0QmI3wN8Ok9PtKGkgkFIk7y6Vpb5PvrY+Wo= | github.com/muesli/termenv v0.15.2 h1:GohcuySI0QmI3wN8Ok9PtKGkgkFIk7y6Vpb5PvrY+Wo= | ||||||
| github.com/muesli/termenv v0.15.2/go.mod h1:Epx+iuz8sNs7mNKhxzH4fWXGNpZwUaJKRS1noLXviQ8= | github.com/muesli/termenv v0.15.2/go.mod h1:Epx+iuz8sNs7mNKhxzH4fWXGNpZwUaJKRS1noLXviQ8= | ||||||
|  | github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f h1:KUppIJq7/+SVif2QVs3tOP0zanoHgBEVAwHxUSIzRqU= | ||||||
|  | github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||||
| github.com/nwaples/rardecode/v2 v2.0.0-beta.2 h1:e3mzJFJs4k83GXBEiTaQ5HgSc/kOK8q0rDaRO0MPaOk= | github.com/nwaples/rardecode/v2 v2.0.0-beta.2 h1:e3mzJFJs4k83GXBEiTaQ5HgSc/kOK8q0rDaRO0MPaOk= | ||||||
| github.com/nwaples/rardecode/v2 v2.0.0-beta.2/go.mod h1:yntwv/HfMc/Hbvtq9I19D1n58te3h6KsqCf3GxyfBGY= | github.com/nwaples/rardecode/v2 v2.0.0-beta.2/go.mod h1:yntwv/HfMc/Hbvtq9I19D1n58te3h6KsqCf3GxyfBGY= | ||||||
| github.com/onsi/gomega v1.27.10 h1:naR28SdDFlqrG6kScpT8VWpu1xWY5nJRCF3XaYyBjhI= | github.com/onsi/gomega v1.27.10 h1:naR28SdDFlqrG6kScpT8VWpu1xWY5nJRCF3XaYyBjhI= | ||||||
| @@ -316,7 +316,15 @@ github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | |||||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
|  | github.com/prometheus/client_golang v1.12.1 h1:ZiaPsmm9uiBeaSMRznKsCDNtPCS0T3JVDGF+06gjBzk= | ||||||
|  | github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY= | ||||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||||
|  | github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= | ||||||
|  | github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||||
|  | github.com/prometheus/common v0.36.0 h1:78hJTing+BLYLjhXE+Z2BubeEymH5Lr0/Mt8FKkxxYo= | ||||||
|  | github.com/prometheus/common v0.36.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA= | ||||||
|  | github.com/prometheus/procfs v0.7.3 h1:4jVXhlkAyzOScmCkXBTOLRLTz8EeU+eyjrwB/EPq0VU= | ||||||
|  | github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= | github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | ||||||
| @@ -332,8 +340,6 @@ github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQD | |||||||
| github.com/rwcarlsen/goexif v0.0.0-20190401172101-9e8deecbddbd/go.mod h1:hPqNNc0+uJM6H+SuU8sEs5K5IQeKccPqeSjfgcKGgPk= | github.com/rwcarlsen/goexif v0.0.0-20190401172101-9e8deecbddbd/go.mod h1:hPqNNc0+uJM6H+SuU8sEs5K5IQeKccPqeSjfgcKGgPk= | ||||||
| github.com/sassoftware/go-rpmutils v0.4.0 h1:ojND82NYBxgwrV+mX1CWsd5QJvvEZTKddtCdFLPWhpg= | github.com/sassoftware/go-rpmutils v0.4.0 h1:ojND82NYBxgwrV+mX1CWsd5QJvvEZTKddtCdFLPWhpg= | ||||||
| github.com/sassoftware/go-rpmutils v0.4.0/go.mod h1:3goNWi7PGAT3/dlql2lv3+MSN5jNYPjT5mVcQcIsYzI= | github.com/sassoftware/go-rpmutils v0.4.0/go.mod h1:3goNWi7PGAT3/dlql2lv3+MSN5jNYPjT5mVcQcIsYzI= | ||||||
| github.com/schollz/progressbar/v3 v3.18.0 h1:uXdoHABRFmNIjUfte/Ex7WtuyVslrw2wVPQmCN62HpA= |  | ||||||
| github.com/schollz/progressbar/v3 v3.18.0/go.mod h1:IsO3lpbaGuzh8zIMzgY3+J8l4C8GjO0Y9S69eFvNsec= |  | ||||||
| github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 h1:n661drycOFuPLCN3Uc8sB6B/s6Z4t2xvBgU1htSHuq8= | github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 h1:n661drycOFuPLCN3Uc8sB6B/s6Z4t2xvBgU1htSHuq8= | ||||||
| github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4= | github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4= | ||||||
| github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXYbsQ= | github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXYbsQ= | ||||||
| @@ -360,6 +366,8 @@ github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO | |||||||
| github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= | github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= | ||||||
| github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | ||||||
| github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||||
|  | github.com/tailscale/goexpect v0.0.0-20210902213824-6e8c725cea41 h1:/V2rCMMWcsjYaYO2MeovLw+ClP63OtXgCF2Y1eb8+Ns= | ||||||
|  | github.com/tailscale/goexpect v0.0.0-20210902213824-6e8c725cea41/go.mod h1:/roCdA6gg6lQyw/Oz6gIIGu3ggJKYhF+WC/AQReE5XQ= | ||||||
| github.com/therootcompany/xz v1.0.1 h1:CmOtsn1CbtmyYiusbfmhmkpAAETj0wBIH6kCYaX+xzw= | github.com/therootcompany/xz v1.0.1 h1:CmOtsn1CbtmyYiusbfmhmkpAAETj0wBIH6kCYaX+xzw= | ||||||
| github.com/therootcompany/xz v1.0.1/go.mod h1:3K3UH1yCKgBneZYhuQUvJ9HPD19UEXEI0BWbMn8qNMY= | github.com/therootcompany/xz v1.0.1/go.mod h1:3K3UH1yCKgBneZYhuQUvJ9HPD19UEXEI0BWbMn8qNMY= | ||||||
| github.com/ulikunitz/xz v0.5.6/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8= | github.com/ulikunitz/xz v0.5.6/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8= | ||||||
| @@ -457,6 +465,8 @@ golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4Iltr | |||||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||||
| golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||||
| golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||||
|  | golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b h1:clP8eMhB30EHdc0bd2Twtq6kgU7yl5ub2cQLSdrv1Dg= | ||||||
|  | golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc= | ||||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| @@ -491,7 +501,6 @@ golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBc | |||||||
| golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| @@ -563,6 +572,8 @@ google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7 | |||||||
| google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||||
| google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= | google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= | ||||||
| google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | ||||||
|  | google.golang.org/appengine v1.6.6 h1:lMO5rYAqUxkmaj76jAkRUvt5JZgFymx/+Q5Mzfivuhc= | ||||||
|  | google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= | ||||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||||
| google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||||
| google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||||
| @@ -583,6 +594,8 @@ google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyac | |||||||
| google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||||
| google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||||
|  | google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||||
|  | google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| @@ -593,6 +606,7 @@ gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= | |||||||
| gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | ||||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
|  | gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= | ||||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
| @@ -632,8 +646,6 @@ modernc.org/z v1.7.3 h1:zDJf6iHjrnB+WRD88stbXokugjyc0/pB91ri1gO6LZY= | |||||||
| modernc.org/z v1.7.3/go.mod h1:Ipv4tsdxZRbQyLq9Q1M6gdbkxYzdlrciF2Hi/lS7nWE= | modernc.org/z v1.7.3/go.mod h1:Ipv4tsdxZRbQyLq9Q1M6gdbkxYzdlrciF2Hi/lS7nWE= | ||||||
| mvdan.cc/sh/v3 v3.10.0 h1:v9z7N1DLZ7owyLM/SXZQkBSXcwr2IGMm2LY2pmhVXj4= | mvdan.cc/sh/v3 v3.10.0 h1:v9z7N1DLZ7owyLM/SXZQkBSXcwr2IGMm2LY2pmhVXj4= | ||||||
| mvdan.cc/sh/v3 v3.10.0/go.mod h1:z/mSSVyLFGZzqb3ZIKojjyqIx/xbmz/UHdCSv9HmqXY= | mvdan.cc/sh/v3 v3.10.0/go.mod h1:z/mSSVyLFGZzqb3ZIKojjyqIx/xbmz/UHdCSv9HmqXY= | ||||||
| plemya-x.ru/fakeroot v0.0.0-20240601131003-c638a3543283 h1:BXCLPeA8g2M6qYngicyxyB/2Bo4J54Q9Rb+8jMmE3ik= |  | ||||||
| plemya-x.ru/fakeroot v0.0.0-20240601131003-c638a3543283/go.mod h1:itzL9Jx52VXOhRaucFHuMpa3y7iwjnuLGdNvypoh/S4= |  | ||||||
| rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= | rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= | ||||||
| rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= | rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= | ||||||
| rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= | rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								info.go
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								info.go
									
									
									
									
									
								
							| @@ -24,13 +24,14 @@ import ( | |||||||
| 	"log/slog" | 	"log/slog" | ||||||
| 	"os" | 	"os" | ||||||
|  |  | ||||||
|  | 	"github.com/jeandeaual/go-locale" | ||||||
| 	"github.com/leonelquinteros/gotext" | 	"github.com/leonelquinteros/gotext" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| 	"gopkg.in/yaml.v3" | 	"gopkg.in/yaml.v3" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cliutils" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cliutils" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/overrides" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/overrides" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| @@ -47,12 +48,52 @@ func InfoCmd() *cli.Command { | |||||||
| 				Usage:   gotext.Get("Show all information, not just for the current distro"), | 				Usage:   gotext.Get("Show all information, not just for the current distro"), | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
|  | 		BashComplete: func(c *cli.Context) { | ||||||
|  | 			ctx := c.Context | ||||||
|  | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			result, err := db.GetPkgs(c.Context, "true") | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error getting packages"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  | 			defer result.Close() | ||||||
|  |  | ||||||
|  | 			for result.Next() { | ||||||
|  | 				var pkg database.Package | ||||||
|  | 				err = result.StructScan(&pkg) | ||||||
|  | 				if err != nil { | ||||||
|  | 					slog.Error(gotext.Get("Error iterating over packages"), "err", err) | ||||||
|  | 					os.Exit(1) | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				fmt.Println(pkg.Name) | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
|  |  | ||||||
| 			cfg := config.New() | 			cfg := config.New() | ||||||
| 			db := db.New(cfg) | 			err := cfg.Load() | ||||||
| 			err := db.Init(ctx) | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error initialization database"), "err", err) | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| @@ -65,8 +106,8 @@ func InfoCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if cfg.AutoPull(ctx) { | 			if cfg.AutoPull() { | ||||||
| 				err := rs.Pull(ctx, cfg.Repos(ctx)) | 				err := rs.Pull(ctx, cfg.Repos()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error pulling repos"), "err", err) | 					slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| @@ -88,6 +129,15 @@ func InfoCmd() *cli.Command { | |||||||
| 			var names []string | 			var names []string | ||||||
| 			all := c.Bool("all") | 			all := c.Bool("all") | ||||||
|  |  | ||||||
|  | 			systemLang, err := locale.GetLanguage() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error("Can't detect system language", "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  | 			if systemLang == "" { | ||||||
|  | 				systemLang = "en" | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			if !all { | 			if !all { | ||||||
| 				info, err := distro.ParseOSRelease(ctx) | 				info, err := distro.ParseOSRelease(ctx) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| @@ -97,7 +147,7 @@ func InfoCmd() *cli.Command { | |||||||
| 				names, err = overrides.Resolve( | 				names, err = overrides.Resolve( | ||||||
| 					info, | 					info, | ||||||
| 					overrides.DefaultOpts. | 					overrides.DefaultOpts. | ||||||
| 						WithLanguages([]string{config.SystemLang()}), | 						WithLanguages([]string{systemLang}), | ||||||
| 				) | 				) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error resolving overrides"), "err", err) | 					slog.Error(gotext.Get("Error resolving overrides"), "err", err) | ||||||
|   | |||||||
							
								
								
									
										63
									
								
								install.go
									
									
									
									
									
								
							
							
						
						
									
										63
									
								
								install.go
									
									
									
									
									
								
							| @@ -29,9 +29,10 @@ import ( | |||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cliutils" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cliutils" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| ) | ) | ||||||
| @@ -45,7 +46,7 @@ func InstallCmd() *cli.Command { | |||||||
| 			&cli.BoolFlag{ | 			&cli.BoolFlag{ | ||||||
| 				Name:    "clean", | 				Name:    "clean", | ||||||
| 				Aliases: []string{"c"}, | 				Aliases: []string{"c"}, | ||||||
| 				Usage:   "Build package from scratch even if there's an already built package available", | 				Usage:   gotext.Get("Build package from scratch even if there's an already built package available"), | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| @@ -63,22 +64,58 @@ func InstallCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if config.GetInstance(ctx).AutoPull(ctx) { | 			cfg := config.New() | ||||||
| 				err := repos.Pull(ctx, config.Config(ctx).Repos) | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if cfg.AutoPull() { | ||||||
|  | 				err := rs.Pull(ctx, cfg.Repos()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			found, notFound, err := repos.FindPkgs(ctx, args.Slice()) | 			found, notFound, err := rs.FindPkgs(ctx, args.Slice()) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error finding packages"), "err", err) | 				slog.Error(gotext.Get("Error finding packages"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			pkgs := cliutils.FlattenPkgs(ctx, found, "install", c.Bool("interactive")) | 			pkgs := cliutils.FlattenPkgs(ctx, found, "install", c.Bool("interactive")) | ||||||
| 			build.InstallPkgs(ctx, pkgs, notFound, types.BuildOpts{ |  | ||||||
|  | 			opts := types.BuildOpts{ | ||||||
|  | 				Manager:     mgr, | ||||||
|  | 				Clean:       c.Bool("clean"), | ||||||
|  | 				Interactive: c.Bool("interactive"), | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			info, err := distro.ParseOSRelease(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error parsing os release"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			builder := build.NewBuilder( | ||||||
|  | 				ctx, | ||||||
|  | 				opts, | ||||||
|  | 				rs, | ||||||
|  | 				info, | ||||||
|  | 				cfg, | ||||||
|  | 			) | ||||||
|  |  | ||||||
|  | 			builder.InstallPkgs(ctx, pkgs, notFound, types.BuildOpts{ | ||||||
| 				Manager:     mgr, | 				Manager:     mgr, | ||||||
| 				Clean:       c.Bool("clean"), | 				Clean:       c.Bool("clean"), | ||||||
| 				Interactive: c.Bool("interactive"), | 				Interactive: c.Bool("interactive"), | ||||||
| @@ -86,6 +123,13 @@ func InstallCmd() *cli.Command { | |||||||
| 			return nil | 			return nil | ||||||
| 		}, | 		}, | ||||||
| 		BashComplete: func(c *cli.Context) { | 		BashComplete: func(c *cli.Context) { | ||||||
|  | 			cfg := config.New() | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err := db.Init(c.Context) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
| 			result, err := db.GetPkgs(c.Context, "true") | 			result, err := db.GetPkgs(c.Context, "true") | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error getting packages"), "err", err) | 				slog.Error(gotext.Get("Error getting packages"), "err", err) | ||||||
| @@ -94,7 +138,7 @@ func InstallCmd() *cli.Command { | |||||||
| 			defer result.Close() | 			defer result.Close() | ||||||
|  |  | ||||||
| 			for result.Next() { | 			for result.Next() { | ||||||
| 				var pkg db.Package | 				var pkg database.Package | ||||||
| 				err = result.StructScan(&pkg) | 				err = result.StructScan(&pkg) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error iterating over packages"), "err", err) | 					slog.Error(gotext.Get("Error iterating over packages"), "err", err) | ||||||
| @@ -125,7 +169,10 @@ func RemoveCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			err := mgr.Remove(nil, c.Args().Slice()...) | 			err := mgr.Remove(&manager.Opts{ | ||||||
|  | 				AsRoot:    true, | ||||||
|  | 				NoConfirm: !c.Bool("interactive"), | ||||||
|  | 			}, c.Args().Slice()...) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error removing packages"), "err", err) | 				slog.Error(gotext.Get("Error removing packages"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
|   | |||||||
							
								
								
									
										102
									
								
								internal/cliutils/template.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								internal/cliutils/template.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 cliutils | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"github.com/leonelquinteros/gotext" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Templates are based on https://github.com/urfave/cli/blob/3b17080d70a630feadadd23dd036cad121dd9a50/template.go | ||||||
|  |  | ||||||
|  | //nolint:unused | ||||||
|  | var ( | ||||||
|  | 	helpNameTemplate    = `{{$v := offset .HelpName 6}}{{wrap .HelpName 3}}{{if .Usage}} - {{wrap .Usage $v}}{{end}}` | ||||||
|  | 	descriptionTemplate = `{{wrap .Description 3}}` | ||||||
|  | 	authorsTemplate     = `{{with $length := len .Authors}}{{if ne 1 $length}}S{{end}}{{end}}: | ||||||
|  |    {{range $index, $author := .Authors}}{{if $index}} | ||||||
|  |    {{end}}{{$author}}{{end}}` | ||||||
|  | 	visibleCommandTemplate = `{{ $cv := offsetCommands .VisibleCommands 5}}{{range .VisibleCommands}} | ||||||
|  |    {{$s := join .Names ", "}}{{$s}}{{ $sp := subtract $cv (offset $s 3) }}{{ indent $sp ""}}{{wrap .Usage $cv}}{{end}}` | ||||||
|  | 	visibleCommandCategoryTemplate = `{{range .VisibleCategories}}{{if .Name}} | ||||||
|  |    {{.Name}}:{{range .VisibleCommands}} | ||||||
|  |      {{join .Names ", "}}{{"\t"}}{{.Usage}}{{end}}{{else}}{{template "visibleCommandTemplate" .}}{{end}}{{end}}` | ||||||
|  | 	visibleFlagCategoryTemplate = `{{range .VisibleFlagCategories}} | ||||||
|  |    {{if .Name}}{{.Name}} | ||||||
|  |  | ||||||
|  |    {{end}}{{$flglen := len .Flags}}{{range $i, $e := .Flags}}{{if eq (subtract $flglen $i) 1}}{{$e}} | ||||||
|  | {{else}}{{$e}} | ||||||
|  |    {{end}}{{end}}{{end}}` | ||||||
|  | 	visibleFlagTemplate = `{{range $i, $e := .VisibleFlags}} | ||||||
|  |    {{wrap $e.String 6}}{{end}}` | ||||||
|  | 	copyrightTemplate = `{{wrap .Copyright 3}}` | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func GetAppCliTemplate() string { | ||||||
|  | 	return fmt.Sprintf(`%s: | ||||||
|  | 	{{template "helpNameTemplate" .}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  | 	{{if .UsageText}}{{wrap .UsageText 3}}{{else}}{{.HelpName}} {{if .VisibleFlags}}[%s]{{end}}{{if .Commands}} %s [%s]{{end}} {{if .ArgsUsage}}{{.ArgsUsage}}{{else}}[%s...]{{end}}{{end}}{{if .Version}}{{if not .HideVersion}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  | 	{{.Version}}{{end}}{{end}}{{if .Description}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  |    {{template "descriptionTemplate" .}}{{end}} | ||||||
|  | {{- if len .Authors}} | ||||||
|  |  | ||||||
|  | %s{{template "authorsTemplate" .}}{{end}}{{if .VisibleCommands}} | ||||||
|  |  | ||||||
|  | %s:{{template "visibleCommandCategoryTemplate" .}}{{end}}{{if .VisibleFlagCategories}} | ||||||
|  |  | ||||||
|  | %s:{{template "visibleFlagCategoryTemplate" .}}{{else if .VisibleFlags}} | ||||||
|  |  | ||||||
|  | %s:{{template "visibleFlagTemplate" .}}{{end}}{{if .Copyright}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  |    {{template "copyrightTemplate" .}}{{end}} | ||||||
|  | `, gotext.Get("NAME"), gotext.Get("USAGE"), gotext.Get("global options"), gotext.Get("command"), gotext.Get("command options"), gotext.Get("arguments"), gotext.Get("VERSION"), gotext.Get("DESCRIPTION"), gotext.Get("AUTHOR"), gotext.Get("COMMANDS"), gotext.Get("GLOBAL OPTIONS"), gotext.Get("GLOBAL OPTIONS"), gotext.Get("COPYRIGHT")) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func GetCommandHelpTemplate() string { | ||||||
|  | 	return fmt.Sprintf(`%s: | ||||||
|  |    {{template "helpNameTemplate" .}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  |    {{if .UsageText}}{{wrap .UsageText 3}}{{else}}{{.HelpName}}{{if .VisibleFlags}} [%s]{{end}} {{if .ArgsUsage}}{{.ArgsUsage}}{{else}}[%s...]{{end}}{{end}}{{if .Category}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  |    {{.Category}}{{end}}{{if .Description}} | ||||||
|  |  | ||||||
|  | %s: | ||||||
|  |    {{template "descriptionTemplate" .}}{{end}}{{if .VisibleFlagCategories}} | ||||||
|  |  | ||||||
|  | %s:{{template "visibleFlagCategoryTemplate" .}}{{else if .VisibleFlags}} | ||||||
|  |  | ||||||
|  | %s:{{template "visibleFlagTemplate" .}}{{end}} | ||||||
|  | `, gotext.Get("NAME"), | ||||||
|  | 		gotext.Get("USAGE"), | ||||||
|  | 		gotext.Get("command options"), | ||||||
|  | 		gotext.Get("arguments"), | ||||||
|  | 		gotext.Get("CATEGORY"), | ||||||
|  | 		gotext.Get("DESCRIPTION"), | ||||||
|  | 		gotext.Get("OPTIONS"), | ||||||
|  | 		gotext.Get("OPTIONS"), | ||||||
|  | 	) | ||||||
|  | } | ||||||
| @@ -20,15 +20,14 @@ | |||||||
| package config | package config | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" |  | ||||||
| 	"log/slog" | 	"log/slog" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"sync" | 	"reflect" | ||||||
|  |  | ||||||
| 	"github.com/pelletier/go-toml/v2" |  | ||||||
|  |  | ||||||
|  | 	"github.com/caarlos0/env" | ||||||
| 	"github.com/leonelquinteros/gotext" | 	"github.com/leonelquinteros/gotext" | ||||||
|  | 	"github.com/pelletier/go-toml/v2" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| ) | ) | ||||||
| @@ -36,9 +35,6 @@ import ( | |||||||
| type ALRConfig struct { | type ALRConfig struct { | ||||||
| 	cfg   *types.Config | 	cfg   *types.Config | ||||||
| 	paths *Paths | 	paths *Paths | ||||||
|  |  | ||||||
| 	cfgOnce   sync.Once |  | ||||||
| 	pathsOnce sync.Once |  | ||||||
| } | } | ||||||
|  |  | ||||||
| var defaultConfig = &types.Config{ | var defaultConfig = &types.Config{ | ||||||
| @@ -46,127 +42,159 @@ var defaultConfig = &types.Config{ | |||||||
| 	PagerStyle:       "native", | 	PagerStyle:       "native", | ||||||
| 	IgnorePkgUpdates: []string{}, | 	IgnorePkgUpdates: []string{}, | ||||||
| 	AutoPull:         true, | 	AutoPull:         true, | ||||||
| 	Repos: []types.Repo{ | 	Repos:            []types.Repo{}, | ||||||
| 		{ |  | ||||||
| 			Name: "default", |  | ||||||
| 			URL:  "https://gitea.plemya-x.ru/xpamych/xpamych-alr-repo.git", |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func New() *ALRConfig { | func New() *ALRConfig { | ||||||
| 	return &ALRConfig{} | 	return &ALRConfig{} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *ALRConfig) Load(ctx context.Context) { | func readConfig(path string) (*types.Config, error) { | ||||||
| 	cfgFl, err := os.Open(c.GetPaths(ctx).ConfigPath) | 	file, err := os.Open(path) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Warn(gotext.Get("Error opening config file, using defaults"), "err", err) | 		return nil, err | ||||||
| 		c.cfg = defaultConfig |  | ||||||
| 		return |  | ||||||
| 	} | 	} | ||||||
| 	defer cfgFl.Close() | 	defer file.Close() | ||||||
|  |  | ||||||
| 	// Copy the default configuration into config | 	config := types.Config{} | ||||||
| 	defCopy := *defaultConfig |  | ||||||
| 	config := &defCopy |  | ||||||
| 	config.Repos = nil |  | ||||||
|  |  | ||||||
| 	err = toml.NewDecoder(cfgFl).Decode(config) | 	if err := toml.NewDecoder(file).Decode(&config); err != nil { | ||||||
| 	if err != nil { | 		return nil, err | ||||||
| 		slog.Warn(gotext.Get("Error decoding config file, using defaults"), "err", err) |  | ||||||
| 		c.cfg = defaultConfig |  | ||||||
| 		return |  | ||||||
| 	} | 	} | ||||||
| 	c.cfg = config |  | ||||||
|  | 	return &config, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *ALRConfig) initPaths() { | func mergeStructs(dst, src interface{}) { | ||||||
| 	paths := &Paths{} | 	srcVal := reflect.ValueOf(src) | ||||||
|  | 	if srcVal.IsNil() { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	srcVal = srcVal.Elem() | ||||||
|  | 	dstVal := reflect.ValueOf(dst).Elem() | ||||||
|  |  | ||||||
|  | 	for i := range srcVal.NumField() { | ||||||
|  | 		srcField := srcVal.Field(i) | ||||||
|  | 		srcFieldName := srcVal.Type().Field(i).Name | ||||||
|  |  | ||||||
|  | 		dstField := dstVal.FieldByName(srcFieldName) | ||||||
|  | 		if dstField.IsValid() && dstField.CanSet() { | ||||||
|  | 			dstField.Set(srcField) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const systemConfigPath = "/etc/alr/alr.toml" | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) Load() error { | ||||||
|  | 	systemConfig, err := readConfig( | ||||||
|  | 		systemConfigPath, | ||||||
|  | 	) | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Debug("Cannot read system config", "err", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	cfgDir, err := os.UserConfigDir() | 	cfgDir, err := os.UserConfigDir() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Unable to detect user config directory"), "err", err) | 		slog.Debug("Cannot read user config directory") | ||||||
| 		os.Exit(1) |  | ||||||
| 	} | 	} | ||||||
|  | 	userConfigPath := filepath.Join(cfgDir, "alr", "alr.toml") | ||||||
|  |  | ||||||
| 	paths.ConfigDir = filepath.Join(cfgDir, "alr") | 	userConfig, err := readConfig( | ||||||
|  | 		userConfigPath, | ||||||
| 	err = os.MkdirAll(paths.ConfigDir, 0o755) | 	) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Unable to create ALR config directory"), "err", err) | 		slog.Debug("Cannot read user config") | ||||||
| 		os.Exit(1) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	paths.ConfigPath = filepath.Join(paths.ConfigDir, "alr.toml") | 	config := &types.Config{} | ||||||
|  |  | ||||||
| 	if _, err := os.Stat(paths.ConfigPath); err != nil { | 	mergeStructs(config, defaultConfig) | ||||||
| 		cfgFl, err := os.Create(paths.ConfigPath) | 	mergeStructs(config, systemConfig) | ||||||
|  | 	mergeStructs(config, userConfig) | ||||||
|  | 	err = env.Parse(config) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 			slog.Error(gotext.Get("Unable to create ALR config file"), "err", err) | 		return err | ||||||
| 			os.Exit(1) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 		err = toml.NewEncoder(cfgFl).Encode(&defaultConfig) | 	c.cfg = config | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error encoding default configuration"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		cfgFl.Close() |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	cacheDir, err := os.UserCacheDir() | 	cacheDir, err := os.UserCacheDir() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Unable to detect cache directory"), "err", err) | 		return err | ||||||
|  | 	} | ||||||
|  | 	c.paths = &Paths{} | ||||||
|  | 	c.paths.UserConfigPath = userConfigPath | ||||||
|  | 	c.paths.CacheDir = filepath.Join(cacheDir, "alr") | ||||||
|  | 	c.paths.RepoDir = filepath.Join(c.paths.CacheDir, "repo") | ||||||
|  | 	c.paths.PkgsDir = filepath.Join(c.paths.CacheDir, "pkgs") | ||||||
|  | 	c.paths.DBPath = filepath.Join(c.paths.CacheDir, "db") | ||||||
|  | 	c.initPaths() | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) RootCmd() string { | ||||||
|  | 	return c.cfg.RootCmd | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) PagerStyle() string { | ||||||
|  | 	return c.cfg.PagerStyle | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) AutoPull() bool { | ||||||
|  | 	return c.cfg.AutoPull | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) AllowRunAsRoot() bool { | ||||||
|  | 	return c.cfg.Unsafe.AllowRunAsRoot | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) Repos() []types.Repo { | ||||||
|  | 	return c.cfg.Repos | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) SetRepos(repos []types.Repo) { | ||||||
|  | 	c.cfg.Repos = repos | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) IgnorePkgUpdates() []string { | ||||||
|  | 	return c.cfg.IgnorePkgUpdates | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) LogLevel() string { | ||||||
|  | 	return c.cfg.LogLevel | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) GetPaths() *Paths { | ||||||
|  | 	return c.paths | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *ALRConfig) initPaths() { | ||||||
|  | 	err := os.MkdirAll(filepath.Dir(c.paths.UserConfigPath), 0o755) | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Error(gotext.Get("Unable to create config directory"), "err", err) | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	paths.CacheDir = filepath.Join(cacheDir, "alr") | 	err = os.MkdirAll(c.paths.RepoDir, 0o755) | ||||||
| 	paths.RepoDir = filepath.Join(paths.CacheDir, "repo") |  | ||||||
| 	paths.PkgsDir = filepath.Join(paths.CacheDir, "pkgs") |  | ||||||
|  |  | ||||||
| 	err = os.MkdirAll(paths.RepoDir, 0o755) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Unable to create repo cache directory"), "err", err) | 		slog.Error(gotext.Get("Unable to create repo cache directory"), "err", err) | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = os.MkdirAll(paths.PkgsDir, 0o755) | 	err = os.MkdirAll(c.paths.PkgsDir, 0o755) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Unable to create package cache directory"), "err", err) | 		slog.Error(gotext.Get("Unable to create package cache directory"), "err", err) | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	paths.DBPath = filepath.Join(paths.CacheDir, "db") |  | ||||||
|  |  | ||||||
| 	c.paths = paths |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *ALRConfig) GetPaths(ctx context.Context) *Paths { | func (c *ALRConfig) SaveUserConfig() error { | ||||||
| 	c.pathsOnce.Do(func() { | 	f, err := os.Create(c.paths.UserConfigPath) | ||||||
| 		c.initPaths() | 	if err != nil { | ||||||
| 	}) | 		return err | ||||||
| 	return c.paths | 	} | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c *ALRConfig) Repos(ctx context.Context) []types.Repo { | 	return toml.NewEncoder(f).Encode(c.cfg) | ||||||
| 	c.cfgOnce.Do(func() { |  | ||||||
| 		c.Load(ctx) |  | ||||||
| 	}) |  | ||||||
| 	return c.cfg.Repos |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c *ALRConfig) IgnorePkgUpdates(ctx context.Context) []string { |  | ||||||
| 	c.cfgOnce.Do(func() { |  | ||||||
| 		c.Load(ctx) |  | ||||||
| 	}) |  | ||||||
| 	return c.cfg.IgnorePkgUpdates |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (c *ALRConfig) AutoPull(ctx context.Context) bool { |  | ||||||
| 	c.cfgOnce.Do(func() { |  | ||||||
| 		c.Load(ctx) |  | ||||||
| 	}) |  | ||||||
| 	return c.cfg.AutoPull |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,69 +0,0 @@ | |||||||
| // This file was originally part of the project "LURE - Linux User REpository", created by Elara Musayelyan. |  | ||||||
| // It has been modified as part of "ALR - Any Linux Repository" by Евгений Храмов. |  | ||||||
| // |  | ||||||
| // ALR - Any Linux Repository |  | ||||||
| // Copyright (C) 2025 Евгений Храмов |  | ||||||
| // |  | ||||||
| // 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 config |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"strings" |  | ||||||
| 	"sync" |  | ||||||
|  |  | ||||||
| 	"github.com/leonelquinteros/gotext" |  | ||||||
| 	"golang.org/x/text/language" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	langMtx sync.Mutex |  | ||||||
| 	lang    language.Tag |  | ||||||
| 	langSet bool |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Language returns the system language. |  | ||||||
| // The first time it's called, it'll detect the langauge based on |  | ||||||
| // the $LANG environment variable. |  | ||||||
| // Subsequent calls will just return the same value. |  | ||||||
| func Language(ctx context.Context) language.Tag { |  | ||||||
| 	langMtx.Lock() |  | ||||||
| 	defer langMtx.Unlock() |  | ||||||
| 	if !langSet { |  | ||||||
| 		syslang := SystemLang() |  | ||||||
| 		tag, err := language.Parse(syslang) |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error parsing system language"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 		} |  | ||||||
| 		base, _ := tag.Base() |  | ||||||
| 		lang = language.Make(base.String()) |  | ||||||
| 		langSet = true |  | ||||||
| 	} |  | ||||||
| 	return lang |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // SystemLang returns the system language based on |  | ||||||
| // the $LANG environment variable. |  | ||||||
| func SystemLang() string { |  | ||||||
| 	lang := os.Getenv("LANG") |  | ||||||
| 	lang, _, _ = strings.Cut(lang, ".") |  | ||||||
| 	if lang == "" || lang == "C" { |  | ||||||
| 		lang = "en" |  | ||||||
| 	} |  | ||||||
| 	return lang |  | ||||||
| } |  | ||||||
| @@ -19,27 +19,11 @@ | |||||||
|  |  | ||||||
| package config | package config | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Paths contains various paths used by ALR | // Paths contains various paths used by ALR | ||||||
| type Paths struct { | type Paths struct { | ||||||
| 	ConfigDir  string | 	UserConfigPath string | ||||||
| 	ConfigPath string |  | ||||||
| 	CacheDir       string | 	CacheDir       string | ||||||
| 	RepoDir        string | 	RepoDir        string | ||||||
| 	PkgsDir        string | 	PkgsDir        string | ||||||
| 	DBPath         string | 	DBPath         string | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetPaths returns a Paths struct. |  | ||||||
| // The first time it's called, it'll generate the struct |  | ||||||
| // using information from the system. |  | ||||||
| // Subsequent calls will return the same value. |  | ||||||
| // |  | ||||||
| // Depreacted: use struct API |  | ||||||
| func GetPaths(ctx context.Context) *Paths { |  | ||||||
| 	alrConfig := GetInstance(ctx) |  | ||||||
| 	return alrConfig.GetPaths(ctx) |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -38,11 +38,12 @@ func armVariant() string { | |||||||
| 		return armEnv | 		return armEnv | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if cpu.ARM.HasVFPv3 { | 	switch { | ||||||
|  | 	case cpu.ARM.HasVFPv3: | ||||||
| 		return "arm7" | 		return "arm7" | ||||||
| 	} else if cpu.ARM.HasVFP { | 	case cpu.ARM.HasVFP: | ||||||
| 		return "arm6" | 		return "arm6" | ||||||
| 	} else { | 	default: | ||||||
| 		return "arm5" | 		return "arm5" | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,10 +31,11 @@ import ( | |||||||
|  |  | ||||||
| // CurrentVersion is the current version of the database. | // CurrentVersion is the current version of the database. | ||||||
| // The database is reset if its version doesn't match this. | // The database is reset if its version doesn't match this. | ||||||
| const CurrentVersion = 2 | const CurrentVersion = 3 | ||||||
|  |  | ||||||
| // Package is a ALR package's database representation | // Package is a ALR package's database representation | ||||||
| type Package struct { | type Package struct { | ||||||
|  | 	BasePkgName   string                    `sh:"base" db:"basepkg_name"` | ||||||
| 	Name          string                    `sh:"name,required" db:"name"` | 	Name          string                    `sh:"name,required" db:"name"` | ||||||
| 	Version       string                    `sh:"version,required" db:"version"` | 	Version       string                    `sh:"version,required" db:"version"` | ||||||
| 	Release       int                       `sh:"release,required" db:"release"` | 	Release       int                       `sh:"release,required" db:"release"` | ||||||
| @@ -58,7 +59,7 @@ type version struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| type Config interface { | type Config interface { | ||||||
| 	GetPaths(ctx context.Context) *config.Paths | 	GetPaths() *config.Paths | ||||||
| } | } | ||||||
|  |  | ||||||
| type Database struct { | type Database struct { | ||||||
| @@ -81,7 +82,7 @@ func (d *Database) Init(ctx context.Context) error { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (d *Database) Connect(ctx context.Context) error { | func (d *Database) Connect(ctx context.Context) error { | ||||||
| 	dsn := d.config.GetPaths(ctx).DBPath | 	dsn := d.config.GetPaths().DBPath | ||||||
| 	db, err := sqlx.Open("sqlite", dsn) | 	db, err := sqlx.Open("sqlite", dsn) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -99,6 +100,7 @@ func (d *Database) initDB(ctx context.Context) error { | |||||||
| 	conn := d.conn | 	conn := d.conn | ||||||
| 	_, err := conn.ExecContext(ctx, ` | 	_, err := conn.ExecContext(ctx, ` | ||||||
| 		CREATE TABLE IF NOT EXISTS pkgs ( | 		CREATE TABLE IF NOT EXISTS pkgs ( | ||||||
|  | 			basepkg_name  TEXT NOT NULL, | ||||||
| 			name          TEXT NOT NULL, | 			name          TEXT NOT NULL, | ||||||
| 			repository    TEXT NOT NULL, | 			repository    TEXT NOT NULL, | ||||||
| 			version       TEXT NOT NULL, | 			version       TEXT NOT NULL, | ||||||
| @@ -129,7 +131,10 @@ func (d *Database) initDB(ctx context.Context) error { | |||||||
| 	ver, ok := d.GetVersion(ctx) | 	ver, ok := d.GetVersion(ctx) | ||||||
| 	if ok && ver != CurrentVersion { | 	if ok && ver != CurrentVersion { | ||||||
| 		slog.Warn(gotext.Get("Database version mismatch; resetting"), "version", ver, "expected", CurrentVersion) | 		slog.Warn(gotext.Get("Database version mismatch; resetting"), "version", ver, "expected", CurrentVersion) | ||||||
| 		d.reset(ctx) | 		err = d.reset(ctx) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
| 		return d.initDB(ctx) | 		return d.initDB(ctx) | ||||||
| 	} else if !ok { | 	} else if !ok { | ||||||
| 		slog.Warn(gotext.Get("Database version does not exist. Run alr fix if something isn't working."), "version", ver, "expected", CurrentVersion) | 		slog.Warn(gotext.Get("Database version does not exist. Run alr fix if something isn't working."), "version", ver, "expected", CurrentVersion) | ||||||
| @@ -193,6 +198,7 @@ func (d *Database) IsEmpty(ctx context.Context) bool { | |||||||
| func (d *Database) InsertPackage(ctx context.Context, pkg Package) error { | func (d *Database) InsertPackage(ctx context.Context, pkg Package) error { | ||||||
| 	_, err := d.conn.NamedExecContext(ctx, ` | 	_, err := d.conn.NamedExecContext(ctx, ` | ||||||
| 		INSERT OR REPLACE INTO pkgs ( | 		INSERT OR REPLACE INTO pkgs ( | ||||||
|  | 			basepkg_name, | ||||||
| 			name, | 			name, | ||||||
| 			repository, | 			repository, | ||||||
| 			version, | 			version, | ||||||
| @@ -210,6 +216,7 @@ func (d *Database) InsertPackage(ctx context.Context, pkg Package) error { | |||||||
| 			builddepends, | 			builddepends, | ||||||
| 			optdepends | 			optdepends | ||||||
| 		) VALUES ( | 		) VALUES ( | ||||||
|  | 		 	:basepkg_name, | ||||||
| 			:name, | 			:name, | ||||||
| 			:repository, | 			:repository, | ||||||
| 			:version, | 			:version, | ||||||
|   | |||||||
| @@ -1,106 +0,0 @@ | |||||||
| // ALR - Any Linux Repository |  | ||||||
| // Copyright (C) 2025 Евгений Храмов |  | ||||||
| // |  | ||||||
| // 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 db |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"sync" |  | ||||||
|  |  | ||||||
| 	"github.com/jmoiron/sqlx" |  | ||||||
| 	"github.com/leonelquinteros/gotext" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // DB returns the ALR database. |  | ||||||
| // The first time it's called, it opens the SQLite database file. |  | ||||||
| // Subsequent calls return the same connection. |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func DB(ctx context.Context) *sqlx.DB { |  | ||||||
| 	return GetInstance(ctx).GetConn() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Close closes the database |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func Close() error { |  | ||||||
| 	if database != nil { |  | ||||||
| 		return database.Close() |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // IsEmpty returns true if the database has no packages in it, otherwise it returns false. |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func IsEmpty(ctx context.Context) bool { |  | ||||||
| 	return GetInstance(ctx).IsEmpty(ctx) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // InsertPackage adds a package to the database |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func InsertPackage(ctx context.Context, pkg Package) error { |  | ||||||
| 	return GetInstance(ctx).InsertPackage(ctx, pkg) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetPkgs returns a result containing packages that match the where conditions |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func GetPkgs(ctx context.Context, where string, args ...any) (*sqlx.Rows, error) { |  | ||||||
| 	return GetInstance(ctx).GetPkgs(ctx, where, args...) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetPkg returns a single package that matches the where conditions |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func GetPkg(ctx context.Context, where string, args ...any) (*Package, error) { |  | ||||||
| 	return GetInstance(ctx).GetPkg(ctx, where, args...) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // DeletePkgs deletes all packages matching the where conditions |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func DeletePkgs(ctx context.Context, where string, args ...any) error { |  | ||||||
| 	return GetInstance(ctx).DeletePkgs(ctx, where, args...) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ======================= |  | ||||||
| // FOR LEGACY ONLY |  | ||||||
| // ======================= |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	dbOnce   sync.Once |  | ||||||
| 	database *Database |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Deprecated: For legacy only |  | ||||||
| func GetInstance(ctx context.Context) *Database { |  | ||||||
| 	dbOnce.Do(func() { |  | ||||||
| 		cfg := config.GetInstance(ctx) |  | ||||||
| 		database = New(cfg) |  | ||||||
| 		err := database.Init(ctx) |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error opening database"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
| 	return database |  | ||||||
| } |  | ||||||
| @@ -33,7 +33,7 @@ import ( | |||||||
|  |  | ||||||
| type TestALRConfig struct{} | type TestALRConfig struct{} | ||||||
|  |  | ||||||
| func (c *TestALRConfig) GetPaths(ctx context.Context) *config.Paths { | func (c *TestALRConfig) GetPaths() *config.Paths { | ||||||
| 	return &config.Paths{ | 	return &config.Paths{ | ||||||
| 		DBPath: ":memory:", | 		DBPath: ":memory:", | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -42,9 +42,6 @@ import ( | |||||||
| 	"golang.org/x/crypto/blake2b" | 	"golang.org/x/crypto/blake2b" | ||||||
| 	"golang.org/x/crypto/blake2s" | 	"golang.org/x/crypto/blake2s" | ||||||
| 	"golang.org/x/exp/slices" | 	"golang.org/x/exp/slices" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/dlcache" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Константа для имени файла манифеста кэша | // Константа для имени файла манифеста кэша | ||||||
| @@ -83,6 +80,11 @@ func (t Type) String() string { | |||||||
| 	return "<unknown>" | 	return "<unknown>" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type DlCache interface { | ||||||
|  | 	Get(context.Context, string) (string, bool) | ||||||
|  | 	New(context.Context, string) (string, error) | ||||||
|  | } | ||||||
|  |  | ||||||
| // Структура Options содержит параметры для загрузки файлов и каталогов | // Структура Options содержит параметры для загрузки файлов и каталогов | ||||||
| type Options struct { | type Options struct { | ||||||
| 	Hash             []byte | 	Hash             []byte | ||||||
| @@ -94,6 +96,7 @@ type Options struct { | |||||||
| 	PostprocDisabled bool | 	PostprocDisabled bool | ||||||
| 	Progress         io.Writer | 	Progress         io.Writer | ||||||
| 	LocalDir         string | 	LocalDir         string | ||||||
|  | 	DlCache          DlCache | ||||||
| } | } | ||||||
|  |  | ||||||
| // Метод для создания нового хеша на основе указанного алгоритма хеширования | // Метод для создания нового хеша на основе указанного алгоритма хеширования | ||||||
| @@ -145,9 +148,6 @@ type UpdatingDownloader interface { | |||||||
|  |  | ||||||
| // Функция Download загружает файл или каталог с использованием указанных параметров | // Функция Download загружает файл или каталог с использованием указанных параметров | ||||||
| func Download(ctx context.Context, opts Options) (err error) { | func Download(ctx context.Context, opts Options) (err error) { | ||||||
| 	cfg := config.GetInstance(ctx) |  | ||||||
| 	dc := dlcache.New(cfg) |  | ||||||
|  |  | ||||||
| 	normalized, err := normalizeURL(opts.URL) | 	normalized, err := normalizeURL(opts.URL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -162,7 +162,7 @@ func Download(ctx context.Context, opts Options) (err error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var t Type | 	var t Type | ||||||
| 	cacheDir, ok := dc.Get(ctx, opts.URL) | 	cacheDir, ok := opts.DlCache.Get(ctx, opts.URL) | ||||||
| 	if ok { | 	if ok { | ||||||
| 		var updated bool | 		var updated bool | ||||||
| 		if d, ok := d.(UpdatingDownloader); ok { | 		if d, ok := d.(UpdatingDownloader); ok { | ||||||
| @@ -221,7 +221,7 @@ func Download(ctx context.Context, opts Options) (err error) { | |||||||
|  |  | ||||||
| 	slog.Info(gotext.Get("Downloading source"), "source", opts.Name, "downloader", d.Name()) | 	slog.Info(gotext.Get("Downloading source"), "source", opts.Name, "downloader", d.Name()) | ||||||
|  |  | ||||||
| 	cacheDir, err = dc.New(ctx, opts.URL) | 	cacheDir, err = opts.DlCache.New(ctx, opts.URL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|   | |||||||
							
								
								
									
										176
									
								
								internal/dl/dl_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								internal/dl/dl_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 dl_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	"net/http" | ||||||
|  | 	"net/http/httptest" | ||||||
|  | 	"net/http/httputil" | ||||||
|  | 	"net/url" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
|  | 	"strings" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/dl" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/dlcache" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type TestALRConfig struct{} | ||||||
|  |  | ||||||
|  | func (c *TestALRConfig) GetPaths() *config.Paths { | ||||||
|  | 	return &config.Paths{ | ||||||
|  | 		CacheDir: "/tmp", | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestDownloadWithoutCache(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		name     string | ||||||
|  | 		path     string | ||||||
|  | 		expected func(*testing.T, error, string) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	prepareServer := func() *httptest.Server { | ||||||
|  | 		// URL вашего Git-сервера | ||||||
|  | 		gitServerURL, err := url.Parse("https://gitea.plemya-x.ru") | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Fatalf("Failed to parse git server URL: %v", err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		proxy := httputil.NewSingleHostReverseProxy(gitServerURL) | ||||||
|  |  | ||||||
|  | 		return httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 			switch { | ||||||
|  | 			case r.URL.Path == "/file-downloader/file": | ||||||
|  | 				w.WriteHeader(http.StatusOK) | ||||||
|  | 				w.Write([]byte("Hello, World!")) | ||||||
|  | 			case strings.HasPrefix(r.URL.Path, "/git-downloader/git"): | ||||||
|  | 				r.URL.Host = gitServerURL.Host | ||||||
|  | 				r.URL.Scheme = gitServerURL.Scheme | ||||||
|  | 				r.Host = gitServerURL.Host | ||||||
|  | 				r.URL.Path, _ = strings.CutPrefix(r.URL.Path, "/git-downloader/git") | ||||||
|  |  | ||||||
|  | 				proxy.ServeHTTP(w, r) | ||||||
|  | 			default: | ||||||
|  | 				w.WriteHeader(http.StatusNotFound) | ||||||
|  | 			} | ||||||
|  | 		})) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{ | ||||||
|  | 		{ | ||||||
|  | 			name: "simple file download", | ||||||
|  | 			path: "%s/file-downloader/file", | ||||||
|  | 			expected: func(t *testing.T, err error, tmpdir string) { | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 				_, err = os.Stat(path.Join(tmpdir, "file")) | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name: "git download", | ||||||
|  | 			path: "git+%s/git-downloader/git/Plemya-x/alr-repo", | ||||||
|  | 			expected: func(t *testing.T, err error, tmpdir string) { | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 				_, err = os.Stat(path.Join(tmpdir, "alr-repo.toml")) | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} { | ||||||
|  | 		t.Run(tc.name, func(t *testing.T) { | ||||||
|  | 			server := prepareServer() | ||||||
|  | 			defer server.Close() | ||||||
|  |  | ||||||
|  | 			tmpdir, err := os.MkdirTemp("", "test-download") | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			defer os.RemoveAll(tmpdir) | ||||||
|  |  | ||||||
|  | 			opts := dl.Options{ | ||||||
|  | 				CacheDisabled: true, | ||||||
|  | 				URL:           fmt.Sprintf(tc.path, server.URL), | ||||||
|  | 				Destination:   tmpdir, | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			err = dl.Download(context.Background(), opts) | ||||||
|  |  | ||||||
|  | 			tc.expected(t, err, tmpdir) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestDownloadFileWithCache(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		name string | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{ | ||||||
|  | 		{ | ||||||
|  | 			name: "simple download", | ||||||
|  | 		}, | ||||||
|  | 	} { | ||||||
|  | 		t.Run(tc.name, func(t *testing.T) { | ||||||
|  | 			called := 0 | ||||||
|  | 			server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 				switch { | ||||||
|  | 				case r.URL.Path == "/file": | ||||||
|  | 					called += 1 | ||||||
|  | 					w.WriteHeader(http.StatusOK) | ||||||
|  | 					w.Write([]byte("Hello, World!")) | ||||||
|  | 				default: | ||||||
|  | 					w.WriteHeader(http.StatusNotFound) | ||||||
|  | 				} | ||||||
|  | 			})) | ||||||
|  | 			defer server.Close() | ||||||
|  |  | ||||||
|  | 			tmpdir, err := os.MkdirTemp("", "test-download") | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			defer os.RemoveAll(tmpdir) | ||||||
|  |  | ||||||
|  | 			cfg := &TestALRConfig{} | ||||||
|  |  | ||||||
|  | 			opts := dl.Options{ | ||||||
|  | 				CacheDisabled: false, | ||||||
|  | 				URL:           server.URL + "/file", | ||||||
|  | 				Destination:   tmpdir, | ||||||
|  | 				DlCache:       dlcache.New(cfg), | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			outputFile := path.Join(tmpdir, "file") | ||||||
|  |  | ||||||
|  | 			err = dl.Download(context.Background(), opts) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			_, err = os.Stat(outputFile) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = os.Remove(outputFile) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = dl.Download(context.Background(), opts) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			assert.Equal(t, 1, called) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -123,6 +123,7 @@ func (FileDownloader) Download(ctx context.Context, opts Options) (Type, string, | |||||||
| 	} else { | 	} else { | ||||||
| 		out = fl | 		out = fl | ||||||
| 	} | 	} | ||||||
|  | 	defer out.Close() | ||||||
|  |  | ||||||
| 	h, err := opts.NewHash() | 	h, err := opts.NewHash() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -143,7 +144,6 @@ func (FileDownloader) Download(ctx context.Context, opts Options) (Type, string, | |||||||
| 		return 0, "", err | 		return 0, "", err | ||||||
| 	} | 	} | ||||||
| 	r.Close() | 	r.Close() | ||||||
| 	out.Close() |  | ||||||
|  |  | ||||||
| 	// Проверка контрольной суммы | 	// Проверка контрольной суммы | ||||||
| 	if opts.Hash != nil { | 	if opts.Hash != nil { | ||||||
|   | |||||||
| @@ -28,7 +28,7 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| type Config interface { | type Config interface { | ||||||
| 	GetPaths(ctx context.Context) *config.Paths | 	GetPaths() *config.Paths | ||||||
| } | } | ||||||
|  |  | ||||||
| type DownloadCache struct { | type DownloadCache struct { | ||||||
| @@ -43,7 +43,7 @@ func New(cfg Config) *DownloadCache { | |||||||
|  |  | ||||||
| func (dc *DownloadCache) BasePath(ctx context.Context) string { | func (dc *DownloadCache) BasePath(ctx context.Context) string { | ||||||
| 	return filepath.Join( | 	return filepath.Join( | ||||||
| 		dc.cfg.GetPaths(ctx).CacheDir, "dl", | 		dc.cfg.GetPaths().CacheDir, "dl", | ||||||
| 	) | 	) | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -32,19 +32,11 @@ import ( | |||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/dlcache" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/dlcache" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func init() { |  | ||||||
| 	dir, err := os.MkdirTemp("/tmp", "alr-dlcache-test.*") |  | ||||||
| 	if err != nil { |  | ||||||
| 		panic(err) |  | ||||||
| 	} |  | ||||||
| 	config.GetPaths(context.Background()).RepoDir = dir |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type TestALRConfig struct { | type TestALRConfig struct { | ||||||
| 	CacheDir string | 	CacheDir string | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *TestALRConfig) GetPaths(ctx context.Context) *config.Paths { | func (c *TestALRConfig) GetPaths() *config.Paths { | ||||||
| 	return &config.Paths{ | 	return &config.Paths{ | ||||||
| 		CacheDir: c.CacheDir, | 		CacheDir: c.CacheDir, | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -62,6 +62,25 @@ func New() *Logger { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func slogLevelToLog(level slog.Level) log.Level { | ||||||
|  | 	switch level { | ||||||
|  | 	case slog.LevelDebug: | ||||||
|  | 		return log.DebugLevel | ||||||
|  | 	case slog.LevelInfo: | ||||||
|  | 		return log.InfoLevel | ||||||
|  | 	case slog.LevelWarn: | ||||||
|  | 		return log.WarnLevel | ||||||
|  | 	case slog.LevelError: | ||||||
|  | 		return log.ErrorLevel | ||||||
|  | 	} | ||||||
|  | 	return log.FatalLevel | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (l *Logger) SetLevel(level slog.Level) { | ||||||
|  | 	l.lOut.(*log.Logger).SetLevel(slogLevelToLog(level)) | ||||||
|  | 	l.lErr.(*log.Logger).SetLevel(slogLevelToLog(level)) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (l *Logger) Enabled(ctx context.Context, level slog.Level) bool { | func (l *Logger) Enabled(ctx context.Context, level slog.Level) bool { | ||||||
| 	if level <= slog.LevelInfo { | 	if level <= slog.LevelInfo { | ||||||
| 		return l.lOut.Enabled(ctx, level) | 		return l.lOut.Enabled(ctx, level) | ||||||
| @@ -90,7 +109,9 @@ func (l *Logger) WithGroup(name string) slog.Handler { | |||||||
| 	return &sl | 	return &sl | ||||||
| } | } | ||||||
|  |  | ||||||
| func SetupDefault() { | func SetupDefault() *Logger { | ||||||
| 	logger := slog.New(New()) | 	l := New() | ||||||
|  | 	logger := slog.New(l) | ||||||
| 	slog.SetDefault(logger) | 	slog.SetDefault(logger) | ||||||
|  | 	return l | ||||||
| } | } | ||||||
|   | |||||||
| @@ -55,12 +55,12 @@ func copyDirOrFile(sourcePath, destPath string) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if sourceInfo.IsDir() { | 	switch { | ||||||
|  | 	case sourceInfo.IsDir(): | ||||||
| 		return copyDir(sourcePath, destPath, sourceInfo) | 		return copyDir(sourcePath, destPath, sourceInfo) | ||||||
| 	} else if sourceInfo.Mode().IsRegular() { | 	case sourceInfo.Mode().IsRegular(): | ||||||
| 		return copyFile(sourcePath, destPath, sourceInfo) | 		return copyFile(sourcePath, destPath, sourceInfo) | ||||||
| 	} else { | 	default: | ||||||
| 		// ignore non-regular files |  | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ func init() { | |||||||
|  |  | ||||||
| 	b2 := lipgloss.RoundedBorder() | 	b2 := lipgloss.RoundedBorder() | ||||||
| 	b2.Left = "\u2524" | 	b2.Left = "\u2524" | ||||||
| 	infoStyle = titleStyle.Copy().BorderStyle(b2) | 	infoStyle = titleStyle.BorderStyle(b2) | ||||||
| } | } | ||||||
|  |  | ||||||
| type Pager struct { | type Pager struct { | ||||||
|   | |||||||
| @@ -164,26 +164,15 @@ func (d *Decoder) DecodeVars(val any) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| type ScriptFunc func(ctx context.Context, opts ...interp.RunnerOption) error | type ( | ||||||
|  | 	ScriptFunc             func(ctx context.Context, opts ...interp.RunnerOption) error | ||||||
|  | 	ScriptFuncWithSubshell func(ctx context.Context, opts ...interp.RunnerOption) (*interp.Runner, error) | ||||||
|  | ) | ||||||
|  |  | ||||||
| // GetFunc returns a function corresponding to a bash function | // GetFunc returns a function corresponding to a bash function | ||||||
| // with the given name | // with the given name | ||||||
| func (d *Decoder) GetFunc(name string) (ScriptFunc, bool) { | func (d *Decoder) GetFunc(name string) (ScriptFunc, bool) { | ||||||
| 	fn := d.getFunc(name) | 	return d.GetFuncP(name, nil) | ||||||
| 	if fn == nil { |  | ||||||
| 		return nil, false |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return func(ctx context.Context, opts ...interp.RunnerOption) error { |  | ||||||
| 		sub := d.Runner.Subshell() |  | ||||||
| 		for _, opt := range opts { |  | ||||||
| 			err := opt(sub) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		return sub.Run(ctx, fn) |  | ||||||
| 	}, true |  | ||||||
| } | } | ||||||
|  |  | ||||||
| type PrepareFunc func(context.Context, *interp.Runner) error | type PrepareFunc func(context.Context, *interp.Runner) error | ||||||
| @@ -211,6 +200,24 @@ func (d *Decoder) GetFuncP(name string, prepare PrepareFunc) (ScriptFunc, bool) | |||||||
| 	}, true | 	}, true | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (d *Decoder) GetFuncWithSubshell(name string) (ScriptFuncWithSubshell, bool) { | ||||||
|  | 	fn := d.getFunc(name) | ||||||
|  | 	if fn == nil { | ||||||
|  | 		return nil, false | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return func(ctx context.Context, opts ...interp.RunnerOption) (*interp.Runner, error) { | ||||||
|  | 		sub := d.Runner.Subshell() | ||||||
|  | 		for _, opt := range opts { | ||||||
|  | 			err := opt(sub) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return sub, sub.Run(ctx, fn) | ||||||
|  | 	}, true | ||||||
|  | } | ||||||
|  |  | ||||||
| func (d *Decoder) getFunc(name string) *syntax.Stmt { | func (d *Decoder) getFunc(name string) *syntax.Stmt { | ||||||
| 	names, err := overrides.Resolve(d.info, overrides.DefaultOpts.WithName(name)) | 	names, err := overrides.Resolve(d.info, overrides.DefaultOpts.WithName(name)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|   | |||||||
| @@ -29,9 +29,9 @@ import ( | |||||||
| 	"syscall" | 	"syscall" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/fakeroot" | ||||||
| 	"mvdan.cc/sh/v3/expand" | 	"mvdan.cc/sh/v3/expand" | ||||||
| 	"mvdan.cc/sh/v3/interp" | 	"mvdan.cc/sh/v3/interp" | ||||||
| 	"plemya-x.ru/fakeroot" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // FakerootExecHandler was extracted from github.com/mvdan/sh/interp/handler.go | // FakerootExecHandler was extracted from github.com/mvdan/sh/interp/handler.go | ||||||
|   | |||||||
| @@ -22,10 +22,11 @@ package handlers | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"io" | 	"io" | ||||||
|  | 	"io/fs" | ||||||
| 	"os" | 	"os" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func NopReadDir(context.Context, string) ([]os.FileInfo, error) { | func NopReadDir(context.Context, string) ([]fs.DirEntry, error) { | ||||||
| 	return nil, os.ErrNotExist | 	return nil, os.ErrNotExist | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,12 +31,12 @@ import ( | |||||||
| 	"mvdan.cc/sh/v3/interp" | 	"mvdan.cc/sh/v3/interp" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func RestrictedReadDir(allowedPrefixes ...string) interp.ReadDirHandlerFunc { | func RestrictedReadDir(allowedPrefixes ...string) interp.ReadDirHandlerFunc2 { | ||||||
| 	return func(ctx context.Context, s string) ([]fs.FileInfo, error) { | 	return func(ctx context.Context, s string) ([]fs.DirEntry, error) { | ||||||
| 		path := filepath.Clean(s) | 		path := filepath.Clean(s) | ||||||
| 		for _, allowedPrefix := range allowedPrefixes { | 		for _, allowedPrefix := range allowedPrefixes { | ||||||
| 			if strings.HasPrefix(path, allowedPrefix) { | 			if strings.HasPrefix(path, allowedPrefix) { | ||||||
| 				return interp.DefaultReadDirHandler()(ctx, s) | 				return interp.DefaultReadDirHandler2()(ctx, s) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -245,10 +245,13 @@ func gitVersionCmd(hc interp.HandlerContext, cmd string, args []string) error { | |||||||
| 		return fmt.Errorf("git-version: %w", err) | 		return fmt.Errorf("git-version: %w", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	commits.ForEach(func(*object.Commit) error { | 	err = commits.ForEach(func(*object.Commit) error { | ||||||
| 		revNum++ | 		revNum++ | ||||||
| 		return nil | 		return nil | ||||||
| 	}) | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("git-version: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	HEAD, err := r.Head() | 	HEAD, err := r.Head() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|   | |||||||
| @@ -9,40 +9,60 @@ msgstr "" | |||||||
| "Content-Transfer-Encoding: 8bit\n" | "Content-Transfer-Encoding: 8bit\n" | ||||||
| "Plural-Forms: nplurals=2; plural=(n != 1);\n" | "Plural-Forms: nplurals=2; plural=(n != 1);\n" | ||||||
|  |  | ||||||
| #: build.go:41 | #: build.go:44 | ||||||
| msgid "Build a local package" | msgid "Build a local package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:47 | #: build.go:50 | ||||||
| msgid "Path to the build script" | msgid "Path to the build script" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:52 | #: build.go:55 | ||||||
|  | msgid "Specify subpackage in script (for multi package script only)" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: build.go:60 | ||||||
| msgid "Name of the package to build and its repo (example: default/go-bin)" | msgid "Name of the package to build and its repo (example: default/go-bin)" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:57 | #: build.go:65 | ||||||
| msgid "" | msgid "" | ||||||
| "Build package from scratch even if there's an already built package available" | "Build package from scratch even if there's an already built package available" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:80 | #: build.go:73 | ||||||
|  | msgid "Error loading config" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: build.go:81 | ||||||
|  | msgid "Error initialization database" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: build.go:110 | ||||||
|  | msgid "Package not found" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: build.go:130 | ||||||
| msgid "Error pulling repositories" | msgid "Error pulling repositories" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:87 | #: build.go:138 | ||||||
| msgid "Unable to detect a supported package manager on the system" | msgid "Unable to detect a supported package manager on the system" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:98 | #: build.go:144 | ||||||
|  | msgid "Error parsing os release" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: build.go:166 | ||||||
| msgid "Error building package" | msgid "Error building package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:104 | #: build.go:173 | ||||||
| msgid "Error getting working directory" | msgid "Error getting working directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: build.go:112 | #: build.go:182 | ||||||
| msgid "Error moving the package" | msgid "Error moving the package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -50,27 +70,27 @@ msgstr "" | |||||||
| msgid "Attempt to fix problems with ALR" | msgid "Attempt to fix problems with ALR" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:44 | #: fix.go:49 | ||||||
| msgid "Removing cache directory" | msgid "Removing cache directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:48 | #: fix.go:53 | ||||||
| msgid "Unable to remove cache directory" | msgid "Unable to remove cache directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:52 | #: fix.go:57 | ||||||
| msgid "Rebuilding cache" | msgid "Rebuilding cache" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:56 | #: fix.go:61 | ||||||
| msgid "Unable to create new cache directory" | msgid "Unable to create new cache directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:62 | #: fix.go:81 | ||||||
| msgid "Error pulling repos" | msgid "Error pulling repos" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: fix.go:66 | #: fix.go:85 | ||||||
| msgid "Done" | msgid "Done" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -98,63 +118,59 @@ msgstr "" | |||||||
| msgid "No such helper command" | msgid "No such helper command" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: info.go:42 | #: info.go:43 | ||||||
| msgid "Print information about a package" | msgid "Print information about a package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: info.go:47 | #: info.go:48 | ||||||
| msgid "Show all information, not just for the current distro" | msgid "Show all information, not just for the current distro" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: info.go:57 | #: info.go:69 | ||||||
| msgid "Error initialization database" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: info.go:64 |  | ||||||
| msgid "Command info expected at least 1 argument, got %d" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: info.go:78 |  | ||||||
| msgid "Error finding packages" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: info.go:94 |  | ||||||
| msgid "Error parsing os-release file" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: info.go:103 |  | ||||||
| msgid "Error resolving overrides" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: info.go:112 info.go:118 |  | ||||||
| msgid "Error encoding script variables" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: install.go:42 |  | ||||||
| msgid "Install a new package" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: install.go:56 |  | ||||||
| msgid "Command install expected at least 1 argument, got %d" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: install.go:91 |  | ||||||
| msgid "Error getting packages" | msgid "Error getting packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: install.go:100 | #: info.go:78 | ||||||
| msgid "Error iterating over packages" | msgid "Error iterating over packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: install.go:113 | #: info.go:105 | ||||||
|  | msgid "Command info expected at least 1 argument, got %d" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: info.go:119 | ||||||
|  | msgid "Error finding packages" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: info.go:144 | ||||||
|  | msgid "Error parsing os-release file" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: info.go:153 | ||||||
|  | msgid "Error resolving overrides" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: info.go:162 info.go:168 | ||||||
|  | msgid "Error encoding script variables" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: install.go:43 | ||||||
|  | msgid "Install a new package" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: install.go:57 | ||||||
|  | msgid "Command install expected at least 1 argument, got %d" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: install.go:157 | ||||||
| msgid "Remove an installed package" | msgid "Remove an installed package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: install.go:118 | #: install.go:162 | ||||||
| msgid "Command remove expected at least 1 argument, got %d" | msgid "Command remove expected at least 1 argument, got %d" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: install.go:130 | #: install.go:177 | ||||||
| msgid "Error removing packages" | msgid "Error removing packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -182,59 +198,83 @@ msgstr "" | |||||||
| msgid "Choose which optional package(s) to install" | msgid "Choose which optional package(s) to install" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:64 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:93 | ||||||
| msgid "Error opening config file, using defaults" | msgid "NAME" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:77 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:94 | ||||||
| msgid "Error decoding config file, using defaults" | msgid "USAGE" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:89 | #: internal/cliutils/template.go:74 | ||||||
| msgid "Unable to detect user config directory" | msgid "global options" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:97 | #: internal/cliutils/template.go:74 | ||||||
| msgid "Unable to create ALR config directory" | msgid "command" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:106 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:95 | ||||||
| msgid "Unable to create ALR config file" | msgid "command options" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:112 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:96 | ||||||
| msgid "Error encoding default configuration" | msgid "arguments" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:121 | #: internal/cliutils/template.go:74 | ||||||
| msgid "Unable to detect cache directory" | msgid "VERSION" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:131 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:98 | ||||||
|  | msgid "DESCRIPTION" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "AUTHOR" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "COMMANDS" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "GLOBAL OPTIONS" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "COPYRIGHT" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:97 | ||||||
|  | msgid "CATEGORY" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:99 internal/cliutils/template.go:100 | ||||||
|  | msgid "OPTIONS" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/config/config.go:176 | ||||||
|  | msgid "Unable to create config directory" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: internal/config/config.go:182 | ||||||
| msgid "Unable to create repo cache directory" | msgid "Unable to create repo cache directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/config.go:137 | #: internal/config/config.go:188 | ||||||
| msgid "Unable to create package cache directory" | msgid "Unable to create package cache directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/config/lang.go:50 | #: internal/db/db.go:133 | ||||||
| msgid "Error parsing system language" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: internal/db/db.go:131 |  | ||||||
| msgid "Database version mismatch; resetting" | msgid "Database version mismatch; resetting" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/db/db.go:135 | #: internal/db/db.go:140 | ||||||
| msgid "" | msgid "" | ||||||
| "Database version does not exist. Run alr fix if something isn't working." | "Database version does not exist. Run alr fix if something isn't working." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: internal/db/db_legacy.go:101 |  | ||||||
| msgid "Error opening database" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: internal/dl/dl.go:170 | #: internal/dl/dl.go:170 | ||||||
| msgid "Source can be updated, updating if required" | msgid "Source can be updated, updating if required" | ||||||
| msgstr "" | msgstr "" | ||||||
| @@ -263,11 +303,11 @@ msgstr "" | |||||||
| msgid "ERROR" | msgid "ERROR" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: list.go:40 | #: list.go:41 | ||||||
| msgid "List ALR repo packages" | msgid "List ALR repo packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: list.go:91 | #: list.go:98 | ||||||
| msgid "Error listing installed packages" | msgid "Error listing installed packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| @@ -283,92 +323,92 @@ msgstr "" | |||||||
| msgid "Enable interactive questions and prompts" | msgid "Enable interactive questions and prompts" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: main.go:90 | #: main.go:96 | ||||||
| msgid "" | msgid "" | ||||||
| "Running ALR as root is forbidden as it may cause catastrophic damage to your " | "Running ALR as root is forbidden as it may cause catastrophic damage to your " | ||||||
| "system" | "system" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: main.go:124 | #: main.go:154 | ||||||
|  | msgid "Show help" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: main.go:158 | ||||||
| msgid "Error while running app" | msgid "Error while running app" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:107 | #: pkg/build/build.go:156 | ||||||
| msgid "Failed to prompt user to view build script" | msgid "Failed to prompt user to view build script" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:111 | #: pkg/build/build.go:160 | ||||||
| msgid "Building package" | msgid "Building package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:155 | #: pkg/build/build.go:208 | ||||||
|  | msgid "The checksums array must be the same length as sources" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:235 | ||||||
| msgid "Downloading sources" | msgid "Downloading sources" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:167 | #: pkg/build/build.go:257 | ||||||
| msgid "Building package metadata" | msgid "Building package metadata" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:189 | #: pkg/build/build.go:279 | ||||||
| msgid "Compressing package" | msgid "Compressing package" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:315 | #: pkg/build/build.go:438 | ||||||
| msgid "" | msgid "" | ||||||
| "Your system's CPU architecture doesn't match this package. Do you want to " | "Your system's CPU architecture doesn't match this package. Do you want to " | ||||||
| "build anyway?" | "build anyway?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:326 | #: pkg/build/build.go:452 | ||||||
| msgid "This package is already installed" | msgid "This package is already installed" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:354 | #: pkg/build/build.go:476 | ||||||
| msgid "Installing build dependencies" | msgid "Installing build dependencies" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:396 | #: pkg/build/build.go:521 | ||||||
| msgid "Installing dependencies" | msgid "Installing dependencies" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:442 | #: pkg/build/build.go:602 | ||||||
| msgid "Executing version()" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:462 |  | ||||||
| msgid "Updating version" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:467 |  | ||||||
| msgid "Executing prepare()" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:477 |  | ||||||
| msgid "Executing build()" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:489 |  | ||||||
| msgid "Executing package()" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:527 |  | ||||||
| msgid "Executing files()" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:605 |  | ||||||
| msgid "AutoProv is not implemented for this package format, so it's skipped" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:616 |  | ||||||
| msgid "AutoReq is not implemented for this package format, so it's skipped" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:723 |  | ||||||
| msgid "Would you like to remove the build dependencies?" | msgid "Would you like to remove the build dependencies?" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:829 | #: pkg/build/build.go:665 | ||||||
| msgid "The checksums array must be the same length as sources" | msgid "Executing prepare()" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:675 | ||||||
|  | msgid "Executing build()" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:705 pkg/build/build.go:725 | ||||||
|  | msgid "Executing %s()" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:784 | ||||||
|  | msgid "Error installing native packages" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:808 | ||||||
|  | msgid "Error installing package" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:867 | ||||||
|  | msgid "AutoProv is not implemented for this package format, so it's skipped" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:878 | ||||||
|  | msgid "AutoReq is not implemented for this package format, so it's skipped" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/findDeps.go:35 | #: pkg/build/findDeps.go:35 | ||||||
| @@ -383,84 +423,104 @@ msgstr "" | |||||||
| msgid "Required dependency found" | msgid "Required dependency found" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/build/install.go:42 | #: pkg/repos/pull.go:79 | ||||||
| msgid "Error installing native packages" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/build/install.go:79 |  | ||||||
| msgid "Error installing package" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:75 |  | ||||||
| msgid "Pulling repository" | msgid "Pulling repository" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:99 | #: pkg/repos/pull.go:103 | ||||||
| msgid "Repository up to date" | msgid "Repository up to date" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:156 | #: pkg/repos/pull.go:160 | ||||||
| msgid "Git repository does not appear to be a valid ALR repo" | msgid "Git repository does not appear to be a valid ALR repo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:172 | #: pkg/repos/pull.go:176 | ||||||
| msgid "" | msgid "" | ||||||
| "ALR repo's minimum ALR version is greater than the current version. Try " | "ALR repo's minimum ALR version is greater than the current version. Try " | ||||||
| "updating ALR if something doesn't work." | "updating ALR if something doesn't work." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:41 | #: repo.go:40 | ||||||
| msgid "Add a new repository" | msgid "Add a new repository" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:48 | #: repo.go:47 | ||||||
| msgid "Name of the new repo" | msgid "Name of the new repo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:54 | #: repo.go:53 | ||||||
| msgid "URL of the new repo" | msgid "URL of the new repo" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:79 repo.go:136 | #: repo.go:86 repo.go:156 | ||||||
| msgid "Error opening config file" | msgid "Error saving config" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:85 repo.go:142 | #: repo.go:111 | ||||||
| msgid "Error encoding config" |  | ||||||
| msgstr "" |  | ||||||
|  |  | ||||||
| #: repo.go:103 |  | ||||||
| msgid "Remove an existing repository" | msgid "Remove an existing repository" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:110 | #: repo.go:118 | ||||||
| msgid "Name of the repo to be deleted" | msgid "Name of the repo to be deleted" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:128 | #: repo.go:142 | ||||||
| msgid "Repo does not exist" | msgid "Repo does not exist" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:148 | #: repo.go:150 | ||||||
| msgid "Error removing repo directory" | msgid "Error removing repo directory" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:154 | #: repo.go:167 | ||||||
| msgid "Error removing packages from database" | msgid "Error removing packages from database" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: repo.go:166 | #: repo.go:179 | ||||||
| msgid "Pull all repositories that have changed" | msgid "Pull all repositories that have changed" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:36 | ||||||
|  | msgid "Search packages" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:42 | ||||||
|  | msgid "Search by name" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:47 | ||||||
|  | msgid "Search by description" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:52 | ||||||
|  | msgid "Search by repository" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:57 | ||||||
|  | msgid "Search by provides" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:62 | ||||||
|  | msgid "Format output using a Go template" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:88 search.go:105 | ||||||
|  | msgid "Error parsing format template" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | #: search.go:113 | ||||||
|  | msgid "Error executing template" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
| #: upgrade.go:47 | #: upgrade.go:47 | ||||||
| msgid "Upgrade all installed packages" | msgid "Upgrade all installed packages" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: upgrade.go:83 | #: upgrade.go:96 | ||||||
| msgid "Error checking for updates" | msgid "Error checking for updates" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: upgrade.go:94 | #: upgrade.go:118 | ||||||
| msgid "There is nothing to do." | msgid "There is nothing to do." | ||||||
| msgstr "" | msgstr "" | ||||||
|   | |||||||
| @@ -1,12 +1,12 @@ | |||||||
| # | # | ||||||
| # Maxim Slipenko <maks1ms@alt-gnome.ru>, 2025. |  | ||||||
| # x1z53 <x1z53@yandex.ru>, 2025. | # x1z53 <x1z53@yandex.ru>, 2025. | ||||||
|  | # Maxim Slipenko <maks1ms@alt-gnome.ru>, 2025. | ||||||
| # | # | ||||||
| msgid "" | msgid "" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Project-Id-Version: unnamed project\n" | "Project-Id-Version: unnamed project\n" | ||||||
| "PO-Revision-Date: 2025-01-24 21:20+0300\n" | "PO-Revision-Date: 2025-03-09 17:31+0300\n" | ||||||
| "Last-Translator: x1z53 <x1z53@yandex.ru>\n" | "Last-Translator: Maxim Slipenko <maks1ms@alt-gnome.ru>\n" | ||||||
| "Language-Team: Russian\n" | "Language-Team: Russian\n" | ||||||
| "Language: ru\n" | "Language: ru\n" | ||||||
| "MIME-Version: 1.0\n" | "MIME-Version: 1.0\n" | ||||||
| @@ -16,40 +16,61 @@ msgstr "" | |||||||
| "%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);\n" | "%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);\n" | ||||||
| "X-Generator: Gtranslator 47.1\n" | "X-Generator: Gtranslator 47.1\n" | ||||||
|  |  | ||||||
| #: build.go:41 | #: build.go:44 | ||||||
| msgid "Build a local package" | msgid "Build a local package" | ||||||
| msgstr "Сборка локального пакета" | msgstr "Сборка локального пакета" | ||||||
|  |  | ||||||
| #: build.go:47 | #: build.go:50 | ||||||
| msgid "Path to the build script" | msgid "Path to the build script" | ||||||
| msgstr "Путь к скрипту сборки" | msgstr "Путь к скрипту сборки" | ||||||
|  |  | ||||||
| #: build.go:52 | #: build.go:55 | ||||||
|  | msgid "Specify subpackage in script (for multi package script only)" | ||||||
|  | msgstr "Укажите подпакет в скрипте (только для многопакетного скрипта)" | ||||||
|  |  | ||||||
|  | #: build.go:60 | ||||||
| msgid "Name of the package to build and its repo (example: default/go-bin)" | msgid "Name of the package to build and its repo (example: default/go-bin)" | ||||||
| msgstr "Имя пакета для сборки и его репозиторий (пример: default/go-bin)" | msgstr "Имя пакета для сборки и его репозиторий (пример: default/go-bin)" | ||||||
|  |  | ||||||
| #: build.go:57 | #: build.go:65 | ||||||
| msgid "" | msgid "" | ||||||
| "Build package from scratch even if there's an already built package available" | "Build package from scratch even if there's an already built package available" | ||||||
| msgstr "Создайте пакет с нуля, даже если уже имеется готовый пакет" | msgstr "Создайте пакет с нуля, даже если уже имеется готовый пакет" | ||||||
|  |  | ||||||
| #: build.go:80 | #: build.go:73 | ||||||
|  | #, fuzzy | ||||||
|  | msgid "Error loading config" | ||||||
|  | msgstr "Ошибка при кодировании конфигурации" | ||||||
|  |  | ||||||
|  | #: build.go:81 | ||||||
|  | msgid "Error initialization database" | ||||||
|  | msgstr "Ошибка инициализации базы данных" | ||||||
|  |  | ||||||
|  | #: build.go:110 | ||||||
|  | msgid "Package not found" | ||||||
|  | msgstr "Пакет не найден" | ||||||
|  |  | ||||||
|  | #: build.go:130 | ||||||
| msgid "Error pulling repositories" | msgid "Error pulling repositories" | ||||||
| msgstr "Ошибка при извлечении репозиториев" | msgstr "Ошибка при извлечении репозиториев" | ||||||
|  |  | ||||||
| #: build.go:87 | #: build.go:138 | ||||||
| msgid "Unable to detect a supported package manager on the system" | msgid "Unable to detect a supported package manager on the system" | ||||||
| msgstr "Не удалось обнаружить поддерживаемый менеджер пакетов в системе" | msgstr "Не удалось обнаружить поддерживаемый менеджер пакетов в системе" | ||||||
|  |  | ||||||
| #: build.go:98 | #: build.go:144 | ||||||
|  | msgid "Error parsing os release" | ||||||
|  | msgstr "Ошибка при разборе файла выпуска операционной системы" | ||||||
|  |  | ||||||
|  | #: build.go:166 | ||||||
| msgid "Error building package" | msgid "Error building package" | ||||||
| msgstr "Ошибка при сборке пакета" | msgstr "Ошибка при сборке пакета" | ||||||
|  |  | ||||||
| #: build.go:104 | #: build.go:173 | ||||||
| msgid "Error getting working directory" | msgid "Error getting working directory" | ||||||
| msgstr "Ошибка при получении рабочего каталога" | msgstr "Ошибка при получении рабочего каталога" | ||||||
|  |  | ||||||
| #: build.go:112 | #: build.go:182 | ||||||
| msgid "Error moving the package" | msgid "Error moving the package" | ||||||
| msgstr "Ошибка при перемещении пакета" | msgstr "Ошибка при перемещении пакета" | ||||||
|  |  | ||||||
| @@ -57,27 +78,27 @@ msgstr "Ошибка при перемещении пакета" | |||||||
| msgid "Attempt to fix problems with ALR" | msgid "Attempt to fix problems with ALR" | ||||||
| msgstr "Попытка устранить проблемы с ALR" | msgstr "Попытка устранить проблемы с ALR" | ||||||
|  |  | ||||||
| #: fix.go:44 | #: fix.go:49 | ||||||
| msgid "Removing cache directory" | msgid "Removing cache directory" | ||||||
| msgstr "Удаление каталога кэша" | msgstr "Удаление каталога кэша" | ||||||
|  |  | ||||||
| #: fix.go:48 | #: fix.go:53 | ||||||
| msgid "Unable to remove cache directory" | msgid "Unable to remove cache directory" | ||||||
| msgstr "Не удалось удалить каталог кэша" | msgstr "Не удалось удалить каталог кэша" | ||||||
|  |  | ||||||
| #: fix.go:52 | #: fix.go:57 | ||||||
| msgid "Rebuilding cache" | msgid "Rebuilding cache" | ||||||
| msgstr "Восстановление кэша" | msgstr "Восстановление кэша" | ||||||
|  |  | ||||||
| #: fix.go:56 | #: fix.go:61 | ||||||
| msgid "Unable to create new cache directory" | msgid "Unable to create new cache directory" | ||||||
| msgstr "Не удалось создать новый каталог кэша" | msgstr "Не удалось создать новый каталог кэша" | ||||||
|  |  | ||||||
| #: fix.go:62 | #: fix.go:81 | ||||||
| msgid "Error pulling repos" | msgid "Error pulling repos" | ||||||
| msgstr "Ошибка при извлечении репозиториев" | msgstr "Ошибка при извлечении репозиториев" | ||||||
|  |  | ||||||
| #: fix.go:66 | #: fix.go:85 | ||||||
| msgid "Done" | msgid "Done" | ||||||
| msgstr "Сделано" | msgstr "Сделано" | ||||||
|  |  | ||||||
| @@ -105,63 +126,59 @@ msgstr "Каталог, в который будут устанавливать | |||||||
| msgid "No such helper command" | msgid "No such helper command" | ||||||
| msgstr "Такой вспомогательной команды нет" | msgstr "Такой вспомогательной команды нет" | ||||||
|  |  | ||||||
| #: info.go:42 | #: info.go:43 | ||||||
| msgid "Print information about a package" | msgid "Print information about a package" | ||||||
| msgstr "Отобразить информацию о пакете" | msgstr "Отобразить информацию о пакете" | ||||||
|  |  | ||||||
| #: info.go:47 | #: info.go:48 | ||||||
| msgid "Show all information, not just for the current distro" | msgid "Show all information, not just for the current distro" | ||||||
| msgstr "Показывать всю информацию, не только для текущего дистрибутива" | msgstr "Показывать всю информацию, не только для текущего дистрибутива" | ||||||
|  |  | ||||||
| #: info.go:57 | #: info.go:69 | ||||||
| msgid "Error initialization database" |  | ||||||
| msgstr "Ошибка инициализации базы данных" |  | ||||||
|  |  | ||||||
| #: info.go:64 |  | ||||||
| msgid "Command info expected at least 1 argument, got %d" |  | ||||||
| msgstr "Для команды info ожидался хотя бы 1 аргумент, получено %d" |  | ||||||
|  |  | ||||||
| #: info.go:78 |  | ||||||
| msgid "Error finding packages" |  | ||||||
| msgstr "Ошибка при поиске пакетов" |  | ||||||
|  |  | ||||||
| #: info.go:94 |  | ||||||
| msgid "Error parsing os-release file" |  | ||||||
| msgstr "Ошибка при разборе файла выпуска операционной системы" |  | ||||||
|  |  | ||||||
| #: info.go:103 |  | ||||||
| msgid "Error resolving overrides" |  | ||||||
| msgstr "Ошибка устранения переорпеделений" |  | ||||||
|  |  | ||||||
| #: info.go:112 info.go:118 |  | ||||||
| msgid "Error encoding script variables" |  | ||||||
| msgstr "Ошибка кодирования переменных скрита" |  | ||||||
|  |  | ||||||
| #: install.go:42 |  | ||||||
| msgid "Install a new package" |  | ||||||
| msgstr "Установить новый пакет" |  | ||||||
|  |  | ||||||
| #: install.go:56 |  | ||||||
| msgid "Command install expected at least 1 argument, got %d" |  | ||||||
| msgstr "Для команды install ожидался хотя бы 1 аргумент, получено %d" |  | ||||||
|  |  | ||||||
| #: install.go:91 |  | ||||||
| msgid "Error getting packages" | msgid "Error getting packages" | ||||||
| msgstr "Ошибка при получении пакетов" | msgstr "Ошибка при получении пакетов" | ||||||
|  |  | ||||||
| #: install.go:100 | #: info.go:78 | ||||||
| msgid "Error iterating over packages" | msgid "Error iterating over packages" | ||||||
| msgstr "Ошибка при переборе пакетов" | msgstr "Ошибка при переборе пакетов" | ||||||
|  |  | ||||||
| #: install.go:113 | #: info.go:105 | ||||||
|  | msgid "Command info expected at least 1 argument, got %d" | ||||||
|  | msgstr "Для команды info ожидался хотя бы 1 аргумент, получено %d" | ||||||
|  |  | ||||||
|  | #: info.go:119 | ||||||
|  | msgid "Error finding packages" | ||||||
|  | msgstr "Ошибка при поиске пакетов" | ||||||
|  |  | ||||||
|  | #: info.go:144 | ||||||
|  | msgid "Error parsing os-release file" | ||||||
|  | msgstr "Ошибка при разборе файла выпуска операционной системы" | ||||||
|  |  | ||||||
|  | #: info.go:153 | ||||||
|  | msgid "Error resolving overrides" | ||||||
|  | msgstr "Ошибка устранения переорпеделений" | ||||||
|  |  | ||||||
|  | #: info.go:162 info.go:168 | ||||||
|  | msgid "Error encoding script variables" | ||||||
|  | msgstr "Ошибка кодирования переменных скрита" | ||||||
|  |  | ||||||
|  | #: install.go:43 | ||||||
|  | msgid "Install a new package" | ||||||
|  | msgstr "Установить новый пакет" | ||||||
|  |  | ||||||
|  | #: install.go:57 | ||||||
|  | msgid "Command install expected at least 1 argument, got %d" | ||||||
|  | msgstr "Для команды install ожидался хотя бы 1 аргумент, получено %d" | ||||||
|  |  | ||||||
|  | #: install.go:157 | ||||||
| msgid "Remove an installed package" | msgid "Remove an installed package" | ||||||
| msgstr "Удалить установленный пакет" | msgstr "Удалить установленный пакет" | ||||||
|  |  | ||||||
| #: install.go:118 | #: install.go:162 | ||||||
| msgid "Command remove expected at least 1 argument, got %d" | msgid "Command remove expected at least 1 argument, got %d" | ||||||
| msgstr "Для команды remove ожидался хотя бы 1 аргумент, получено %d" | msgstr "Для команды remove ожидался хотя бы 1 аргумент, получено %d" | ||||||
|  |  | ||||||
| #: install.go:130 | #: install.go:177 | ||||||
| msgid "Error removing packages" | msgid "Error removing packages" | ||||||
| msgstr "Ошибка при удалении пакетов" | msgstr "Ошибка при удалении пакетов" | ||||||
|  |  | ||||||
| @@ -189,64 +206,85 @@ msgstr "Выберите, какой пакет использовать для | |||||||
| msgid "Choose which optional package(s) to install" | msgid "Choose which optional package(s) to install" | ||||||
| msgstr "Выберите, какой дополнительный пакет(ы) следует установить" | msgstr "Выберите, какой дополнительный пакет(ы) следует установить" | ||||||
|  |  | ||||||
| #: internal/config/config.go:64 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:93 | ||||||
| msgid "Error opening config file, using defaults" | msgid "NAME" | ||||||
| msgstr "" | msgstr "НАЗВАНИЕ" | ||||||
| "Ошибка при открытии конфигурационного файла, используются значения по " |  | ||||||
| "умолчанию" |  | ||||||
|  |  | ||||||
| #: internal/config/config.go:77 | #: internal/cliutils/template.go:74 internal/cliutils/template.go:94 | ||||||
| msgid "Error decoding config file, using defaults" | msgid "USAGE" | ||||||
| msgstr "" | msgstr "ИСПОЛЬЗОВАНИЕ" | ||||||
| "Ошибка при декодировании конфигурационного файла, используются значения по " |  | ||||||
| "умолчанию" |  | ||||||
|  |  | ||||||
| #: internal/config/config.go:89 | #: internal/cliutils/template.go:74 | ||||||
| msgid "Unable to detect user config directory" | msgid "global options" | ||||||
| msgstr "Не удалось обнаружить каталог конфигурации пользователя" | msgstr "глобальные опции" | ||||||
|  |  | ||||||
| #: internal/config/config.go:97 | #: internal/cliutils/template.go:74 | ||||||
| msgid "Unable to create ALR config directory" | msgid "command" | ||||||
|  | msgstr "команда" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 internal/cliutils/template.go:95 | ||||||
|  | msgid "command options" | ||||||
|  | msgstr "опции команды" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 internal/cliutils/template.go:96 | ||||||
|  | msgid "arguments" | ||||||
|  | msgstr "аргументы" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "VERSION" | ||||||
|  | msgstr "ВЕРСИЯ" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 internal/cliutils/template.go:98 | ||||||
|  | msgid "DESCRIPTION" | ||||||
|  | msgstr "ОПИСАНИЕ" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "AUTHOR" | ||||||
|  | msgstr "АВТОР" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "COMMANDS" | ||||||
|  | msgstr "КОМАНДЫ" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "GLOBAL OPTIONS" | ||||||
|  | msgstr "ГЛОБАЛЬНЫЕ ОПЦИИ" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:74 | ||||||
|  | msgid "COPYRIGHT" | ||||||
|  | msgstr "АВТОРСКОЕ ПРАВО" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:97 | ||||||
|  | msgid "CATEGORY" | ||||||
|  | msgstr "КАТЕГОРИЯ" | ||||||
|  |  | ||||||
|  | #: internal/cliutils/template.go:99 internal/cliutils/template.go:100 | ||||||
|  | msgid "OPTIONS" | ||||||
|  | msgstr "ПАРАМЕТРЫ" | ||||||
|  |  | ||||||
|  | #: internal/config/config.go:176 | ||||||
|  | #, fuzzy | ||||||
|  | msgid "Unable to create config directory" | ||||||
| msgstr "Не удалось создать каталог конфигурации ALR" | msgstr "Не удалось создать каталог конфигурации ALR" | ||||||
|  |  | ||||||
| #: internal/config/config.go:106 | #: internal/config/config.go:182 | ||||||
| msgid "Unable to create ALR config file" |  | ||||||
| msgstr "Не удалось создать конфигурационный файл ALR" |  | ||||||
|  |  | ||||||
| #: internal/config/config.go:112 |  | ||||||
| msgid "Error encoding default configuration" |  | ||||||
| msgstr "Ошибка кодирования конфигурации по умолчанию" |  | ||||||
|  |  | ||||||
| #: internal/config/config.go:121 |  | ||||||
| msgid "Unable to detect cache directory" |  | ||||||
| msgstr "Не удалось обнаружить каталог кэша" |  | ||||||
|  |  | ||||||
| #: internal/config/config.go:131 |  | ||||||
| msgid "Unable to create repo cache directory" | msgid "Unable to create repo cache directory" | ||||||
| msgstr "Не удалось создать каталог кэша репозитория" | msgstr "Не удалось создать каталог кэша репозитория" | ||||||
|  |  | ||||||
| #: internal/config/config.go:137 | #: internal/config/config.go:188 | ||||||
| msgid "Unable to create package cache directory" | msgid "Unable to create package cache directory" | ||||||
| msgstr "Не удалось создать каталог кэша пакетов" | msgstr "Не удалось создать каталог кэша пакетов" | ||||||
|  |  | ||||||
| #: internal/config/lang.go:50 | #: internal/db/db.go:133 | ||||||
| msgid "Error parsing system language" |  | ||||||
| msgstr "Ошибка при парсинге языка системы" |  | ||||||
|  |  | ||||||
| #: internal/db/db.go:131 |  | ||||||
| msgid "Database version mismatch; resetting" | msgid "Database version mismatch; resetting" | ||||||
| msgstr "Несоответствие версий базы данных; сброс настроек" | msgstr "Несоответствие версий базы данных; сброс настроек" | ||||||
|  |  | ||||||
| #: internal/db/db.go:135 | #: internal/db/db.go:140 | ||||||
| msgid "" | msgid "" | ||||||
| "Database version does not exist. Run alr fix if something isn't working." | "Database version does not exist. Run alr fix if something isn't working." | ||||||
| msgstr "" | msgstr "" | ||||||
| "Версия базы данных не существует. Запустите alr fix, если что-то не работает." | "Версия базы данных не существует. Запустите alr fix, если что-то не работает." | ||||||
|  |  | ||||||
| #: internal/db/db_legacy.go:101 |  | ||||||
| msgid "Error opening database" |  | ||||||
| msgstr "Ошибка при открытии базы данных" |  | ||||||
|  |  | ||||||
| #: internal/dl/dl.go:170 | #: internal/dl/dl.go:170 | ||||||
| msgid "Source can be updated, updating if required" | msgid "Source can be updated, updating if required" | ||||||
| msgstr "Исходный код можно обновлять, обновляя при необходимости" | msgstr "Исходный код можно обновлять, обновляя при необходимости" | ||||||
| @@ -265,21 +303,21 @@ msgstr "Скачивание источника" | |||||||
|  |  | ||||||
| #: internal/dl/progress_tui.go:100 | #: internal/dl/progress_tui.go:100 | ||||||
| msgid "%s: done!\n" | msgid "%s: done!\n" | ||||||
| msgstr "" | msgstr "%s: выполнено!\n" | ||||||
|  |  | ||||||
| #: internal/dl/progress_tui.go:104 | #: internal/dl/progress_tui.go:104 | ||||||
| msgid "%s %s downloading at %s/s\n" | msgid "%s %s downloading at %s/s\n" | ||||||
| msgstr "" | msgstr "%s %s загружается — %s/с\n" | ||||||
|  |  | ||||||
| #: internal/logger/log.go:47 | #: internal/logger/log.go:47 | ||||||
| msgid "ERROR" | msgid "ERROR" | ||||||
| msgstr "ОШИБКА" | msgstr "ОШИБКА" | ||||||
|  |  | ||||||
| #: list.go:40 | #: list.go:41 | ||||||
| msgid "List ALR repo packages" | msgid "List ALR repo packages" | ||||||
| msgstr "Список пакетов репозитория ALR" | msgstr "Список пакетов репозитория ALR" | ||||||
|  |  | ||||||
| #: list.go:91 | #: list.go:98 | ||||||
| msgid "Error listing installed packages" | msgid "Error listing installed packages" | ||||||
| msgstr "Ошибка при составлении списка установленных пакетов" | msgstr "Ошибка при составлении списка установленных пакетов" | ||||||
|  |  | ||||||
| @@ -295,7 +333,7 @@ msgstr "Аргументы, которые будут переданы мене | |||||||
| msgid "Enable interactive questions and prompts" | msgid "Enable interactive questions and prompts" | ||||||
| msgstr "Включение интерактивных вопросов и запросов" | msgstr "Включение интерактивных вопросов и запросов" | ||||||
|  |  | ||||||
| #: main.go:90 | #: main.go:96 | ||||||
| msgid "" | msgid "" | ||||||
| "Running ALR as root is forbidden as it may cause catastrophic damage to your " | "Running ALR as root is forbidden as it may cause catastrophic damage to your " | ||||||
| "system" | "system" | ||||||
| @@ -303,31 +341,39 @@ msgstr "" | |||||||
| "Запуск ALR от имени root запрещён, так как это может привести к " | "Запуск ALR от имени root запрещён, так как это может привести к " | ||||||
| "катастрофическому повреждению вашей системы" | "катастрофическому повреждению вашей системы" | ||||||
|  |  | ||||||
| #: main.go:124 | #: main.go:154 | ||||||
|  | msgid "Show help" | ||||||
|  | msgstr "Показать справку" | ||||||
|  |  | ||||||
|  | #: main.go:158 | ||||||
| msgid "Error while running app" | msgid "Error while running app" | ||||||
| msgstr "Ошибка при запуске приложения" | msgstr "Ошибка при запуске приложения" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:107 | #: pkg/build/build.go:156 | ||||||
| msgid "Failed to prompt user to view build script" | msgid "Failed to prompt user to view build script" | ||||||
| msgstr "Не удалось предложить пользователю просмотреть скрипт сборки" | msgstr "Не удалось предложить пользователю просмотреть скрипт сборки" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:111 | #: pkg/build/build.go:160 | ||||||
| msgid "Building package" | msgid "Building package" | ||||||
| msgstr "Сборка пакета" | msgstr "Сборка пакета" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:155 | #: pkg/build/build.go:208 | ||||||
|  | msgid "The checksums array must be the same length as sources" | ||||||
|  | msgstr "Массив контрольных сумм должен быть той же длины, что и источники" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:235 | ||||||
| msgid "Downloading sources" | msgid "Downloading sources" | ||||||
| msgstr "Скачивание источников" | msgstr "Скачивание источников" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:167 | #: pkg/build/build.go:257 | ||||||
| msgid "Building package metadata" | msgid "Building package metadata" | ||||||
| msgstr "Сборка метаданных пакета" | msgstr "Сборка метаданных пакета" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:189 | #: pkg/build/build.go:279 | ||||||
| msgid "Compressing package" | msgid "Compressing package" | ||||||
| msgstr "Сжатие пакета" | msgstr "Сжатие пакета" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:315 | #: pkg/build/build.go:438 | ||||||
| msgid "" | msgid "" | ||||||
| "Your system's CPU architecture doesn't match this package. Do you want to " | "Your system's CPU architecture doesn't match this package. Do you want to " | ||||||
| "build anyway?" | "build anyway?" | ||||||
| @@ -335,60 +381,52 @@ msgstr "" | |||||||
| "Архитектура процессора вашей системы не соответствует этому пакету. Вы все " | "Архитектура процессора вашей системы не соответствует этому пакету. Вы все " | ||||||
| "равно хотите выполнить сборку?" | "равно хотите выполнить сборку?" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:326 | #: pkg/build/build.go:452 | ||||||
| msgid "This package is already installed" | msgid "This package is already installed" | ||||||
| msgstr "Этот пакет уже установлен" | msgstr "Этот пакет уже установлен" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:354 | #: pkg/build/build.go:476 | ||||||
| msgid "Installing build dependencies" | msgid "Installing build dependencies" | ||||||
| msgstr "Установка зависимостей сборки" | msgstr "Установка зависимостей сборки" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:396 | #: pkg/build/build.go:521 | ||||||
| msgid "Installing dependencies" | msgid "Installing dependencies" | ||||||
| msgstr "Установка зависимостей" | msgstr "Установка зависимостей" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:442 | #: pkg/build/build.go:602 | ||||||
| msgid "Executing version()" | msgid "Would you like to remove the build dependencies?" | ||||||
| msgstr "Исполнение версия()" | msgstr "Хотели бы вы удалить зависимости сборки?" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:462 | #: pkg/build/build.go:665 | ||||||
| msgid "Updating version" |  | ||||||
| msgstr "Обновление версии" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:467 |  | ||||||
| msgid "Executing prepare()" | msgid "Executing prepare()" | ||||||
| msgstr "Исполнение prepare()" | msgstr "Исполнение prepare()" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:477 | #: pkg/build/build.go:675 | ||||||
| msgid "Executing build()" | msgid "Executing build()" | ||||||
| msgstr "Исполнение build()" | msgstr "Исполнение build()" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:489 | #: pkg/build/build.go:705 pkg/build/build.go:725 | ||||||
| msgid "Executing package()" | msgid "Executing %s()" | ||||||
| msgstr "Исполнение package()" | msgstr "Исполнение %s()" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:527 | #: pkg/build/build.go:784 | ||||||
| msgid "Executing files()" | msgid "Error installing native packages" | ||||||
| msgstr "Исполнение files()" | msgstr "Ошибка при установке нативных пакетов" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:605 | #: pkg/build/build.go:808 | ||||||
|  | msgid "Error installing package" | ||||||
|  | msgstr "Ошибка при установке пакета" | ||||||
|  |  | ||||||
|  | #: pkg/build/build.go:867 | ||||||
| msgid "AutoProv is not implemented for this package format, so it's skipped" | msgid "AutoProv is not implemented for this package format, so it's skipped" | ||||||
| msgstr "" | msgstr "" | ||||||
| "AutoProv не реализовано для этого формата пакета, поэтому будет пропущено" | "AutoProv не реализовано для этого формата пакета, поэтому будет пропущено" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:616 | #: pkg/build/build.go:878 | ||||||
| msgid "AutoReq is not implemented for this package format, so it's skipped" | msgid "AutoReq is not implemented for this package format, so it's skipped" | ||||||
| msgstr "" | msgstr "" | ||||||
| "AutoReq не реализовано для этого формата пакета, поэтому будет пропущено" | "AutoReq не реализовано для этого формата пакета, поэтому будет пропущено" | ||||||
|  |  | ||||||
| #: pkg/build/build.go:723 |  | ||||||
| msgid "Would you like to remove the build dependencies?" |  | ||||||
| msgstr "Хотели бы вы удалить зависимости сборки?" |  | ||||||
|  |  | ||||||
| #: pkg/build/build.go:829 |  | ||||||
| msgid "The checksums array must be the same length as sources" |  | ||||||
| msgstr "Массив контрольных сумм должен быть той же длины, что и источники" |  | ||||||
|  |  | ||||||
| #: pkg/build/findDeps.go:35 | #: pkg/build/findDeps.go:35 | ||||||
| msgid "Command not found on the system" | msgid "Command not found on the system" | ||||||
| msgstr "Команда не найдена в системе" | msgstr "Команда не найдена в системе" | ||||||
| @@ -401,27 +439,19 @@ msgstr "Найденная предоставленная зависимость | |||||||
| msgid "Required dependency found" | msgid "Required dependency found" | ||||||
| msgstr "Найдена требуемая зависимость" | msgstr "Найдена требуемая зависимость" | ||||||
|  |  | ||||||
| #: pkg/build/install.go:42 | #: pkg/repos/pull.go:79 | ||||||
| msgid "Error installing native packages" |  | ||||||
| msgstr "Ошибка при установке нативных пакетов" |  | ||||||
|  |  | ||||||
| #: pkg/build/install.go:79 |  | ||||||
| msgid "Error installing package" |  | ||||||
| msgstr "Ошибка при установке пакета" |  | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:75 |  | ||||||
| msgid "Pulling repository" | msgid "Pulling repository" | ||||||
| msgstr "Скачивание репозитория" | msgstr "Скачивание репозитория" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:99 | #: pkg/repos/pull.go:103 | ||||||
| msgid "Repository up to date" | msgid "Repository up to date" | ||||||
| msgstr "Репозиторий уже обновлён" | msgstr "Репозиторий уже обновлён" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:156 | #: pkg/repos/pull.go:160 | ||||||
| msgid "Git repository does not appear to be a valid ALR repo" | msgid "Git repository does not appear to be a valid ALR repo" | ||||||
| msgstr "Репозиторий Git не поддерживается репозиторием ALR" | msgstr "Репозиторий Git не поддерживается репозиторием ALR" | ||||||
|  |  | ||||||
| #: pkg/repos/pull.go:172 | #: pkg/repos/pull.go:176 | ||||||
| msgid "" | msgid "" | ||||||
| "ALR repo's minimum ALR version is greater than the current version. Try " | "ALR repo's minimum ALR version is greater than the current version. Try " | ||||||
| "updating ALR if something doesn't work." | "updating ALR if something doesn't work." | ||||||
| @@ -429,58 +459,127 @@ msgstr "" | |||||||
| "Минимальная версия ALR для ALR-репозитория выше текущей версии. Попробуйте " | "Минимальная версия ALR для ALR-репозитория выше текущей версии. Попробуйте " | ||||||
| "обновить ALR, если что-то не работает." | "обновить ALR, если что-то не работает." | ||||||
|  |  | ||||||
| #: repo.go:41 | #: repo.go:40 | ||||||
| msgid "Add a new repository" | msgid "Add a new repository" | ||||||
| msgstr "Добавить новый репозиторий" | msgstr "Добавить новый репозиторий" | ||||||
|  |  | ||||||
| #: repo.go:48 | #: repo.go:47 | ||||||
| msgid "Name of the new repo" | msgid "Name of the new repo" | ||||||
| msgstr "Название нового репозитория" | msgstr "Название нового репозитория" | ||||||
|  |  | ||||||
| #: repo.go:54 | #: repo.go:53 | ||||||
| msgid "URL of the new repo" | msgid "URL of the new repo" | ||||||
| msgstr "URL-адрес нового репозитория" | msgstr "URL-адрес нового репозитория" | ||||||
|  |  | ||||||
| #: repo.go:79 repo.go:136 | #: repo.go:86 repo.go:156 | ||||||
| msgid "Error opening config file" | #, fuzzy | ||||||
| msgstr "Ошибка при открытии конфигурационного файла" | msgid "Error saving config" | ||||||
|  |  | ||||||
| #: repo.go:85 repo.go:142 |  | ||||||
| msgid "Error encoding config" |  | ||||||
| msgstr "Ошибка при кодировании конфигурации" | msgstr "Ошибка при кодировании конфигурации" | ||||||
|  |  | ||||||
| #: repo.go:103 | #: repo.go:111 | ||||||
| msgid "Remove an existing repository" | msgid "Remove an existing repository" | ||||||
| msgstr "Удалить существующий репозиторий" | msgstr "Удалить существующий репозиторий" | ||||||
|  |  | ||||||
| #: repo.go:110 | #: repo.go:118 | ||||||
| msgid "Name of the repo to be deleted" | msgid "Name of the repo to be deleted" | ||||||
| msgstr "Название репозитория  удалён" | msgstr "Название репозитория  удалён" | ||||||
|  |  | ||||||
| #: repo.go:128 | #: repo.go:142 | ||||||
| msgid "Repo does not exist" | msgid "Repo does not exist" | ||||||
| msgstr "Репозитория не существует" | msgstr "Репозитория не существует" | ||||||
|  |  | ||||||
| #: repo.go:148 | #: repo.go:150 | ||||||
| msgid "Error removing repo directory" | msgid "Error removing repo directory" | ||||||
| msgstr "Ошибка при удалении каталога репозитория" | msgstr "Ошибка при удалении каталога репозитория" | ||||||
|  |  | ||||||
| #: repo.go:154 | #: repo.go:167 | ||||||
| msgid "Error removing packages from database" | msgid "Error removing packages from database" | ||||||
| msgstr "Ошибка при удалении пакетов из базы данных" | msgstr "Ошибка при удалении пакетов из базы данных" | ||||||
|  |  | ||||||
| #: repo.go:166 | #: repo.go:179 | ||||||
| msgid "Pull all repositories that have changed" | msgid "Pull all repositories that have changed" | ||||||
| msgstr "Скачать все изменённые репозитории" | msgstr "Скачать все изменённые репозитории" | ||||||
|  |  | ||||||
|  | #: search.go:36 | ||||||
|  | msgid "Search packages" | ||||||
|  | msgstr "Поиск пакетов" | ||||||
|  |  | ||||||
|  | #: search.go:42 | ||||||
|  | msgid "Search by name" | ||||||
|  | msgstr "Искать по имени" | ||||||
|  |  | ||||||
|  | #: search.go:47 | ||||||
|  | msgid "Search by description" | ||||||
|  | msgstr "Искать по описанию" | ||||||
|  |  | ||||||
|  | #: search.go:52 | ||||||
|  | msgid "Search by repository" | ||||||
|  | msgstr "Искать по репозиторию" | ||||||
|  |  | ||||||
|  | #: search.go:57 | ||||||
|  | msgid "Search by provides" | ||||||
|  | msgstr "Иcкать по provides" | ||||||
|  |  | ||||||
|  | #: search.go:62 | ||||||
|  | msgid "Format output using a Go template" | ||||||
|  | msgstr "Формат выходных данных с использованием шаблона Go" | ||||||
|  |  | ||||||
|  | #: search.go:88 search.go:105 | ||||||
|  | msgid "Error parsing format template" | ||||||
|  | msgstr "Ошибка при разборе шаблона" | ||||||
|  |  | ||||||
|  | #: search.go:113 | ||||||
|  | msgid "Error executing template" | ||||||
|  | msgstr "Ошибка при выполнении шаблона" | ||||||
|  |  | ||||||
| #: upgrade.go:47 | #: upgrade.go:47 | ||||||
| msgid "Upgrade all installed packages" | msgid "Upgrade all installed packages" | ||||||
| msgstr "Обновить все установленные пакеты" | msgstr "Обновить все установленные пакеты" | ||||||
|  |  | ||||||
| #: upgrade.go:83 | #: upgrade.go:96 | ||||||
| msgid "Error checking for updates" | msgid "Error checking for updates" | ||||||
| msgstr "Ошибка при проверке обновлений" | msgstr "Ошибка при проверке обновлений" | ||||||
|  |  | ||||||
| #: upgrade.go:94 | #: upgrade.go:118 | ||||||
| msgid "There is nothing to do." | msgid "There is nothing to do." | ||||||
| msgstr "Здесь нечего делать." | msgstr "Здесь нечего делать." | ||||||
|  |  | ||||||
|  | #~ msgid "Error opening config file, using defaults" | ||||||
|  | #~ msgstr "" | ||||||
|  | #~ "Ошибка при открытии конфигурационного файла, используются значения по " | ||||||
|  | #~ "умолчанию" | ||||||
|  |  | ||||||
|  | #~ msgid "Error decoding config file, using defaults" | ||||||
|  | #~ msgstr "" | ||||||
|  | #~ "Ошибка при декодировании конфигурационного файла, используются значения " | ||||||
|  | #~ "по умолчанию" | ||||||
|  |  | ||||||
|  | #~ msgid "Unable to detect user config directory" | ||||||
|  | #~ msgstr "Не удалось обнаружить каталог конфигурации пользователя" | ||||||
|  |  | ||||||
|  | #~ msgid "Unable to create ALR config file" | ||||||
|  | #~ msgstr "Не удалось создать конфигурационный файл ALR" | ||||||
|  |  | ||||||
|  | #~ msgid "Error encoding default configuration" | ||||||
|  | #~ msgstr "Ошибка кодирования конфигурации по умолчанию" | ||||||
|  |  | ||||||
|  | #~ msgid "Unable to detect cache directory" | ||||||
|  | #~ msgstr "Не удалось обнаружить каталог кэша" | ||||||
|  |  | ||||||
|  | #~ msgid "Error opening config file" | ||||||
|  | #~ msgstr "Ошибка при открытии конфигурационного файла" | ||||||
|  |  | ||||||
|  | #~ msgid "Error parsing system language" | ||||||
|  | #~ msgstr "Ошибка при парсинге языка системы" | ||||||
|  |  | ||||||
|  | #~ msgid "Error opening database" | ||||||
|  | #~ msgstr "Ошибка при открытии базы данных" | ||||||
|  |  | ||||||
|  | #~ msgid "Executing version()" | ||||||
|  | #~ msgstr "Исполнение версия()" | ||||||
|  |  | ||||||
|  | #~ msgid "Updating version" | ||||||
|  | #~ msgstr "Обновление версии" | ||||||
|  |  | ||||||
|  | #~ msgid "Executing package()" | ||||||
|  | #~ msgstr "Исполнение package()" | ||||||
|   | |||||||
| @@ -23,11 +23,62 @@ import "gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | |||||||
|  |  | ||||||
| type BuildOpts struct { | type BuildOpts struct { | ||||||
| 	Script      string | 	Script      string | ||||||
|  | 	Repository  string | ||||||
|  | 	Packages    []string | ||||||
| 	Manager     manager.Manager | 	Manager     manager.Manager | ||||||
| 	Clean       bool | 	Clean       bool | ||||||
| 	Interactive bool | 	Interactive bool | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type BuildVarsPre struct { | ||||||
|  | 	Version       string   `sh:"version,required"` | ||||||
|  | 	Release       int      `sh:"release,required"` | ||||||
|  | 	Epoch         uint     `sh:"epoch"` | ||||||
|  | 	Description   string   `sh:"desc"` | ||||||
|  | 	Homepage      string   `sh:"homepage"` | ||||||
|  | 	Maintainer    string   `sh:"maintainer"` | ||||||
|  | 	Architectures []string `sh:"architectures"` | ||||||
|  | 	Licenses      []string `sh:"license"` | ||||||
|  | 	Provides      []string `sh:"provides"` | ||||||
|  | 	Conflicts     []string `sh:"conflicts"` | ||||||
|  | 	Depends       []string `sh:"deps"` | ||||||
|  | 	BuildDepends  []string `sh:"build_deps"` | ||||||
|  | 	OptDepends    []string `sh:"opt_deps"` | ||||||
|  | 	Replaces      []string `sh:"replaces"` | ||||||
|  | 	Sources       []string `sh:"sources"` | ||||||
|  | 	Checksums     []string `sh:"checksums"` | ||||||
|  | 	Backup        []string `sh:"backup"` | ||||||
|  | 	Scripts       Scripts  `sh:"scripts"` | ||||||
|  | 	AutoReq       []string `sh:"auto_req"` | ||||||
|  | 	AutoProv      []string `sh:"auto_prov"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (bv *BuildVarsPre) ToBuildVars() BuildVars { | ||||||
|  | 	return BuildVars{ | ||||||
|  | 		Name:          "", | ||||||
|  | 		Version:       bv.Version, | ||||||
|  | 		Release:       bv.Release, | ||||||
|  | 		Epoch:         bv.Epoch, | ||||||
|  | 		Description:   bv.Description, | ||||||
|  | 		Homepage:      bv.Homepage, | ||||||
|  | 		Maintainer:    bv.Maintainer, | ||||||
|  | 		Architectures: bv.Architectures, | ||||||
|  | 		Licenses:      bv.Licenses, | ||||||
|  | 		Provides:      bv.Provides, | ||||||
|  | 		Conflicts:     bv.Conflicts, | ||||||
|  | 		Depends:       bv.Depends, | ||||||
|  | 		BuildDepends:  bv.BuildDepends, | ||||||
|  | 		OptDepends:    bv.OptDepends, | ||||||
|  | 		Replaces:      bv.Replaces, | ||||||
|  | 		Sources:       bv.Sources, | ||||||
|  | 		Checksums:     bv.Checksums, | ||||||
|  | 		Backup:        bv.Backup, | ||||||
|  | 		Scripts:       bv.Scripts, | ||||||
|  | 		AutoReq:       bv.AutoReq, | ||||||
|  | 		AutoProv:      bv.AutoProv, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| // BuildVars represents the script variables required | // BuildVars represents the script variables required | ||||||
| // to build a package | // to build a package | ||||||
| type BuildVars struct { | type BuildVars struct { | ||||||
| @@ -52,6 +103,7 @@ type BuildVars struct { | |||||||
| 	Scripts       Scripts  `sh:"scripts"` | 	Scripts       Scripts  `sh:"scripts"` | ||||||
| 	AutoReq       []string `sh:"auto_req"` | 	AutoReq       []string `sh:"auto_req"` | ||||||
| 	AutoProv      []string `sh:"auto_prov"` | 	AutoProv      []string `sh:"auto_prov"` | ||||||
|  | 	Base          string | ||||||
| } | } | ||||||
|  |  | ||||||
| type Scripts struct { | type Scripts struct { | ||||||
|   | |||||||
| @@ -21,12 +21,13 @@ package types | |||||||
|  |  | ||||||
| // Config represents the ALR configuration file | // Config represents the ALR configuration file | ||||||
| type Config struct { | type Config struct { | ||||||
| 	RootCmd          string   `toml:"rootCmd"` | 	RootCmd          string   `toml:"rootCmd" env:"ALR_ROOT_CMD"` | ||||||
| 	PagerStyle       string   `toml:"pagerStyle"` | 	PagerStyle       string   `toml:"pagerStyle" env:"ALR_PAGER_STYLE"` | ||||||
| 	IgnorePkgUpdates []string `toml:"ignorePkgUpdates"` | 	IgnorePkgUpdates []string `toml:"ignorePkgUpdates"` | ||||||
| 	Repos            []Repo   `toml:"repo"` | 	Repos            []Repo   `toml:"repo"` | ||||||
| 	Unsafe           Unsafe   `toml:"unsafe"` | 	Unsafe           Unsafe   `toml:"unsafe"` | ||||||
| 	AutoPull         bool     `toml:"autoPull"` | 	AutoPull         bool     `toml:"autoPull" env:"ALR_AUTOPULL"` | ||||||
|  | 	LogLevel         string   `toml:"logLevel" env:"ALR_LOG_LEVEL"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // Repo represents a ALR repo within a configuration file | // Repo represents a ALR repo within a configuration file | ||||||
| @@ -36,5 +37,5 @@ type Repo struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| type Unsafe struct { | type Unsafe struct { | ||||||
| 	AllowRunAsRoot bool `toml:"allowRunAsRoot"` | 	AllowRunAsRoot bool `toml:"allowRunAsRoot" env:"ALR_UNSAFE_ALLOW_RUN_AS_ROOT"` | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								list.go
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								list.go
									
									
									
									
									
								
							| @@ -30,6 +30,7 @@ import ( | |||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| ) | ) | ||||||
| @@ -48,16 +49,22 @@ func ListCmd() *cli.Command { | |||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
| 			cfg := config.New() | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			db := database.New(cfg) | 			db := database.New(cfg) | ||||||
| 			err := db.Init(ctx) | 			err = db.Init(ctx) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error initialization database"), "err", err) | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
| 			rs := repos.New(cfg, db) | 			rs := repos.New(cfg, db) | ||||||
|  |  | ||||||
| 			if cfg.AutoPull(ctx) { | 			if cfg.AutoPull() { | ||||||
| 				err = rs.Pull(ctx, cfg.Repos(ctx)) | 				err = rs.Pull(ctx, cfg.Repos()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | 					slog.Error(gotext.Get("Error pulling repositories"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| @@ -78,7 +85,7 @@ func ListCmd() *cli.Command { | |||||||
| 			} | 			} | ||||||
| 			defer result.Close() | 			defer result.Close() | ||||||
|  |  | ||||||
| 			var installed map[string]string | 			installedAlrPackages := map[string]string{} | ||||||
| 			if c.Bool("installed") { | 			if c.Bool("installed") { | ||||||
| 				mgr := manager.Detect() | 				mgr := manager.Detect() | ||||||
| 				if mgr == nil { | 				if mgr == nil { | ||||||
| @@ -86,11 +93,20 @@ func ListCmd() *cli.Command { | |||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				installed, err = mgr.ListInstalled(&manager.Opts{AsRoot: false}) | 				installed, err := mgr.ListInstalled(&manager.Opts{AsRoot: false}) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error listing installed packages"), "err", err) | 					slog.Error(gotext.Get("Error listing installed packages"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
|  | 				for pkgName, version := range installed { | ||||||
|  | 					matches := build.RegexpALRPackageName.FindStringSubmatch(pkgName) | ||||||
|  | 					if matches != nil { | ||||||
|  | 						packageName := matches[build.RegexpALRPackageName.SubexpIndex("package")] | ||||||
|  | 						repoName := matches[build.RegexpALRPackageName.SubexpIndex("repo")] | ||||||
|  | 						installedAlrPackages[fmt.Sprintf("%s/%s", repoName, packageName)] = version | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			for result.Next() { | 			for result.Next() { | ||||||
| @@ -100,13 +116,13 @@ func ListCmd() *cli.Command { | |||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				if slices.Contains(cfg.IgnorePkgUpdates(ctx), pkg.Name) { | 				if slices.Contains(cfg.IgnorePkgUpdates(), pkg.Name) { | ||||||
| 					continue | 					continue | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				version := pkg.Version | 				version := pkg.Version | ||||||
| 				if c.Bool("installed") { | 				if c.Bool("installed") { | ||||||
| 					instVersion, ok := installed[pkg.Name] | 					instVersion, ok := installedAlrPackages[fmt.Sprintf("%s/%s", pkg.Repository, pkg.Name)] | ||||||
| 					if !ok { | 					if !ok { | ||||||
| 						continue | 						continue | ||||||
| 					} else { | 					} else { | ||||||
|   | |||||||
							
								
								
									
										58
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										58
									
								
								main.go
									
									
									
									
									
								
							| @@ -31,8 +31,8 @@ import ( | |||||||
| 	"github.com/mattn/go-isatty" | 	"github.com/mattn/go-isatty" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cliutils" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/translations" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/translations" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
|  |  | ||||||
| @@ -81,12 +81,18 @@ func GetApp() *cli.App { | |||||||
| 			GenCmd(), | 			GenCmd(), | ||||||
| 			HelperCmd(), | 			HelperCmd(), | ||||||
| 			VersionCmd(), | 			VersionCmd(), | ||||||
|  | 			SearchCmd(), | ||||||
| 		}, | 		}, | ||||||
| 		Before: func(c *cli.Context) error { | 		Before: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			cmd := c.Args().First() | 			cmd := c.Args().First() | ||||||
| 			if cmd != "helper" && !config.Config(ctx).Unsafe.AllowRunAsRoot && os.Geteuid() == 0 { | 			if cmd != "helper" && !cfg.AllowRunAsRoot() && os.Geteuid() == 0 { | ||||||
| 				slog.Error(gotext.Get("Running ALR as root is forbidden as it may cause catastrophic damage to your system")) | 				slog.Error(gotext.Get("Running ALR as root is forbidden as it may cause catastrophic damage to your system")) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
| @@ -98,28 +104,56 @@ func GetApp() *cli.App { | |||||||
|  |  | ||||||
| 			return nil | 			return nil | ||||||
| 		}, | 		}, | ||||||
| 		After: func(ctx *cli.Context) error { |  | ||||||
| 			return db.Close() |  | ||||||
| 		}, |  | ||||||
| 		EnableBashCompletion: true, | 		EnableBashCompletion: true, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func main() { | func setLogLevel(newLevel string) { | ||||||
| 	translations.Setup() | 	level := slog.LevelInfo | ||||||
| 	logger.SetupDefault() | 	switch newLevel { | ||||||
|  | 	case "DEBUG": | ||||||
|  | 		level = slog.LevelDebug | ||||||
|  | 	case "INFO": | ||||||
|  | 		level = slog.LevelInfo | ||||||
|  | 	case "WARN": | ||||||
|  | 		level = slog.LevelWarn | ||||||
|  | 	case "ERROR": | ||||||
|  | 		level = slog.LevelError | ||||||
|  | 	} | ||||||
|  | 	logger, ok := slog.Default().Handler().(*logger.Logger) | ||||||
|  | 	if !ok { | ||||||
|  | 		panic("unexpected") | ||||||
|  | 	} | ||||||
|  | 	logger.SetLevel(level) | ||||||
|  | } | ||||||
|  |  | ||||||
| 	app := GetApp() | func main() { | ||||||
|  | 	logger.SetupDefault() | ||||||
|  | 	setLogLevel(os.Getenv("ALR_LOG_LEVEL")) | ||||||
|  | 	translations.Setup() | ||||||
|  |  | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
|  |  | ||||||
|  | 	app := GetApp() | ||||||
|  | 	cfg := config.New() | ||||||
|  | 	err := cfg.Load() | ||||||
|  | 	if err != nil { | ||||||
|  | 		slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 	setLogLevel(cfg.LogLevel()) | ||||||
| 	// Set the root command to the one set in the ALR config | 	// Set the root command to the one set in the ALR config | ||||||
| 	manager.DefaultRootCmd = config.Config(ctx).RootCmd | 	manager.DefaultRootCmd = cfg.RootCmd() | ||||||
|  |  | ||||||
| 	ctx, cancel := signal.NotifyContext(ctx, syscall.SIGINT, syscall.SIGTERM) | 	ctx, cancel := signal.NotifyContext(ctx, syscall.SIGINT, syscall.SIGTERM) | ||||||
| 	defer cancel() | 	defer cancel() | ||||||
|  |  | ||||||
| 	err := app.RunContext(ctx, os.Args) | 	// Make the application more internationalized | ||||||
|  | 	cli.AppHelpTemplate = cliutils.GetAppCliTemplate() | ||||||
|  | 	cli.CommandHelpTemplate = cliutils.GetCommandHelpTemplate() | ||||||
|  | 	cli.HelpFlag.(*cli.BoolFlag).Usage = gotext.Get("Show help") | ||||||
|  |  | ||||||
|  | 	err = app.RunContext(ctx, os.Args) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		slog.Error(gotext.Get("Error while running app"), "err", err) | 		slog.Error(gotext.Get("Error while running app"), "err", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -18,12 +18,18 @@ package build | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
|  | 	"mvdan.cc/sh/v3/syntax" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -136,90 +142,145 @@ func (m *TestManager) IsInstalled(pkg string) (bool, error) { | |||||||
| 	return true, nil | 	return true, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestInstallBuildDeps(t *testing.T) { | type TestConfig struct{} | ||||||
| 	type testEnv struct { |  | ||||||
| 		pf   PackageFinder |  | ||||||
| 		vars *types.BuildVars |  | ||||||
| 		opts types.BuildOpts |  | ||||||
|  |  | ||||||
| 		// Contains pkgs captured by FindPkgsFunc | func (c *TestConfig) PagerStyle() string { | ||||||
| 		capturedPkgs []string | 	return "native" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestConfig) GetPaths() *config.Paths { | ||||||
|  | 	return &config.Paths{ | ||||||
|  | 		CacheDir: "/tmp", | ||||||
| 	} | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| 	type testCase struct { | func TestExecuteFirstPassIsSecure(t *testing.T) { | ||||||
| 		Name     string | 	cfg := &TestConfig{} | ||||||
| 		Prepare  func() *testEnv | 	pf := &TestPackageFinder{} | ||||||
| 		Expected func(t *testing.T, e *testEnv, res []string, err error) | 	info := &distro.OSRelease{} | ||||||
| 	} | 	m := &TestManager{} | ||||||
|  |  | ||||||
| 	for _, tc := range []testCase{ |  | ||||||
| 		{ |  | ||||||
| 			Name: "install only needed deps", |  | ||||||
| 			Prepare: func() *testEnv { |  | ||||||
| 				pf := TestPackageFinder{} |  | ||||||
| 				vars := types.BuildVars{} |  | ||||||
| 				m := TestManager{} |  | ||||||
| 	opts := types.BuildOpts{ | 	opts := types.BuildOpts{ | ||||||
| 					Manager:     &m, | 		Manager:     m, | ||||||
| 		Interactive: false, | 		Interactive: false, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 				env := &testEnv{ |  | ||||||
| 					pf:           &pf, |  | ||||||
| 					vars:         &vars, |  | ||||||
| 					opts:         opts, |  | ||||||
| 					capturedPkgs: []string{}, |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				pf.FindPkgsFunc = func(ctx context.Context, pkgs []string) (map[string][]db.Package, []string, error) { |  | ||||||
| 					env.capturedPkgs = append(env.capturedPkgs, pkgs...) |  | ||||||
| 					result := make(map[string][]db.Package) |  | ||||||
| 					result["bar"] = []db.Package{{ |  | ||||||
| 						Name: "bar-pkg", |  | ||||||
| 					}} |  | ||||||
| 					result["buz"] = []db.Package{{ |  | ||||||
| 						Name: "buz-pkg", |  | ||||||
| 					}} |  | ||||||
|  |  | ||||||
| 					return result, []string{}, nil |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				vars.BuildDepends = []string{ |  | ||||||
| 					"foo", |  | ||||||
| 					"bar", |  | ||||||
| 					"buz", |  | ||||||
| 				} |  | ||||||
| 				m.IsInstalledFunc = func(pkg string) (bool, error) { |  | ||||||
| 					if pkg == "foo" { |  | ||||||
| 						return true, nil |  | ||||||
| 					} else { |  | ||||||
| 						return false, nil |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				return env |  | ||||||
| 			}, |  | ||||||
| 			Expected: func(t *testing.T, e *testEnv, res []string, err error) { |  | ||||||
| 				assert.NoError(t, err) |  | ||||||
| 				assert.Len(t, res, 2) |  | ||||||
| 				assert.ElementsMatch(t, res, []string{"bar-pkg", "buz-pkg"}) |  | ||||||
|  |  | ||||||
| 				assert.ElementsMatch(t, e.capturedPkgs, []string{"bar", "buz"}) |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 	} { |  | ||||||
| 		t.Run(tc.Name, func(tt *testing.T) { |  | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
| 			env := tc.Prepare() |  | ||||||
|  |  | ||||||
| 			result, err := installBuildDeps( | 	b := NewBuilder( | ||||||
| 		ctx, | 		ctx, | ||||||
| 				env.pf, | 		opts, | ||||||
| 				env.vars, | 		pf, | ||||||
| 				env.opts, | 		info, | ||||||
|  | 		cfg, | ||||||
| 	) | 	) | ||||||
|  |  | ||||||
| 			tc.Expected(tt, env, result, err) | 	tmpFile, err := os.CreateTemp("", "testfile-") | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	tmpFilePath := tmpFile.Name() | ||||||
|  | 	defer os.Remove(tmpFilePath) | ||||||
|  |  | ||||||
|  | 	_, err = os.Stat(tmpFilePath) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 	testScript := fmt.Sprintf(`name='test' | ||||||
|  | version=1.0.0 | ||||||
|  | release=1 | ||||||
|  | rm -f %s`, tmpFilePath) | ||||||
|  |  | ||||||
|  | 	fl, err := syntax.NewParser().Parse(strings.NewReader(testScript), "alr.sh") | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 	_, _, err = b.executeFirstPass(fl) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 	_, err = os.Stat(tmpFilePath) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestExecuteFirstPassIsCorrect(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		Name     string | ||||||
|  | 		Script   string | ||||||
|  | 		Opts     types.BuildOpts | ||||||
|  | 		Expected func(t *testing.T, vars []*types.BuildVars) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{{ | ||||||
|  | 		Name: "single package", | ||||||
|  | 		Script: `name='test' | ||||||
|  | version='1.0.0' | ||||||
|  | release=1 | ||||||
|  | epoch=2 | ||||||
|  | desc="Test package" | ||||||
|  | homepage='https://example.com' | ||||||
|  | maintainer='Ivan Ivanov' | ||||||
|  | `, | ||||||
|  | 		Opts: types.BuildOpts{ | ||||||
|  | 			Manager:     &TestManager{}, | ||||||
|  | 			Interactive: false, | ||||||
|  | 		}, | ||||||
|  | 		Expected: func(t *testing.T, vars []*types.BuildVars) { | ||||||
|  | 			assert.Equal(t, 1, len(vars)) | ||||||
|  | 			assert.Equal(t, vars[0].Name, "test") | ||||||
|  | 			assert.Equal(t, vars[0].Version, "1.0.0") | ||||||
|  | 			assert.Equal(t, vars[0].Release, int(1)) | ||||||
|  | 			assert.Equal(t, vars[0].Epoch, uint(2)) | ||||||
|  | 			assert.Equal(t, vars[0].Description, "Test package") | ||||||
|  | 		}, | ||||||
|  | 	}, { | ||||||
|  | 		Name: "multiple packages", | ||||||
|  | 		Script: `name=( | ||||||
|  | 	foo | ||||||
|  | 	bar | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | version='0.0.1' | ||||||
|  | release=1 | ||||||
|  | epoch=2 | ||||||
|  | desc="Test package" | ||||||
|  |  | ||||||
|  | meta_foo() { | ||||||
|  | 	desc="Foo package" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | meta_bar() { | ||||||
|  |  | ||||||
|  | } | ||||||
|  | `, | ||||||
|  | 		Opts: types.BuildOpts{ | ||||||
|  | 			Packages:    []string{"foo"}, | ||||||
|  | 			Manager:     &TestManager{}, | ||||||
|  | 			Interactive: false, | ||||||
|  | 		}, | ||||||
|  | 		Expected: func(t *testing.T, vars []*types.BuildVars) { | ||||||
|  | 			assert.Equal(t, 1, len(vars)) | ||||||
|  | 			assert.Equal(t, vars[0].Name, "foo") | ||||||
|  | 			assert.Equal(t, vars[0].Description, "Foo package") | ||||||
|  | 		}, | ||||||
|  | 	}} { | ||||||
|  | 		t.Run(tc.Name, func(t *testing.T) { | ||||||
|  | 			cfg := &TestConfig{} | ||||||
|  | 			pf := &TestPackageFinder{} | ||||||
|  | 			info := &distro.OSRelease{} | ||||||
|  |  | ||||||
|  | 			ctx := context.Background() | ||||||
|  |  | ||||||
|  | 			b := NewBuilder( | ||||||
|  | 				ctx, | ||||||
|  | 				tc.Opts, | ||||||
|  | 				pf, | ||||||
|  | 				info, | ||||||
|  | 				cfg, | ||||||
|  | 			) | ||||||
|  |  | ||||||
|  | 			fl, err := syntax.NewParser().Parse(strings.NewReader(tc.Script), "alr.sh") | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			_, allVars, err := b.executeFirstPass(fl) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			tc.Expected(t, allVars) | ||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,84 +0,0 @@ | |||||||
| // This file was originally part of the project "LURE - Linux User REpository", created by Elara Musayelyan. |  | ||||||
| // It has been modified as part of "ALR - Any Linux Repository" by Евгений Храмов. |  | ||||||
| // |  | ||||||
| // ALR - Any Linux Repository |  | ||||||
| // Copyright (C) 2025 Евгений Храмов |  | ||||||
| // |  | ||||||
| // 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 build |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"log/slog" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
|  |  | ||||||
| 	"github.com/leonelquinteros/gotext" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // InstallPkgs устанавливает нативные пакеты с использованием менеджера пакетов, |  | ||||||
| // затем строит и устанавливает пакеты ALR |  | ||||||
| func InstallPkgs(ctx context.Context, alrPkgs []db.Package, nativePkgs []string, opts types.BuildOpts) { |  | ||||||
| 	if len(nativePkgs) > 0 { |  | ||||||
| 		err := opts.Manager.Install(nil, nativePkgs...) |  | ||||||
| 		// Если есть нативные пакеты, выполняем их установку |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error installing native packages"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 			// Логируем и завершаем выполнение при ошибке |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	InstallScripts(ctx, GetScriptPaths(ctx, alrPkgs), opts) |  | ||||||
| 	// Устанавливаем скрипты сборки через функцию InstallScripts |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetScriptPaths возвращает срез путей к скриптам, соответствующий |  | ||||||
| // данным пакетам |  | ||||||
| func GetScriptPaths(ctx context.Context, pkgs []db.Package) []string { |  | ||||||
| 	var scripts []string |  | ||||||
| 	for _, pkg := range pkgs { |  | ||||||
| 		// Для каждого пакета создаем путь к скрипту сборки |  | ||||||
| 		scriptPath := filepath.Join(config.GetPaths(ctx).RepoDir, pkg.Repository, pkg.Name, "alr.sh") |  | ||||||
| 		scripts = append(scripts, scriptPath) |  | ||||||
| 	} |  | ||||||
| 	return scripts |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // InstallScripts строит и устанавливает переданные alr скрипты сборки |  | ||||||
| func InstallScripts(ctx context.Context, scripts []string, opts types.BuildOpts) { |  | ||||||
| 	for _, script := range scripts { |  | ||||||
| 		opts.Script = script // Устанавливаем текущий скрипт в опции |  | ||||||
| 		builtPkgs, _, err := BuildPackage(ctx, opts) |  | ||||||
| 		// Выполняем сборку пакета |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error building package"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 			// Логируем и завершаем выполнение при ошибке сборки |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		err = opts.Manager.InstallLocal(nil, builtPkgs...) |  | ||||||
| 		// Устанавливаем локально собранные пакеты |  | ||||||
| 		if err != nil { |  | ||||||
| 			slog.Error(gotext.Get("Error installing package"), "err", err) |  | ||||||
| 			os.Exit(1) |  | ||||||
| 			// Логируем и завершаем выполнение при ошибке установки |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
							
								
								
									
										336
									
								
								pkg/build/utils.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										336
									
								
								pkg/build/utils.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,336 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 build | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"regexp" | ||||||
|  | 	"runtime" | ||||||
|  | 	"slices" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	// Импортируем пакеты для поддержки различных форматов пакетов (APK, DEB, RPM и ARCH). | ||||||
|  |  | ||||||
|  | 	_ "github.com/goreleaser/nfpm/v2/apk" | ||||||
|  | 	_ "github.com/goreleaser/nfpm/v2/arch" | ||||||
|  | 	_ "github.com/goreleaser/nfpm/v2/deb" | ||||||
|  | 	_ "github.com/goreleaser/nfpm/v2/rpm" | ||||||
|  | 	"mvdan.cc/sh/v3/syntax" | ||||||
|  |  | ||||||
|  | 	"github.com/goreleaser/nfpm/v2" | ||||||
|  | 	"github.com/goreleaser/nfpm/v2/files" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/cpu" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/overrides" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Функция readScript анализирует скрипт сборки с использованием встроенной реализации bash | ||||||
|  | func readScript(script string) (*syntax.File, error) { | ||||||
|  | 	fl, err := os.Open(script) // Открываем файл скрипта | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	defer fl.Close() // Закрываем файл после выполнения | ||||||
|  |  | ||||||
|  | 	file, err := syntax.NewParser().Parse(fl, "alr.sh") // Парсим скрипт с помощью синтаксического анализатора | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return file, nil // Возвращаем синтаксическое дерево | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция prepareDirs подготавливает директории для сборки. | ||||||
|  | func prepareDirs(dirs types.Directories) error { | ||||||
|  | 	err := os.RemoveAll(dirs.BaseDir) // Удаляем базовую директорию, если она существует | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	err = os.MkdirAll(dirs.SrcDir, 0o755) // Создаем директорию для источников | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return os.MkdirAll(dirs.PkgDir, 0o755) // Создаем директорию для пакетов | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция buildContents создает секцию содержимого пакета, которая содержит файлы, | ||||||
|  | // которые будут включены в конечный пакет. | ||||||
|  | func buildContents(vars *types.BuildVars, dirs types.Directories, preferedContents *[]string) ([]*files.Content, error) { | ||||||
|  | 	contents := []*files.Content{} | ||||||
|  |  | ||||||
|  | 	processPath := func(path, trimmed string, prefered bool) error { | ||||||
|  | 		fi, err := os.Lstat(path) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if fi.IsDir() { | ||||||
|  | 			f, err := os.Open(path) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			defer f.Close() | ||||||
|  |  | ||||||
|  | 			if !prefered { | ||||||
|  | 				_, err = f.Readdirnames(1) | ||||||
|  | 				if err != io.EOF { | ||||||
|  | 					return nil | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			contents = append(contents, &files.Content{ | ||||||
|  | 				Source:      path, | ||||||
|  | 				Destination: trimmed, | ||||||
|  | 				Type:        "dir", | ||||||
|  | 				FileInfo: &files.ContentFileInfo{ | ||||||
|  | 					MTime: fi.ModTime(), | ||||||
|  | 				}, | ||||||
|  | 			}) | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if fi.Mode()&os.ModeSymlink != 0 { | ||||||
|  | 			link, err := os.Readlink(path) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			link = strings.TrimPrefix(link, dirs.PkgDir) | ||||||
|  |  | ||||||
|  | 			contents = append(contents, &files.Content{ | ||||||
|  | 				Source:      link, | ||||||
|  | 				Destination: trimmed, | ||||||
|  | 				Type:        "symlink", | ||||||
|  | 				FileInfo: &files.ContentFileInfo{ | ||||||
|  | 					MTime: fi.ModTime(), | ||||||
|  | 					Mode:  fi.Mode(), | ||||||
|  | 				}, | ||||||
|  | 			}) | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		fileContent := &files.Content{ | ||||||
|  | 			Source:      path, | ||||||
|  | 			Destination: trimmed, | ||||||
|  | 			FileInfo: &files.ContentFileInfo{ | ||||||
|  | 				MTime: fi.ModTime(), | ||||||
|  | 				Mode:  fi.Mode(), | ||||||
|  | 				Size:  fi.Size(), | ||||||
|  | 			}, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if slices.Contains(vars.Backup, trimmed) { | ||||||
|  | 			fileContent.Type = "config|noreplace" | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		contents = append(contents, fileContent) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if preferedContents != nil { | ||||||
|  | 		for _, trimmed := range *preferedContents { | ||||||
|  | 			path := filepath.Join(dirs.PkgDir, trimmed) | ||||||
|  | 			if err := processPath(path, trimmed, true); err != nil { | ||||||
|  | 				return nil, err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		err := filepath.Walk(dirs.PkgDir, func(path string, fi os.FileInfo, err error) error { | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			trimmed := strings.TrimPrefix(path, dirs.PkgDir) | ||||||
|  | 			return processPath(path, trimmed, false) | ||||||
|  | 		}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return contents, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var RegexpALRPackageName = regexp.MustCompile(`^(?P<package>[^+]+)\+alr-(?P<repo>.+)$`) | ||||||
|  |  | ||||||
|  | func getBasePkgInfo(vars *types.BuildVars, info *distro.OSRelease, opts *types.BuildOpts) *nfpm.Info { | ||||||
|  | 	return &nfpm.Info{ | ||||||
|  | 		Name:    fmt.Sprintf("%s+alr-%s", vars.Name, opts.Repository), | ||||||
|  | 		Arch:    cpu.Arch(), | ||||||
|  | 		Version: vars.Version, | ||||||
|  | 		Release: overrides.ReleasePlatformSpecific(vars.Release, info), | ||||||
|  | 		Epoch:   strconv.FormatUint(uint64(vars.Epoch), 10), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция getPkgFormat возвращает формат пакета из менеджера пакетов, | ||||||
|  | // или ALR_PKG_FORMAT, если он установлен. | ||||||
|  | func getPkgFormat(mgr manager.Manager) string { | ||||||
|  | 	pkgFormat := mgr.Format() | ||||||
|  | 	if format, ok := os.LookupEnv("ALR_PKG_FORMAT"); ok { | ||||||
|  | 		pkgFormat = format | ||||||
|  | 	} | ||||||
|  | 	return pkgFormat | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция createBuildEnvVars создает переменные окружения, которые будут установлены | ||||||
|  | // в скрипте сборки при его выполнении. | ||||||
|  | func createBuildEnvVars(info *distro.OSRelease, dirs types.Directories) []string { | ||||||
|  | 	env := os.Environ() | ||||||
|  |  | ||||||
|  | 	env = append( | ||||||
|  | 		env, | ||||||
|  | 		"DISTRO_NAME="+info.Name, | ||||||
|  | 		"DISTRO_PRETTY_NAME="+info.PrettyName, | ||||||
|  | 		"DISTRO_ID="+info.ID, | ||||||
|  | 		"DISTRO_VERSION_ID="+info.VersionID, | ||||||
|  | 		"DISTRO_ID_LIKE="+strings.Join(info.Like, " "), | ||||||
|  | 		"ARCH="+cpu.Arch(), | ||||||
|  | 		"NCPU="+strconv.Itoa(runtime.NumCPU()), | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	if dirs.ScriptDir != "" { | ||||||
|  | 		env = append(env, "scriptdir="+dirs.ScriptDir) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if dirs.PkgDir != "" { | ||||||
|  | 		env = append(env, "pkgdir="+dirs.PkgDir) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if dirs.SrcDir != "" { | ||||||
|  | 		env = append(env, "srcdir="+dirs.SrcDir) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return env | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция setScripts добавляет скрипты-перехватчики к метаданным пакета. | ||||||
|  | func setScripts(vars *types.BuildVars, info *nfpm.Info, scriptDir string) { | ||||||
|  | 	if vars.Scripts.PreInstall != "" { | ||||||
|  | 		info.Scripts.PreInstall = filepath.Join(scriptDir, vars.Scripts.PreInstall) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PostInstall != "" { | ||||||
|  | 		info.Scripts.PostInstall = filepath.Join(scriptDir, vars.Scripts.PostInstall) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PreRemove != "" { | ||||||
|  | 		info.Scripts.PreRemove = filepath.Join(scriptDir, vars.Scripts.PreRemove) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PostRemove != "" { | ||||||
|  | 		info.Scripts.PostRemove = filepath.Join(scriptDir, vars.Scripts.PostRemove) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PreUpgrade != "" { | ||||||
|  | 		info.ArchLinux.Scripts.PreUpgrade = filepath.Join(scriptDir, vars.Scripts.PreUpgrade) | ||||||
|  | 		info.APK.Scripts.PreUpgrade = filepath.Join(scriptDir, vars.Scripts.PreUpgrade) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PostUpgrade != "" { | ||||||
|  | 		info.ArchLinux.Scripts.PostUpgrade = filepath.Join(scriptDir, vars.Scripts.PostUpgrade) | ||||||
|  | 		info.APK.Scripts.PostUpgrade = filepath.Join(scriptDir, vars.Scripts.PostUpgrade) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PreTrans != "" { | ||||||
|  | 		info.RPM.Scripts.PreTrans = filepath.Join(scriptDir, vars.Scripts.PreTrans) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if vars.Scripts.PostTrans != "" { | ||||||
|  | 		info.RPM.Scripts.PostTrans = filepath.Join(scriptDir, vars.Scripts.PostTrans) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | // Функция setVersion изменяет переменную версии в скрипте runner. | ||||||
|  | // Она используется для установки версии на вывод функции version(). | ||||||
|  | func setVersion(ctx context.Context, r *interp.Runner, to string) error { | ||||||
|  | 	fl, err := syntax.NewParser().Parse(strings.NewReader("version='"+to+"'"), "") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return r.Run(ctx, fl) | ||||||
|  | } | ||||||
|  | */ | ||||||
|  | // Returns not installed dependencies | ||||||
|  | func removeAlreadyInstalled(opts types.BuildOpts, dependencies []string) ([]string, error) { | ||||||
|  | 	filteredPackages := []string{} | ||||||
|  |  | ||||||
|  | 	for _, dep := range dependencies { | ||||||
|  | 		installed, err := opts.Manager.IsInstalled(dep) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		if installed { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		filteredPackages = append(filteredPackages, dep) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return filteredPackages, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция packageNames возвращает имена всех предоставленных пакетов. | ||||||
|  | func packageNames(pkgs []db.Package) []string { | ||||||
|  | 	names := make([]string, len(pkgs)) | ||||||
|  | 	for i, p := range pkgs { | ||||||
|  | 		names[i] = p.Name | ||||||
|  | 	} | ||||||
|  | 	return names | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Функция removeDuplicates убирает любые дубликаты из предоставленного среза. | ||||||
|  | func removeDuplicates(slice []string) []string { | ||||||
|  | 	seen := map[string]struct{}{} | ||||||
|  | 	result := []string{} | ||||||
|  |  | ||||||
|  | 	for _, s := range slice { | ||||||
|  | 		if _, ok := seen[s]; !ok { | ||||||
|  | 			seen[s] = struct{}{} | ||||||
|  | 			result = append(result, s) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return result | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func removeDuplicatesSources(sources, checksums []string) ([]string, []string) { | ||||||
|  | 	seen := map[string]string{} | ||||||
|  | 	keys := make([]string, 0) | ||||||
|  | 	for i, s := range sources { | ||||||
|  | 		if val, ok := seen[s]; !ok || strings.EqualFold(val, "SKIP") { | ||||||
|  | 			if !ok { | ||||||
|  | 				keys = append(keys, s) | ||||||
|  | 			} | ||||||
|  | 			seen[s] = checksums[i] | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	newSources := make([]string, len(keys)) | ||||||
|  | 	newChecksums := make([]string, len(keys)) | ||||||
|  | 	for i, k := range keys { | ||||||
|  | 		newSources[i] = k | ||||||
|  | 		newChecksums[i] = seen[k] | ||||||
|  | 	} | ||||||
|  | 	return newSources, newChecksums | ||||||
|  | } | ||||||
							
								
								
									
										47
									
								
								pkg/build/utils_internal_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								pkg/build/utils_internal_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 build | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestRemoveDuplicatesSources(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		Name         string | ||||||
|  | 		Sources      []string | ||||||
|  | 		Checksums    []string | ||||||
|  | 		NewSources   []string | ||||||
|  | 		NewChecksums []string | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{{ | ||||||
|  | 		Name:         "prefer non-skip values", | ||||||
|  | 		Sources:      []string{"a", "b", "c", "a"}, | ||||||
|  | 		Checksums:    []string{"skip", "skip", "skip", "1"}, | ||||||
|  | 		NewSources:   []string{"a", "b", "c"}, | ||||||
|  | 		NewChecksums: []string{"1", "skip", "skip"}, | ||||||
|  | 	}} { | ||||||
|  | 		t.Run(tc.Name, func(t *testing.T) { | ||||||
|  | 			s, c := removeDuplicatesSources(tc.Sources, tc.Checksums) | ||||||
|  | 			assert.Equal(t, s, tc.NewSources) | ||||||
|  | 			assert.Equal(t, c, tc.NewChecksums) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -79,7 +79,7 @@ func ParseOSRelease(ctx context.Context) (*OSRelease, error) { | |||||||
| 	runner, err := interp.New( | 	runner, err := interp.New( | ||||||
| 		interp.OpenHandler(handlers.NopOpen), | 		interp.OpenHandler(handlers.NopOpen), | ||||||
| 		interp.ExecHandler(handlers.NopExec), | 		interp.ExecHandler(handlers.NopExec), | ||||||
| 		interp.ReadDirHandler(handlers.NopReadDir), | 		interp.ReadDirHandler2(handlers.NopReadDir), | ||||||
| 		interp.StatHandler(handlers.NopStat), | 		interp.StatHandler(handlers.NopStat), | ||||||
| 		interp.Env(expand.ListEnviron()), | 		interp.Env(expand.ListEnviron()), | ||||||
| 	) | 	) | ||||||
|   | |||||||
| @@ -32,19 +32,19 @@ deps=("python3") | |||||||
| deps_arch=("python") | deps_arch=("python") | ||||||
| deps_alpine=("python3") | deps_alpine=("python3") | ||||||
|  |  | ||||||
| build_deps=("python3" "python3-setuptools") | build_deps=("python3" "python3-pip") | ||||||
| build_deps_arch=("python" "python-setuptools") | build_deps_arch=("python" "python-pip") | ||||||
| build_deps_alpine=("python3" "py3-setuptools") | build_deps_alpine=("python3" "py3-pip") | ||||||
|  |  | ||||||
| sources=("https://files.pythonhosted.org/packages/source/{{.SourceURL.Filename | firstchar}}/{{.Info.Name}}/{{.SourceURL.Filename}}") | sources=("https://files.pythonhosted.org/packages/source/{{.SourceURL.Filename | firstchar}}/{{.Info.Name}}/{{.SourceURL.Filename}}") | ||||||
| checksums=('blake2b-256:{{.SourceURL.Digests.blake2b_256}}') | checksums=('blake2b-256:{{.SourceURL.Digests.blake2b_256}}') | ||||||
|  |  | ||||||
| build() { | build() { | ||||||
| 	cd "$srcdir/{{.Info.Name}}-${version}" | 	cd "$srcdir/{{.Info.Name}}-${version}" | ||||||
| 	python3 setup.py build |   python3 -m build | ||||||
| } | } | ||||||
|  |  | ||||||
| package() { | package() { | ||||||
| 	cd "$srcdir/{{.Info.Name}}-${version}" | 	cd "$srcdir/{{.Info.Name}}-${version}" | ||||||
| 	python3 setup.py install --root="${pkgdir}/" --optimize=1 || return 1 | 	pip install --root="${pkgdir}/" . --no-deps --disable-pip-version-check | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,8 @@ package repos | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
| 	"log/slog" | 	"log/slog" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"os" | 	"os" | ||||||
| @@ -41,8 +43,10 @@ import ( | |||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/shutils/decoder" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/shutils/handlers" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/shutils/handlers" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type actionType uint8 | type actionType uint8 | ||||||
| @@ -63,7 +67,7 @@ type action struct { | |||||||
| // If repos is set to nil, the repos in the ALR config will be used. | // If repos is set to nil, the repos in the ALR config will be used. | ||||||
| func (rs *Repos) Pull(ctx context.Context, repos []types.Repo) error { | func (rs *Repos) Pull(ctx context.Context, repos []types.Repo) error { | ||||||
| 	if repos == nil { | 	if repos == nil { | ||||||
| 		repos = rs.cfg.Repos(ctx) | 		repos = rs.cfg.Repos() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, repo := range repos { | 	for _, repo := range repos { | ||||||
| @@ -73,7 +77,7 @@ func (rs *Repos) Pull(ctx context.Context, repos []types.Repo) error { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		slog.Info(gotext.Get("Pulling repository"), "name", repo.Name) | 		slog.Info(gotext.Get("Pulling repository"), "name", repo.Name) | ||||||
| 		repoDir := filepath.Join(config.GetPaths(ctx).RepoDir, repo.Name) | 		repoDir := filepath.Join(rs.cfg.GetPaths().RepoDir, repo.Name) | ||||||
|  |  | ||||||
| 		var repoFS billy.Filesystem | 		var repoFS billy.Filesystem | ||||||
| 		gitDir := filepath.Join(repoDir, ".git") | 		gitDir := filepath.Join(repoDir, ".git") | ||||||
| @@ -177,6 +181,96 @@ func (rs *Repos) Pull(ctx context.Context, repos []types.Repo) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (rs *Repos) updatePkg(ctx context.Context, repo types.Repo, runner *interp.Runner, scriptFl io.ReadCloser) error { | ||||||
|  | 	parser := syntax.NewParser() | ||||||
|  |  | ||||||
|  | 	defer scriptFl.Close() | ||||||
|  | 	fl, err := parser.Parse(scriptFl, "alr.sh") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	runner.Reset() | ||||||
|  | 	err = runner.Run(ctx, fl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	type packages struct { | ||||||
|  | 		BasePkgName string   `sh:"basepkg_name"` | ||||||
|  | 		Names       []string `sh:"name"` | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var pkgs packages | ||||||
|  |  | ||||||
|  | 	d := decoder.New(&distro.OSRelease{}, runner) | ||||||
|  | 	d.Overrides = false | ||||||
|  | 	d.LikeDistros = false | ||||||
|  | 	err = d.DecodeVars(&pkgs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(pkgs.Names) > 1 { | ||||||
|  | 		if pkgs.BasePkgName == "" { | ||||||
|  | 			pkgs.BasePkgName = pkgs.Names[0] | ||||||
|  | 		} | ||||||
|  | 		for _, pkgName := range pkgs.Names { | ||||||
|  | 			pkgInfo := PackageInfo{} | ||||||
|  | 			funcName := fmt.Sprintf("meta_%s", pkgName) | ||||||
|  | 			runner.Reset() | ||||||
|  | 			err = runner.Run(ctx, fl) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			meta, ok := d.GetFuncWithSubshell(funcName) | ||||||
|  | 			if !ok { | ||||||
|  | 				return errors.New("func is missing") | ||||||
|  | 			} | ||||||
|  | 			r, err := meta(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			d := decoder.New(&distro.OSRelease{}, r) | ||||||
|  | 			d.Overrides = false | ||||||
|  | 			d.LikeDistros = false | ||||||
|  | 			err = d.DecodeVars(&pkgInfo) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			pkg := pkgInfo.ToPackage(repo.Name) | ||||||
|  | 			resolveOverrides(r, pkg) | ||||||
|  | 			pkg.Name = pkgName | ||||||
|  | 			pkg.BasePkgName = pkgs.BasePkgName | ||||||
|  | 			err = rs.db.InsertPackage(ctx, *pkg) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	pkg := EmptyPackage(repo.Name) | ||||||
|  | 	err = d.DecodeVars(pkg) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	resolveOverrides(runner, pkg) | ||||||
|  | 	return rs.db.InsertPackage(ctx, *pkg) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (rs *Repos) processRepoChangesRunner(repoDir, scriptDir string) (*interp.Runner, error) { | ||||||
|  | 	env := append(os.Environ(), "scriptdir="+scriptDir) | ||||||
|  | 	return interp.New( | ||||||
|  | 		interp.Env(expand.ListEnviron(env...)), | ||||||
|  | 		interp.ExecHandler(handlers.NopExec), | ||||||
|  | 		interp.ReadDirHandler2(handlers.RestrictedReadDir(repoDir)), | ||||||
|  | 		interp.StatHandler(handlers.RestrictedStat(repoDir)), | ||||||
|  | 		interp.OpenHandler(handlers.RestrictedOpen(repoDir)), | ||||||
|  | 		interp.StdIO(handlers.NopRWC{}, handlers.NopRWC{}, handlers.NopRWC{}), | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (rs *Repos) processRepoChanges(ctx context.Context, repo types.Repo, r *git.Repository, w *git.Worktree, old, new *plumbing.Reference) error { | func (rs *Repos) processRepoChanges(ctx context.Context, repo types.Repo, r *git.Repository, w *git.Worktree, old, new *plumbing.Reference) error { | ||||||
| 	oldCommit, err := r.CommitObject(old.Hash()) | 	oldCommit, err := r.CommitObject(old.Hash()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -201,18 +295,18 @@ func (rs *Repos) processRepoChanges(ctx context.Context, repo types.Repo, r *git | |||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if to == nil { | 		switch { | ||||||
|  | 		case to == nil: | ||||||
| 			actions = append(actions, action{ | 			actions = append(actions, action{ | ||||||
| 				Type: actionDelete, | 				Type: actionDelete, | ||||||
| 				File: from.Path(), | 				File: from.Path(), | ||||||
| 			}) | 			}) | ||||||
| 		} else if from == nil { | 		case from == nil: | ||||||
| 			actions = append(actions, action{ | 			actions = append(actions, action{ | ||||||
| 				Type: actionUpdate, | 				Type: actionUpdate, | ||||||
| 				File: to.Path(), | 				File: to.Path(), | ||||||
| 			}) | 			}) | ||||||
| 		} else { | 		case from.Path() != to.Path(): | ||||||
| 			if from.Path() != to.Path() { |  | ||||||
| 			actions = append(actions, | 			actions = append(actions, | ||||||
| 				action{ | 				action{ | ||||||
| 					Type: actionDelete, | 					Type: actionDelete, | ||||||
| @@ -223,28 +317,19 @@ func (rs *Repos) processRepoChanges(ctx context.Context, repo types.Repo, r *git | |||||||
| 					File: to.Path(), | 					File: to.Path(), | ||||||
| 				}, | 				}, | ||||||
| 			) | 			) | ||||||
| 			} else { | 		default: | ||||||
| 			actions = append(actions, action{ | 			actions = append(actions, action{ | ||||||
| 				Type: actionUpdate, | 				Type: actionUpdate, | ||||||
| 				File: to.Path(), | 				File: to.Path(), | ||||||
| 			}) | 			}) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	repoDir := w.Filesystem.Root() | 	repoDir := w.Filesystem.Root() | ||||||
| 	parser := syntax.NewParser() | 	parser := syntax.NewParser() | ||||||
|  |  | ||||||
| 	for _, action := range actions { | 	for _, action := range actions { | ||||||
| 		env := append(os.Environ(), "scriptdir="+filepath.Dir(filepath.Join(repoDir, action.File))) | 		runner, err := rs.processRepoChangesRunner(repoDir, filepath.Dir(filepath.Join(repoDir, action.File))) | ||||||
| 		runner, err := interp.New( |  | ||||||
| 			interp.Env(expand.ListEnviron(env...)), |  | ||||||
| 			interp.ExecHandler(handlers.NopExec), |  | ||||||
| 			interp.ReadDirHandler(handlers.RestrictedReadDir(repoDir)), |  | ||||||
| 			interp.StatHandler(handlers.RestrictedStat(repoDir)), |  | ||||||
| 			interp.OpenHandler(handlers.RestrictedOpen(repoDir)), |  | ||||||
| 			interp.StdIO(handlers.NopRWC{}, handlers.NopRWC{}, handlers.NopRWC{}), |  | ||||||
| 		) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| @@ -290,23 +375,7 @@ func (rs *Repos) processRepoChanges(ctx context.Context, repo types.Repo, r *git | |||||||
| 				return nil | 				return nil | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			pkg := db.Package{ | 			err = rs.updatePkg(ctx, repo, runner, r) | ||||||
| 				Description:  db.NewJSON(map[string]string{}), |  | ||||||
| 				Homepage:     db.NewJSON(map[string]string{}), |  | ||||||
| 				Maintainer:   db.NewJSON(map[string]string{}), |  | ||||||
| 				Depends:      db.NewJSON(map[string][]string{}), |  | ||||||
| 				BuildDepends: db.NewJSON(map[string][]string{}), |  | ||||||
| 				Repository:   repo.Name, |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			err = parseScript(ctx, parser, runner, r, &pkg) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			resolveOverrides(runner, &pkg) |  | ||||||
|  |  | ||||||
| 			err = rs.db.InsertPackage(ctx, pkg) |  | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| @@ -323,18 +392,8 @@ func (rs *Repos) processRepoFull(ctx context.Context, repo types.Repo, repoDir s | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	parser := syntax.NewParser() |  | ||||||
|  |  | ||||||
| 	for _, match := range matches { | 	for _, match := range matches { | ||||||
| 		env := append(os.Environ(), "scriptdir="+filepath.Dir(match)) | 		runner, err := rs.processRepoChangesRunner(repoDir, filepath.Dir(match)) | ||||||
| 		runner, err := interp.New( |  | ||||||
| 			interp.Env(expand.ListEnviron(env...)), |  | ||||||
| 			interp.ExecHandler(handlers.NopExec), |  | ||||||
| 			interp.ReadDirHandler(handlers.RestrictedReadDir(repoDir)), |  | ||||||
| 			interp.StatHandler(handlers.RestrictedStat(repoDir)), |  | ||||||
| 			interp.OpenHandler(handlers.RestrictedOpen(repoDir)), |  | ||||||
| 			interp.StdIO(handlers.NopRWC{}, handlers.NopRWC{}, handlers.NopRWC{}), |  | ||||||
| 		) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| @@ -344,23 +403,7 @@ func (rs *Repos) processRepoFull(ctx context.Context, repo types.Repo, repoDir s | |||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		pkg := db.Package{ | 		err = rs.updatePkg(ctx, repo, runner, scriptFl) | ||||||
| 			Description:  db.NewJSON(map[string]string{}), |  | ||||||
| 			Homepage:     db.NewJSON(map[string]string{}), |  | ||||||
| 			Maintainer:   db.NewJSON(map[string]string{}), |  | ||||||
| 			Depends:      db.NewJSON(map[string][]string{}), |  | ||||||
| 			BuildDepends: db.NewJSON(map[string][]string{}), |  | ||||||
| 			Repository:   repo.Name, |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		err = parseScript(ctx, parser, runner, scriptFl, &pkg) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		resolveOverrides(runner, &pkg) |  | ||||||
|  |  | ||||||
| 		err = rs.db.InsertPackage(ctx, pkg) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|   | |||||||
							
								
								
									
										173
									
								
								pkg/repos/pull_internal_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								pkg/repos/pull_internal_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 repos | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"io" | ||||||
|  | 	"os" | ||||||
|  | 	"strings" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type TestALRConfig struct{} | ||||||
|  |  | ||||||
|  | func (c *TestALRConfig) GetPaths() *config.Paths { | ||||||
|  | 	return &config.Paths{ | ||||||
|  | 		DBPath: ":memory:", | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (c *TestALRConfig) Repos() []types.Repo { | ||||||
|  | 	return []types.Repo{ | ||||||
|  | 		{ | ||||||
|  | 			Name: "test", | ||||||
|  | 			URL:  "https://test", | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func createReadCloserFromString(input string) io.ReadCloser { | ||||||
|  | 	reader := strings.NewReader(input) | ||||||
|  | 	return struct { | ||||||
|  | 		io.Reader | ||||||
|  | 		io.Closer | ||||||
|  | 	}{ | ||||||
|  | 		Reader: reader, | ||||||
|  | 		Closer: io.NopCloser(reader), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestUpdatePkg(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		name   string | ||||||
|  | 		file   string | ||||||
|  | 		verify func(context.Context, *db.Database) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	repo := types.Repo{ | ||||||
|  | 		Name: "test", | ||||||
|  | 		URL:  "https://test", | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{ | ||||||
|  | 		{ | ||||||
|  | 			name: "single package", | ||||||
|  | 			file: `name=foo | ||||||
|  | version='0.0.1' | ||||||
|  | release=1 | ||||||
|  | desc="main desc" | ||||||
|  | deps=('sudo') | ||||||
|  | build_deps=('golang') | ||||||
|  | `, | ||||||
|  | 			verify: func(ctx context.Context, database *db.Database) { | ||||||
|  | 				result, err := database.GetPkgs(ctx, "1 = 1") | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  | 				pkgCount := 0 | ||||||
|  | 				for result.Next() { | ||||||
|  | 					var dbPkg db.Package | ||||||
|  | 					err = result.StructScan(&dbPkg) | ||||||
|  | 					if err != nil { | ||||||
|  | 						t.Errorf("Expected no error, got %s", err) | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					assert.Equal(t, "foo", dbPkg.Name) | ||||||
|  | 					assert.Equal(t, db.NewJSON(map[string]string{"": "main desc"}), dbPkg.Description) | ||||||
|  | 					assert.Equal(t, db.NewJSON(map[string][]string{"": {"sudo"}}), dbPkg.Depends) | ||||||
|  | 					pkgCount++ | ||||||
|  | 				} | ||||||
|  | 				assert.Equal(t, 1, pkgCount) | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name: "multiple package", | ||||||
|  | 			file: `basepkg_name=foo | ||||||
|  | name=( | ||||||
|  | 	bar | ||||||
|  | 	buz | ||||||
|  | ) | ||||||
|  | version='0.0.1' | ||||||
|  | release=1 | ||||||
|  | desc="main desc" | ||||||
|  | deps=('sudo') | ||||||
|  | build_deps=('golang') | ||||||
|  | 		 | ||||||
|  | meta_bar() { | ||||||
|  | 	desc="foo desc" | ||||||
|  | } | ||||||
|  | 			 | ||||||
|  | meta_buz() { | ||||||
|  | 	deps+=('doas') | ||||||
|  | } | ||||||
|  | `, | ||||||
|  | 			verify: func(ctx context.Context, database *db.Database) { | ||||||
|  | 				result, err := database.GetPkgs(ctx, "1 = 1") | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 				pkgCount := 0 | ||||||
|  | 				for result.Next() { | ||||||
|  | 					var dbPkg db.Package | ||||||
|  | 					err = result.StructScan(&dbPkg) | ||||||
|  | 					if err != nil { | ||||||
|  | 						t.Errorf("Expected no error, got %s", err) | ||||||
|  | 					} | ||||||
|  | 					if dbPkg.Name == "bar" { | ||||||
|  | 						assert.Equal(t, db.NewJSON(map[string]string{"": "foo desc"}), dbPkg.Description) | ||||||
|  | 						assert.Equal(t, db.NewJSON(map[string][]string{"": {"sudo"}}), dbPkg.Depends) | ||||||
|  | 					} | ||||||
|  |  | ||||||
|  | 					if dbPkg.Name == "buz" { | ||||||
|  | 						assert.Equal(t, db.NewJSON(map[string]string{"": "main desc"}), dbPkg.Description) | ||||||
|  | 						assert.Equal(t, db.NewJSON(map[string][]string{"": {"sudo", "doas"}}), dbPkg.Depends) | ||||||
|  | 					} | ||||||
|  | 					pkgCount++ | ||||||
|  | 				} | ||||||
|  | 				assert.Equal(t, 2, pkgCount) | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} { | ||||||
|  | 		t.Run(tc.name, func(t *testing.T) { | ||||||
|  | 			cfg := &TestALRConfig{} | ||||||
|  | 			ctx := context.Background() | ||||||
|  |  | ||||||
|  | 			database := db.New(&TestALRConfig{}) | ||||||
|  | 			database.Init(ctx) | ||||||
|  |  | ||||||
|  | 			rs := New(cfg, database) | ||||||
|  |  | ||||||
|  | 			path, err := os.MkdirTemp("", "test-update-pkg") | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			defer os.RemoveAll(path) | ||||||
|  |  | ||||||
|  | 			runner, err := rs.processRepoChangesRunner(path, path) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			err = rs.updatePkg(ctx, repo, runner, createReadCloserFromString( | ||||||
|  | 				tc.file, | ||||||
|  | 			)) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 			tc.verify(ctx, database) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -44,7 +44,7 @@ type TestALRConfig struct { | |||||||
| 	PkgsDir  string | 	PkgsDir  string | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *TestALRConfig) GetPaths(ctx context.Context) *config.Paths { | func (c *TestALRConfig) GetPaths() *config.Paths { | ||||||
| 	return &config.Paths{ | 	return &config.Paths{ | ||||||
| 		DBPath:   ":memory:", | 		DBPath:   ":memory:", | ||||||
| 		CacheDir: c.CacheDir, | 		CacheDir: c.CacheDir, | ||||||
| @@ -53,7 +53,7 @@ func (c *TestALRConfig) GetPaths(ctx context.Context) *config.Paths { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (c *TestALRConfig) Repos(ctx context.Context) []types.Repo { | func (c *TestALRConfig) Repos() []types.Repo { | ||||||
| 	return []types.Repo{} | 	return []types.Repo{} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,16 +17,14 @@ | |||||||
| package repos | package repos | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Config interface { | type Config interface { | ||||||
| 	GetPaths(ctx context.Context) *config.Paths | 	GetPaths() *config.Paths | ||||||
| 	Repos(ctx context.Context) []types.Repo | 	Repos() []types.Repo | ||||||
| } | } | ||||||
|  |  | ||||||
| type Repos struct { | type Repos struct { | ||||||
|   | |||||||
| @@ -1,70 +0,0 @@ | |||||||
| // ALR - Any Linux Repository |  | ||||||
| // Copyright (C) 2025 Евгений Храмов |  | ||||||
| // |  | ||||||
| // 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 repos |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"sync" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" |  | ||||||
| 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Pull pulls the provided repositories. If a repo doesn't exist, it will be cloned |  | ||||||
| // and its packages will be written to the DB. If it does exist, it will be pulled. |  | ||||||
| // In this case, only changed packages will be processed if possible. |  | ||||||
| // If repos is set to nil, the repos in the ALR config will be used. |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func Pull(ctx context.Context, repos []types.Repo) error { |  | ||||||
| 	return GetInstance(ctx).Pull(ctx, repos) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // FindPkgs looks for packages matching the inputs inside the database. |  | ||||||
| // It returns a map that maps the package name input to any packages found for it. |  | ||||||
| // It also returns a slice that contains the names of all packages that were not found. |  | ||||||
| // |  | ||||||
| // Deprecated: use struct method |  | ||||||
| func FindPkgs(ctx context.Context, pkgs []string) (map[string][]db.Package, []string, error) { |  | ||||||
| 	return GetInstance(ctx).FindPkgs(ctx, pkgs) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // ======================= |  | ||||||
| // FOR LEGACY ONLY |  | ||||||
| // ======================= |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	reposInstance *Repos |  | ||||||
| 	alrConfigOnce sync.Once |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Deprecated: For legacy only |  | ||||||
| func GetInstance(ctx context.Context) *Repos { |  | ||||||
| 	alrConfigOnce.Do(func() { |  | ||||||
| 		cfg := config.GetInstance(ctx) |  | ||||||
| 		db := database.GetInstance(ctx) |  | ||||||
|  |  | ||||||
| 		reposInstance = New( |  | ||||||
| 			cfg, |  | ||||||
| 			db, |  | ||||||
| 		) |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	return reposInstance |  | ||||||
| } |  | ||||||
| @@ -67,6 +67,47 @@ func parseScript(ctx context.Context, parser *syntax.Parser, runner *interp.Runn | |||||||
| 	return d.DecodeVars(pkg) | 	return d.DecodeVars(pkg) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type PackageInfo struct { | ||||||
|  | 	Version       string            `sh:"version,required"` | ||||||
|  | 	Release       int               `sh:"release,required"` | ||||||
|  | 	Epoch         uint              `sh:"epoch"` | ||||||
|  | 	Architectures db.JSON[[]string] `sh:"architectures"` | ||||||
|  | 	Licenses      db.JSON[[]string] `sh:"license"` | ||||||
|  | 	Provides      db.JSON[[]string] `sh:"provides"` | ||||||
|  | 	Conflicts     db.JSON[[]string] `sh:"conflicts"` | ||||||
|  | 	Replaces      db.JSON[[]string] `sh:"replaces"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (inf *PackageInfo) ToPackage(repoName string) *db.Package { | ||||||
|  | 	return &db.Package{ | ||||||
|  | 		Version:       inf.Version, | ||||||
|  | 		Release:       inf.Release, | ||||||
|  | 		Epoch:         inf.Epoch, | ||||||
|  | 		Architectures: inf.Architectures, | ||||||
|  | 		Licenses:      inf.Licenses, | ||||||
|  | 		Provides:      inf.Provides, | ||||||
|  | 		Conflicts:     inf.Conflicts, | ||||||
|  | 		Replaces:      inf.Replaces, | ||||||
|  | 		Description:   db.NewJSON(map[string]string{}), | ||||||
|  | 		Homepage:      db.NewJSON(map[string]string{}), | ||||||
|  | 		Maintainer:    db.NewJSON(map[string]string{}), | ||||||
|  | 		Depends:       db.NewJSON(map[string][]string{}), | ||||||
|  | 		BuildDepends:  db.NewJSON(map[string][]string{}), | ||||||
|  | 		Repository:    repoName, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func EmptyPackage(repoName string) *db.Package { | ||||||
|  | 	return &db.Package{ | ||||||
|  | 		Description:  db.NewJSON(map[string]string{}), | ||||||
|  | 		Homepage:     db.NewJSON(map[string]string{}), | ||||||
|  | 		Maintainer:   db.NewJSON(map[string]string{}), | ||||||
|  | 		Depends:      db.NewJSON(map[string][]string{}), | ||||||
|  | 		BuildDepends: db.NewJSON(map[string][]string{}), | ||||||
|  | 		Repository:   repoName, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| var overridable = map[string]string{ | var overridable = map[string]string{ | ||||||
| 	"deps":       "Depends", | 	"deps":       "Depends", | ||||||
| 	"build_deps": "BuildDepends", | 	"build_deps": "BuildDepends", | ||||||
|   | |||||||
| @@ -21,166 +21,46 @@ package search | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"errors" |  | ||||||
| 	"io" |  | ||||||
| 	"io/fs" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"github.com/jmoiron/sqlx" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" |  | ||||||
|  | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Filter represents search filters. | type PackagesProvider interface { | ||||||
| type Filter int | 	GetPkgs(ctx context.Context, where string, args ...any) (*sqlx.Rows, error) | ||||||
|  |  | ||||||
| // Filters |  | ||||||
| const ( |  | ||||||
| 	FilterNone Filter = iota |  | ||||||
| 	FilterInRepo |  | ||||||
| 	FilterSupportsArch |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // SoryBy represents a value that packages can be sorted by. |  | ||||||
| type SortBy int |  | ||||||
|  |  | ||||||
| // Sort values |  | ||||||
| const ( |  | ||||||
| 	SortByNone = iota |  | ||||||
| 	SortByName |  | ||||||
| 	SortByRepo |  | ||||||
| 	SortByVersion |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // Package represents a package from ALR's database |  | ||||||
| type Package struct { |  | ||||||
| 	Name          string |  | ||||||
| 	Version       string |  | ||||||
| 	Release       int |  | ||||||
| 	Epoch         uint |  | ||||||
| 	Description   map[string]string |  | ||||||
| 	Homepage      map[string]string |  | ||||||
| 	Maintainer    map[string]string |  | ||||||
| 	Architectures []string |  | ||||||
| 	Licenses      []string |  | ||||||
| 	Provides      []string |  | ||||||
| 	Conflicts     []string |  | ||||||
| 	Replaces      []string |  | ||||||
| 	Depends       map[string][]string |  | ||||||
| 	BuildDepends  map[string][]string |  | ||||||
| 	OptDepends    map[string][]string |  | ||||||
| 	Repository    string |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func convertPkg(p db.Package) Package { | type Searcher struct { | ||||||
| 	return Package{ | 	pp PackagesProvider | ||||||
| 		Name:          p.Name, | } | ||||||
| 		Version:       p.Version, |  | ||||||
| 		Release:       p.Release, | func New(pp PackagesProvider) *Searcher { | ||||||
| 		Epoch:         p.Epoch, | 	return &Searcher{ | ||||||
| 		Description:   p.Description.Val, | 		pp: pp, | ||||||
| 		Homepage:      p.Homepage.Val, |  | ||||||
| 		Maintainer:    p.Maintainer.Val, |  | ||||||
| 		Architectures: p.Architectures.Val, |  | ||||||
| 		Licenses:      p.Licenses.Val, |  | ||||||
| 		Provides:      p.Provides.Val, |  | ||||||
| 		Conflicts:     p.Conflicts.Val, |  | ||||||
| 		Replaces:      p.Replaces.Val, |  | ||||||
| 		Depends:       p.Depends.Val, |  | ||||||
| 		BuildDepends:  p.BuildDepends.Val, |  | ||||||
| 		OptDepends:    p.OptDepends.Val, |  | ||||||
| 		Repository:    p.Repository, |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // Options contains the options for a search. | func (s *Searcher) Search( | ||||||
| type Options struct { | 	ctx context.Context, | ||||||
| 	Filter      Filter | 	opts *SearchOptions, | ||||||
| 	FilterValue string | ) ([]database.Package, error) { | ||||||
| 	SortBy      SortBy | 	var packages []database.Package | ||||||
| 	Limit       int64 |  | ||||||
| 	Query       string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Search searches for packages in the database based on the given options. | 	where, args := opts.WhereClause() | ||||||
| func Search(ctx context.Context, opts Options) ([]Package, error) { | 	result, err := s.pp.GetPkgs(ctx, where, args...) | ||||||
| 	query := "(name LIKE ? OR description LIKE ? OR json_array_contains(provides, ?))" |  | ||||||
| 	args := []any{"%" + opts.Query + "%", "%" + opts.Query + "%", opts.Query} |  | ||||||
|  |  | ||||||
| 	if opts.Filter != FilterNone { |  | ||||||
| 		switch opts.Filter { |  | ||||||
| 		case FilterInRepo: |  | ||||||
| 			query += " AND repository = ?" |  | ||||||
| 		case FilterSupportsArch: |  | ||||||
| 			query += " AND json_array_contains(architectures, ?)" |  | ||||||
| 		} |  | ||||||
| 		args = append(args, opts.FilterValue) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if opts.SortBy != SortByNone { |  | ||||||
| 		switch opts.SortBy { |  | ||||||
| 		case SortByName: |  | ||||||
| 			query += " ORDER BY name" |  | ||||||
| 		case SortByRepo: |  | ||||||
| 			query += " ORDER BY repository" |  | ||||||
| 		case SortByVersion: |  | ||||||
| 			query += " ORDER BY version" |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if opts.Limit != 0 { |  | ||||||
| 		query += " LIMIT " + strconv.FormatInt(opts.Limit, 10) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	result, err := db.GetPkgs(ctx, query, args...) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	var out []Package |  | ||||||
| 	for result.Next() { | 	for result.Next() { | ||||||
| 		pkg := db.Package{} | 		var dbPkg database.Package | ||||||
| 		err = result.StructScan(&pkg) | 		err = result.StructScan(&dbPkg) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		out = append(out, convertPkg(pkg)) | 		packages = append(packages, dbPkg) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return out, err | 	return packages, nil | ||||||
| } |  | ||||||
|  |  | ||||||
| // GetPkg gets a single package from the database and returns it. |  | ||||||
| func GetPkg(ctx context.Context, repo, name string) (Package, error) { |  | ||||||
| 	pkg, err := db.GetPkg(ctx, "name = ? AND repository = ?", name, repo) |  | ||||||
| 	return convertPkg(*pkg), err |  | ||||||
| } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	// ErrInvalidArgument is an error returned by GetScript when one of its arguments |  | ||||||
| 	// contain invalid characters |  | ||||||
| 	ErrInvalidArgument = errors.New("name and repository must not contain . or /") |  | ||||||
|  |  | ||||||
| 	// ErrScriptNotFound is returned by GetScript if it can't find the script requested |  | ||||||
| 	// by the user. |  | ||||||
| 	ErrScriptNotFound = errors.New("requested script not found") |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| // GetScript returns a reader containing the build script for a given package. |  | ||||||
| func GetScript(ctx context.Context, repo, name string) (io.ReadCloser, error) { |  | ||||||
| 	if strings.Contains(name, "./") || strings.ContainsAny(repo, "./") { |  | ||||||
| 		return nil, ErrInvalidArgument |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	scriptPath := filepath.Join(config.GetPaths(ctx).RepoDir, repo, name, "alr.sh") |  | ||||||
| 	fl, err := os.Open(scriptPath) |  | ||||||
| 	if errors.Is(err, fs.ErrNotExist) { |  | ||||||
| 		return nil, ErrScriptNotFound |  | ||||||
| 	} else if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return fl, nil |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										86
									
								
								pkg/search/search_options_builder.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								pkg/search/search_options_builder.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 search | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type SearchOptions struct { | ||||||
|  | 	conditions []string | ||||||
|  | 	args       []any | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (o *SearchOptions) WhereClause() (string, []any) { | ||||||
|  | 	if len(o.conditions) == 0 { | ||||||
|  | 		return "", nil | ||||||
|  | 	} | ||||||
|  | 	return strings.Join(o.conditions, " AND "), o.args | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type SearchOptionsBuilder struct { | ||||||
|  | 	options SearchOptions | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewSearchOptions() *SearchOptionsBuilder { | ||||||
|  | 	return &SearchOptionsBuilder{} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) withGeneralLike(key, value string) *SearchOptionsBuilder { | ||||||
|  | 	if value != "" { | ||||||
|  | 		b.options.conditions = append(b.options.conditions, fmt.Sprintf("%s LIKE ?", key)) | ||||||
|  | 		b.options.args = append(b.options.args, "%"+value+"%") | ||||||
|  | 	} | ||||||
|  | 	return b | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) withGeneralEqual(key string, value any) *SearchOptionsBuilder { | ||||||
|  | 	if value != "" { | ||||||
|  | 		b.options.conditions = append(b.options.conditions, fmt.Sprintf("%s = ?", key)) | ||||||
|  | 		b.options.args = append(b.options.args, value) | ||||||
|  | 	} | ||||||
|  | 	return b | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) withGeneralJsonArrayContains(key string, value any) *SearchOptionsBuilder { | ||||||
|  | 	if value != "" { | ||||||
|  | 		b.options.conditions = append(b.options.conditions, fmt.Sprintf("json_array_contains(%s, ?)", key)) | ||||||
|  | 		b.options.args = append(b.options.args, value) | ||||||
|  | 	} | ||||||
|  | 	return b | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) WithName(name string) *SearchOptionsBuilder { | ||||||
|  | 	return b.withGeneralLike("name", name) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) WithDescription(description string) *SearchOptionsBuilder { | ||||||
|  | 	return b.withGeneralLike("description", description) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) WithRepository(repository string) *SearchOptionsBuilder { | ||||||
|  | 	return b.withGeneralEqual("repository", repository) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) WithProvides(provides string) *SearchOptionsBuilder { | ||||||
|  | 	return b.withGeneralJsonArrayContains("provides", provides) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *SearchOptionsBuilder) Build() *SearchOptions { | ||||||
|  | 	return &b.options | ||||||
|  | } | ||||||
							
								
								
									
										65
									
								
								pkg/search/search_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								pkg/search/search_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 search_test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/search" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestSearhOptionsBuilder(t *testing.T) { | ||||||
|  | 	type testCase struct { | ||||||
|  | 		name          string | ||||||
|  | 		prepare       func() *search.SearchOptions | ||||||
|  | 		expectedWhere string | ||||||
|  | 		expectedArgs  []any | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, tc := range []testCase{ | ||||||
|  | 		{ | ||||||
|  | 			name: "Empty fields", | ||||||
|  | 			prepare: func() *search.SearchOptions { | ||||||
|  | 				return search.NewSearchOptions(). | ||||||
|  | 					Build() | ||||||
|  | 			}, | ||||||
|  | 			expectedWhere: "", | ||||||
|  | 			expectedArgs:  []any{}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name: "All fields", | ||||||
|  | 			prepare: func() *search.SearchOptions { | ||||||
|  | 				return search.NewSearchOptions(). | ||||||
|  | 					WithName("foo"). | ||||||
|  | 					WithDescription("bar"). | ||||||
|  | 					WithRepository("buz"). | ||||||
|  | 					WithProvides("test"). | ||||||
|  | 					Build() | ||||||
|  | 			}, | ||||||
|  | 			expectedWhere: "name LIKE ? AND description LIKE ? AND repository = ? AND json_array_contains(provides, ?)", | ||||||
|  | 			expectedArgs:  []any{"%foo%", "%bar%", "buz", "test"}, | ||||||
|  | 		}, | ||||||
|  | 	} { | ||||||
|  | 		t.Run(tc.name, func(t *testing.T) { | ||||||
|  | 			whereClause, args := tc.prepare().WhereClause() | ||||||
|  | 			assert.Equal(t, tc.expectedWhere, whereClause) | ||||||
|  | 			assert.ElementsMatch(t, tc.expectedArgs, args) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										82
									
								
								repo.go
									
									
									
									
									
								
							
							
						
						
									
										82
									
								
								repo.go
									
									
									
									
									
								
							| @@ -25,12 +25,11 @@ import ( | |||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
|  |  | ||||||
| 	"github.com/leonelquinteros/gotext" | 	"github.com/leonelquinteros/gotext" | ||||||
| 	"github.com/pelletier/go-toml/v2" |  | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| 	"golang.org/x/exp/slices" | 	"golang.org/x/exp/slices" | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
| ) | ) | ||||||
| @@ -60,33 +59,42 @@ func AddRepoCmd() *cli.Command { | |||||||
| 			name := c.String("name") | 			name := c.String("name") | ||||||
| 			repoURL := c.String("url") | 			repoURL := c.String("url") | ||||||
|  |  | ||||||
| 			cfg := config.Config(ctx) | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			for _, repo := range cfg.Repos { | 			reposSlice := cfg.Repos() | ||||||
|  |  | ||||||
|  | 			for _, repo := range reposSlice { | ||||||
| 				if repo.URL == repoURL { | 				if repo.URL == repoURL { | ||||||
| 					slog.Error("Repo already exists", "name", repo.Name) | 					slog.Error("Repo already exists", "name", repo.Name) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			cfg.Repos = append(cfg.Repos, types.Repo{ | 			reposSlice = append(reposSlice, types.Repo{ | ||||||
| 				Name: name, | 				Name: name, | ||||||
| 				URL:  repoURL, | 				URL:  repoURL, | ||||||
| 			}) | 			}) | ||||||
|  | 			cfg.SetRepos(reposSlice) | ||||||
|  |  | ||||||
| 			cfgFl, err := os.Create(config.GetPaths(ctx).ConfigPath) | 			err = cfg.SaveUserConfig() | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error opening config file"), "err", err) | 				slog.Error(gotext.Get("Error saving config"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			err = toml.NewEncoder(cfgFl).Encode(cfg) | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error encoding config"), "err", err) | 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 				os.Exit(1) |  | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			err = repos.Pull(ctx, cfg.Repos) | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = rs.Pull(ctx, cfg.Repos()) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| @@ -114,11 +122,17 @@ func RemoveRepoCmd() *cli.Command { | |||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
|  |  | ||||||
| 			name := c.String("name") | 			name := c.String("name") | ||||||
| 			cfg := config.Config(ctx) | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			found := false | 			found := false | ||||||
| 			index := 0 | 			index := 0 | ||||||
| 			for i, repo := range cfg.Repos { | 			reposSlice := cfg.Repos() | ||||||
|  | 			for i, repo := range reposSlice { | ||||||
| 				if repo.Name == name { | 				if repo.Name == name { | ||||||
| 					index = i | 					index = i | ||||||
| 					found = true | 					found = true | ||||||
| @@ -129,26 +143,25 @@ func RemoveRepoCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			cfg.Repos = slices.Delete(cfg.Repos, index, index+1) | 			cfg.SetRepos(slices.Delete(reposSlice, index, index+1)) | ||||||
|  |  | ||||||
| 			cfgFl, err := os.Create(config.GetPaths(ctx).ConfigPath) | 			err = os.RemoveAll(filepath.Join(cfg.GetPaths().RepoDir, name)) | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Error(gotext.Get("Error opening config file"), "err", err) |  | ||||||
| 				os.Exit(1) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			err = toml.NewEncoder(cfgFl).Encode(&cfg) |  | ||||||
| 			if err != nil { |  | ||||||
| 				slog.Error(gotext.Get("Error encoding config"), "err", err) |  | ||||||
| 				os.Exit(1) |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			err = os.RemoveAll(filepath.Join(config.GetPaths(ctx).RepoDir, name)) |  | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error removing repo directory"), "err", err) | 				slog.Error(gotext.Get("Error removing repo directory"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
|  | 			err = cfg.SaveUserConfig() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error saving config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
| 			err = db.DeletePkgs(ctx, "repository = ?", name) | 			err = db.DeletePkgs(ctx, "repository = ?", name) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error removing packages from database"), "err", err) | 				slog.Error(gotext.Get("Error removing packages from database"), "err", err) | ||||||
| @@ -167,7 +180,20 @@ func RefreshCmd() *cli.Command { | |||||||
| 		Aliases: []string{"ref"}, | 		Aliases: []string{"ref"}, | ||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
| 			err := repos.Pull(ctx, config.Config(ctx).Repos) | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = rs.Pull(ctx, cfg.Repos()) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | 				slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								scripts/coverage-badge.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										46
									
								
								scripts/coverage-badge.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | # ALR - Any Linux Repository | ||||||
|  | # Copyright (C) 2025 Евгений Храмов | ||||||
|  | # | ||||||
|  | # 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/>. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #!/bin/bash | ||||||
|  | COVERAGE=$(go tool cover -func=coverage.out | grep total | awk '{print $3}' | sed 's/%//') | ||||||
|  |  | ||||||
|  | COLOR="#4c1" | ||||||
|  | if (( $(echo "$COVERAGE < 50" | bc -l) )); then | ||||||
|  |     COLOR="#e05d44" | ||||||
|  | elif (( $(echo "$COVERAGE < 80" | bc -l) )); then | ||||||
|  |     COLOR="#dfb317" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | cat <<EOF > assets/coverage-badge.svg | ||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="109" height="20"> | ||||||
|  |     <linearGradient id="smooth" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/> | ||||||
|  |     <stop offset="1" stop-opacity=".1"/></linearGradient> | ||||||
|  |     <mask id="round"> | ||||||
|  |         <rect width="109" height="20" rx="3" fill="#fff"/> | ||||||
|  |     </mask> | ||||||
|  |     <g mask="url(#round)"><rect width="65" height="20" fill="#555"/> | ||||||
|  |         <rect x="65" width="44" height="20" fill="${COLOR}"/> | ||||||
|  |         <rect width="109" height="20" fill="url(#smooth)"/> | ||||||
|  |     </g> | ||||||
|  |     <g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"> | ||||||
|  |         <text x="33.5" y="15" fill="#010101" fill-opacity=".3">coverage</text> | ||||||
|  |         <text x="33.5" y="14">coverage</text> | ||||||
|  |         <text x="86" y="15" fill="#010101" fill-opacity=".3">${COVERAGE}%</text> | ||||||
|  |         <text x="86" y="14">${COVERAGE}%</text> | ||||||
|  |     </g> | ||||||
|  | </svg> | ||||||
|  | EOF | ||||||
							
								
								
									
										84
									
								
								scripts/i18n-badge.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										84
									
								
								scripts/i18n-badge.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,84 @@ | |||||||
|  | #!/bin/bash | ||||||
|  | # ALR - Any Linux Repository | ||||||
|  | # Copyright (C) 2025 Евгений Храмов | ||||||
|  | # | ||||||
|  | # 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/>. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | TRANSLATIONS_DIR="internal/translations/po" | ||||||
|  |  | ||||||
|  | if [ ! -d "$TRANSLATIONS_DIR" ]; then | ||||||
|  |     echo "Error: directory '$TRANSLATIONS_DIR' not found" | ||||||
|  |     exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | declare -A TOTAL_STRINGS_MAP | ||||||
|  | declare -A TRANSLATED_STRINGS_MAP | ||||||
|  |  | ||||||
|  | for PO_FILE in $(find "$TRANSLATIONS_DIR" -type f -name "*.po"); do | ||||||
|  |     LANG_DIR=$(dirname "$PO_FILE") | ||||||
|  |     LANG=$(basename "$LANG_DIR") | ||||||
|  |  | ||||||
|  |     STATS=$(LC_ALL=C msgfmt --statistics -o /dev/null "$PO_FILE" 2>&1) | ||||||
|  |  | ||||||
|  |     NUMBERS=($(echo "$STATS" | grep -o '[0-9]\+')) | ||||||
|  |  | ||||||
|  |     case ${#NUMBERS[@]} in | ||||||
|  |         1) TRANSLATED_STRINGS=${NUMBERS[0]}; UNTRANSLATED_STRINGS=0 ;;  # all translated | ||||||
|  |         2) TRANSLATED_STRINGS=${NUMBERS[0]}; UNTRANSLATED_STRINGS=${NUMBERS[1]} ;;  # no fuzzy | ||||||
|  |         3) TRANSLATED_STRINGS=${NUMBERS[0]}; UNTRANSLATED_STRINGS=${NUMBERS[2]} ;;  # with fuzzy | ||||||
|  |         *) TRANSLATED_STRINGS=0; UNTRANSLATED_STRINGS=0 ;;  | ||||||
|  |     esac | ||||||
|  |  | ||||||
|  |     TOTAL_STRINGS=$((TRANSLATED_STRINGS + UNTRANSLATED_STRINGS)) | ||||||
|  |  | ||||||
|  |     TOTAL_STRINGS_MAP[$LANG]=$((TOTAL_STRINGS_MAP[$LANG] + TOTAL_STRINGS)) | ||||||
|  |     TRANSLATED_STRINGS_MAP[$LANG]=$((TRANSLATED_STRINGS_MAP[$LANG] + TRANSLATED_STRINGS)) | ||||||
|  | done | ||||||
|  |  | ||||||
|  | for LANG in "${!TOTAL_STRINGS_MAP[@]}"; do | ||||||
|  |     TOTAL=${TOTAL_STRINGS_MAP[$LANG]} | ||||||
|  |     TRANSLATED=${TRANSLATED_STRINGS_MAP[$LANG]} | ||||||
|  |     if [ "$TOTAL" -eq 0 ]; then | ||||||
|  |         PERCENTAGE="0.00" | ||||||
|  |     else | ||||||
|  |         PERCENTAGE=$(echo "scale=2; ($TRANSLATED / $TOTAL) * 100" | bc) | ||||||
|  |     fi | ||||||
|  |     COLOR="#4c1" | ||||||
|  |     if (( $(echo "$PERCENTAGE < 50" | bc -l) )); then | ||||||
|  |         COLOR="#e05d44" | ||||||
|  |     elif (( $(echo "$PERCENTAGE < 80" | bc -l) )); then | ||||||
|  |         COLOR="#dfb317" | ||||||
|  |     fi | ||||||
|  | cat <<EOF > assets/i18n-$LANG-badge.svg | ||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="129" height="20"> | ||||||
|  |     <linearGradient id="smooth" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/> | ||||||
|  |     <stop offset="1" stop-opacity=".1"/></linearGradient> | ||||||
|  |     <mask id="round"> | ||||||
|  |         <rect width="129" height="20" rx="3" fill="#fff"/> | ||||||
|  |     </mask> | ||||||
|  |     <g mask="url(#round)"> | ||||||
|  |         <rect width="75" height="20" fill="#555"/> | ||||||
|  |         <rect x="75" width="64" height="20" fill="${COLOR}"/> | ||||||
|  |         <rect width="129" height="20" fill="url(#smooth)"/> | ||||||
|  |     </g> | ||||||
|  |     <g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="11"> | ||||||
|  |         <text x="37" y="15" fill="#010101" fill-opacity=".3">$LANG translate</text> | ||||||
|  |         <text x="37" y="14">$LANG translate</text> | ||||||
|  |         <text x="100" y="15" fill="#010101" fill-opacity=".3">${PERCENTAGE}%</text> | ||||||
|  |         <text x="100" y="14">${PERCENTAGE}%</text> | ||||||
|  |     </g> | ||||||
|  | </svg> | ||||||
|  | EOF | ||||||
|  | done | ||||||
| @@ -78,6 +78,10 @@ elif command -v apk &>/dev/null; then | |||||||
|   info "Обнаружен apk" |   info "Обнаружен apk" | ||||||
|   pkgFormat="apk" |   pkgFormat="apk" | ||||||
|   pkgMgr="apk" |   pkgMgr="apk" | ||||||
|  | elif command -v apt-get &>/dev/null; then | ||||||
|  |   info "Обнаружен apt-get" | ||||||
|  |   pkgFormat="rpm" | ||||||
|  |   pkgMgr="apt-get" | ||||||
| else | else | ||||||
|   warn "Не обнаружен поддерживаемый менеджер пакетов!" |   warn "Не обнаружен поддерживаемый менеджер пакетов!" | ||||||
|   noPkgMgr=true |   noPkgMgr=true | ||||||
| @@ -98,7 +102,10 @@ if [ -z "$noPkgMgr" ]; then | |||||||
|   elif [ "$pkgMgr" == "apt" ]; then |   elif [ "$pkgMgr" == "apt" ]; then | ||||||
|     latestFile=$(echo "$fileList" | grep -E 'alr-bin-.*.amd64.deb' | sort -V | tail -n 1) |     latestFile=$(echo "$fileList" | grep -E 'alr-bin-.*.amd64.deb' | sort -V | tail -n 1) | ||||||
|   elif [[ "$pkgMgr" == "dnf" || "$pkgMgr" == "yum" || "$pkgMgr" == "zypper" ]]; then |   elif [[ "$pkgMgr" == "dnf" || "$pkgMgr" == "yum" || "$pkgMgr" == "zypper" ]]; then | ||||||
|     latestFile=$(echo "$fileList" | grep -E 'alr-bin-.*.x86_64.rpm' | sort -V | tail -n 1) |     latestFile=$(echo "$fileList" | grep -E 'alr-bin-.*\.x86_64\.rpm' | grep -v 'alt1' | sort -V | tail -n 1) | ||||||
|  |   elif [[ "$pkgMgr" == "apt-get"  ]]; then | ||||||
|  |     latestFile=$(echo "$fileList" | grep -E 'alr-bin-.*alt1.x86_64.rpm' | sort -V | tail -n 1) | ||||||
|  |  | ||||||
|   else |   else | ||||||
|     error "Не поддерживаемый менеджер пакетов для автоматической установки" |     error "Не поддерживаемый менеджер пакетов для автоматической установки" | ||||||
|   fi |   fi | ||||||
|   | |||||||
							
								
								
									
										125
									
								
								search.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								search.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | // ALR - Any Linux Repository | ||||||
|  | // Copyright (C) 2025 Евгений Храмов | ||||||
|  | // | ||||||
|  | // 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 main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log/slog" | ||||||
|  | 	"os" | ||||||
|  | 	"text/template" | ||||||
|  |  | ||||||
|  | 	"github.com/leonelquinteros/gotext" | ||||||
|  | 	"github.com/urfave/cli/v2" | ||||||
|  |  | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
|  | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/search" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func SearchCmd() *cli.Command { | ||||||
|  | 	return &cli.Command{ | ||||||
|  | 		Name:    "search", | ||||||
|  | 		Usage:   gotext.Get("Search packages"), | ||||||
|  | 		Aliases: []string{"s"}, | ||||||
|  | 		Flags: []cli.Flag{ | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "name", | ||||||
|  | 				Aliases: []string{"n"}, | ||||||
|  | 				Usage:   gotext.Get("Search by name"), | ||||||
|  | 			}, | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "description", | ||||||
|  | 				Aliases: []string{"d"}, | ||||||
|  | 				Usage:   gotext.Get("Search by description"), | ||||||
|  | 			}, | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "repository", | ||||||
|  | 				Aliases: []string{"repo"}, | ||||||
|  | 				Usage:   gotext.Get("Search by repository"), | ||||||
|  | 			}, | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "provides", | ||||||
|  | 				Aliases: []string{"p"}, | ||||||
|  | 				Usage:   gotext.Get("Search by provides"), | ||||||
|  | 			}, | ||||||
|  | 			&cli.StringFlag{ | ||||||
|  | 				Name:    "format", | ||||||
|  | 				Aliases: []string{"f"}, | ||||||
|  | 				Usage:   gotext.Get("Format output using a Go template"), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		Action: func(c *cli.Context) error { | ||||||
|  | 			ctx := c.Context | ||||||
|  | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			defer db.Close() | ||||||
|  |  | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			format := c.String("format") | ||||||
|  | 			var tmpl *template.Template | ||||||
|  | 			if format != "" { | ||||||
|  | 				tmpl, err = template.New("format").Parse(format) | ||||||
|  | 				if err != nil { | ||||||
|  | 					slog.Error(gotext.Get("Error parsing format template"), "err", err) | ||||||
|  | 					os.Exit(1) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			s := search.New(db) | ||||||
|  |  | ||||||
|  | 			packages, err := s.Search( | ||||||
|  | 				ctx, | ||||||
|  | 				search.NewSearchOptions(). | ||||||
|  | 					WithName(c.String("name")). | ||||||
|  | 					WithDescription(c.String("description")). | ||||||
|  | 					WithRepository(c.String("repository")). | ||||||
|  | 					WithProvides(c.String("provides")). | ||||||
|  | 					Build(), | ||||||
|  | 			) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error parsing format template"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			for _, dbPkg := range packages { | ||||||
|  | 				if tmpl != nil { | ||||||
|  | 					err = tmpl.Execute(os.Stdout, dbPkg) | ||||||
|  | 					if err != nil { | ||||||
|  | 						slog.Error(gotext.Get("Error executing template"), "err", err) | ||||||
|  | 						os.Exit(1) | ||||||
|  | 					} | ||||||
|  | 					fmt.Println() | ||||||
|  | 				} else { | ||||||
|  | 					fmt.Println(dbPkg.Name) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			return nil | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										80
									
								
								upgrade.go
									
									
									
									
									
								
							
							
						
						
									
										80
									
								
								upgrade.go
									
									
									
									
									
								
							| @@ -29,16 +29,16 @@ import ( | |||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| 	"go.elara.ws/vercmp" | 	"go.elara.ws/vercmp" | ||||||
| 	"golang.org/x/exp/maps" | 	"golang.org/x/exp/maps" | ||||||
| 	"golang.org/x/exp/slices" |  | ||||||
|  |  | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/config" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | 	database "gitea.plemya-x.ru/Plemya-x/ALR/internal/db" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/overrides" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/overrides" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | 	"gitea.plemya-x.ru/Plemya-x/ALR/internal/types" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/build" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/distro" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/manager" | ||||||
| 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/repos" | ||||||
|  | 	"gitea.plemya-x.ru/Plemya-x/ALR/pkg/search" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func UpgradeCmd() *cli.Command { | func UpgradeCmd() *cli.Command { | ||||||
| @@ -56,7 +56,20 @@ func UpgradeCmd() *cli.Command { | |||||||
| 		Action: func(c *cli.Context) error { | 		Action: func(c *cli.Context) error { | ||||||
| 			ctx := c.Context | 			ctx := c.Context | ||||||
|  |  | ||||||
| 			cfg := config.GetInstance(ctx) | 			cfg := config.New() | ||||||
|  | 			err := cfg.Load() | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error loading config"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			db := database.New(cfg) | ||||||
|  | 			rs := repos.New(cfg, db) | ||||||
|  | 			err = db.Init(ctx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				slog.Error(gotext.Get("Error initialization database"), "err", err) | ||||||
|  | 				os.Exit(1) | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			info, err := distro.ParseOSRelease(ctx) | 			info, err := distro.ParseOSRelease(ctx) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| @@ -70,22 +83,33 @@ func UpgradeCmd() *cli.Command { | |||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if cfg.AutoPull(ctx) { | 			if cfg.AutoPull() { | ||||||
| 				err = repos.Pull(ctx, config.Config(ctx).Repos) | 				err = rs.Pull(ctx, cfg.Repos()) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					slog.Error(gotext.Get("Error pulling repos"), "err", err) | 					slog.Error(gotext.Get("Error pulling repos"), "err", err) | ||||||
| 					os.Exit(1) | 					os.Exit(1) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			updates, err := checkForUpdates(ctx, mgr, info) | 			updates, err := checkForUpdates(ctx, mgr, cfg, db, rs, info) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				slog.Error(gotext.Get("Error checking for updates"), "err", err) | 				slog.Error(gotext.Get("Error checking for updates"), "err", err) | ||||||
| 				os.Exit(1) | 				os.Exit(1) | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if len(updates) > 0 { | 			if len(updates) > 0 { | ||||||
| 				build.InstallPkgs(ctx, updates, nil, types.BuildOpts{ | 				builder := build.NewBuilder( | ||||||
|  | 					ctx, | ||||||
|  | 					types.BuildOpts{ | ||||||
|  | 						Manager:     mgr, | ||||||
|  | 						Clean:       c.Bool("clean"), | ||||||
|  | 						Interactive: c.Bool("interactive"), | ||||||
|  | 					}, | ||||||
|  | 					rs, | ||||||
|  | 					info, | ||||||
|  | 					cfg, | ||||||
|  | 				) | ||||||
|  | 				builder.InstallPkgs(ctx, updates, nil, types.BuildOpts{ | ||||||
| 					Manager:     mgr, | 					Manager:     mgr, | ||||||
| 					Clean:       c.Bool("clean"), | 					Clean:       c.Bool("clean"), | ||||||
| 					Interactive: c.Bool("interactive"), | 					Interactive: c.Bool("interactive"), | ||||||
| @@ -99,32 +123,45 @@ func UpgradeCmd() *cli.Command { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func checkForUpdates(ctx context.Context, mgr manager.Manager, info *distro.OSRelease) ([]db.Package, error) { | func checkForUpdates( | ||||||
|  | 	ctx context.Context, | ||||||
|  | 	mgr manager.Manager, | ||||||
|  | 	cfg *config.ALRConfig, | ||||||
|  | 	db *database.Database, | ||||||
|  | 	rs *repos.Repos, | ||||||
|  | 	info *distro.OSRelease, | ||||||
|  | ) ([]database.Package, error) { | ||||||
| 	installed, err := mgr.ListInstalled(nil) | 	installed, err := mgr.ListInstalled(nil) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	pkgNames := maps.Keys(installed) | 	pkgNames := maps.Keys(installed) | ||||||
| 	found, _, err := repos.FindPkgs(ctx, pkgNames) |  | ||||||
|  | 	s := search.New(db) | ||||||
|  |  | ||||||
|  | 	var out []database.Package | ||||||
|  | 	for _, pkgName := range pkgNames { | ||||||
|  | 		matches := build.RegexpALRPackageName.FindStringSubmatch(pkgName) | ||||||
|  | 		if matches != nil { | ||||||
|  | 			packageName := matches[build.RegexpALRPackageName.SubexpIndex("package")] | ||||||
|  | 			repoName := matches[build.RegexpALRPackageName.SubexpIndex("repo")] | ||||||
|  |  | ||||||
|  | 			pkgs, err := s.Search( | ||||||
|  | 				ctx, | ||||||
|  | 				search.NewSearchOptions(). | ||||||
|  | 					WithName(packageName). | ||||||
|  | 					WithRepository(repoName). | ||||||
|  | 					Build(), | ||||||
|  | 			) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return nil, err | 				return nil, err | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 	var out []db.Package | 			if len(pkgs) == 0 { | ||||||
| 	for pkgName, pkgs := range found { |  | ||||||
| 		if slices.Contains(config.Config(ctx).IgnorePkgUpdates, pkgName) { |  | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 		if len(pkgs) > 1 { |  | ||||||
| 			// Puts the element with the highest version first |  | ||||||
| 			slices.SortFunc(pkgs, func(a, b db.Package) int { |  | ||||||
| 				return vercmp.Compare(a.Version, b.Version) |  | ||||||
| 			}) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		// First element is the package we want to install |  | ||||||
| 			pkg := pkgs[0] | 			pkg := pkgs[0] | ||||||
|  |  | ||||||
| 			repoVer := pkg.Version | 			repoVer := pkg.Version | ||||||
| @@ -143,5 +180,8 @@ func checkForUpdates(ctx context.Context, mgr manager.Manager, info *distro.OSRe | |||||||
| 				out = append(out, pkg) | 				out = append(out, pkg) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return out, nil | 	return out, nil | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user