Compare commits

..

14 Commits

5 changed files with 524 additions and 119 deletions

View File

@@ -1,5 +1,5 @@
BINARY=ginie BINARY=ginie
VERSION=0.1.2 VERSION=0.1.6
build: build:
go build -o $(BINARY) ./cmd/ginie go build -o $(BINARY) ./cmd/ginie

View File

@@ -1,65 +1,123 @@
package main package main
import ( import (
"fmt" "fmt"
"os" "os"
"gitea.home.musaberdem.de/musabe24/ginie/internal/repos" "gitea.home.musaberdem.de/musabe24/ginie/internal/install"
"gitea.home.musaberdem.de/musabe24/ginie/internal/version" "gitea.home.musaberdem.de/musabe24/ginie/internal/repos"
"gitea.home.musaberdem.de/musabe24/ginie/internal/version"
) )
func main() { func main() {
if len(os.Args) < 2 { if len(os.Args) < 2 {
fmt.Println("Ginie Git Native Installer Engine") fmt.Println("Ginie Git Native Installer Engine")
fmt.Println("Usage: ginie <command> [arguments]") fmt.Println("Usage: ginie <command> [arguments]")
return return
} }
cmd := os.Args[1] cmd := os.Args[1]
switch cmd { switch cmd {
case "--version", "-v": case "--version", "-v":
fmt.Println("ginie version", version.Version) fmt.Println("ginie version", version.Version)
return return
case "add": case "add":
if len(os.Args) < 3 { if len(os.Args) < 3 {
fmt.Println("Usage: ginie add <GitHub-URL>") fmt.Println("Usage: ginie add <GitHub-URL>")
return return
} }
url := os.Args[2] url := os.Args[2]
err := repos.AddRepo(url) err := repos.AddRepo(url)
if err != nil { if err != nil {
fmt.Println("Fehler:", err) fmt.Println("Fehler:", err)
return return
} }
fmt.Println("Repo hinzugefügt:", url) fmt.Println("Repo hinzugefügt:", url)
return return
case "list": case "remove":
reposList, err := repos.ListRepos() if len(os.Args) < 3 {
if err != nil { fmt.Println("Usage: ginie remove <GitHub-URL>")
fmt.Println("Fehler beim Laden:", err) return
return }
} url := os.Args[2]
if len(reposList) == 0 { err := repos.RemoveRepo(url)
fmt.Println("Noch keine Repositories hinzugefügt. Nutze:") if err != nil {
fmt.Println(" ginie add <GitHub-URL>") fmt.Println("Fehler:", err)
return return
} }
fmt.Println("Registrierte Repositories:") fmt.Println("Repo entfernt:", url)
for _, r := range reposList { return
fmt.Printf("- %s/%s (%s)\n", r.Owner, r.Name, r.URL)
}
return
default: case "list":
fmt.Println("Unbekannter Befehl:", cmd) reposList, err := repos.ListRepos()
return if err != nil {
} fmt.Println("Fehler beim Laden:", err)
return
}
if len(reposList) == 0 {
fmt.Println("Noch keine Repositories hinzugefügt. Nutze:")
fmt.Println(" ginie add <GitHub-URL>")
return
}
fmt.Println("Registrierte Repositories:")
for _, r := range reposList {
version := r.InstalledVersion
if version == "" {
version = "nicht installiert"
}
fmt.Printf("- %s/%s (%s) Version: %s\n", r.Owner, r.Name, r.URL, version)
}
return
case "install":
if len(os.Args) < 3 {
fmt.Println("Usage: ginie install <name[:version]>")
return
}
target := os.Args[2]
err := install.Install(target)
if err != nil {
fmt.Println("Fehler:", err)
return
}
return
case "update":
if len(os.Args) < 3 {
fmt.Println("Usage: ginie update <name|--all>")
return
}
target := os.Args[2]
if target == "--all" {
err := install.UpdateAll()
if err != nil {
fmt.Println("Fehler:", err)
return
}
return
}
err := install.Update(target)
if err != nil {
fmt.Println("Fehler:", err)
return
}
return
default:
fmt.Println("Unbekannter Befehl:", cmd)
return
}
} }

301
internal/install/install.go Normal file
View File

@@ -0,0 +1,301 @@
package install
import (
"encoding/json"
"errors"
"fmt"
"io"
"net/http"
"os"
"os/exec"
"path/filepath"
"strings"
"gitea.home.musaberdem.de/musabe24/ginie/internal/repos"
)
type Installed struct {
Name string `json:"name"`
Version string `json:"version"`
AssetName string `json:"asset"`
AssetURL string `json:"url"`
}
type releaseAsset struct {
Name string `json:"name"`
BrowserDownloadURL string `json:"browser_download_url"`
}
func configPath() (string, error) {
home, err := os.UserHomeDir()
if err != nil {
return "", err
}
dir := filepath.Join(home, ".config", "ginie")
os.MkdirAll(dir, 0755)
return filepath.Join(dir, "installed.json"), nil
}
func LoadInstalled() ([]Installed, error) {
path, _ := configPath()
data, err := os.ReadFile(path)
if os.IsNotExist(err) {
return []Installed{}, nil
}
if err != nil {
return nil, err
}
var out []Installed
err = json.Unmarshal(data, &out)
return out, err
}
func SaveInstalled(list []Installed) error {
path, _ := configPath()
data, err := json.MarshalIndent(list, "", " ")
if err != nil {
return err
}
return os.WriteFile(path, data, 0644)
}
func parseNameVersion(input string) (name string, version string) {
parts := strings.Split(input, ":")
if len(parts) == 1 {
return parts[0], "" // keine Version
}
return parts[0], parts[1]
}
func Install(pkg string) error {
name, version := parseNameVersion(pkg)
// Repo suchen
all, err := repos.LoadRepos()
if err != nil {
return err
}
var repo repos.Repo
found := false
for _, r := range all {
if r.Name == name {
repo = r
found = true
break
}
}
if !found {
return errors.New("Repo nicht gefunden. Nutze: ginie add <URL>")
}
// Release-API-URL
api := ""
if version == "" {
api = fmt.Sprintf("https://api.github.com/repos/%s/%s/releases/latest", repo.Owner, repo.Name)
} else {
api = fmt.Sprintf("https://api.github.com/repos/%s/%s/releases/tags/%s", repo.Owner, repo.Name, version)
}
// API abfragen
resp, err := http.Get(api)
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode != 200 {
return fmt.Errorf("GitHub API Fehler: %s", resp.Status)
}
var release struct {
TagName string `json:"tag_name"`
Assets []releaseAsset `json:"assets"`
}
if err := json.NewDecoder(resp.Body).Decode(&release); err != nil {
return err
}
if len(release.Assets) == 0 {
return errors.New("Release hat keine Assets")
}
installedVersion := release.TagName
if installedVersion == "" {
installedVersion = version
}
var debAssets []releaseAsset
for _, a := range release.Assets {
if strings.HasSuffix(strings.ToLower(a.Name), ".deb") {
debAssets = append(debAssets, a)
}
}
if len(debAssets) == 0 {
return errors.New("Release enthält keine .deb Assets")
}
// Asset-Auswahl
fmt.Println("Verfügbare Assets:")
for i, a := range debAssets {
fmt.Printf("[%d] %s\n", i, a.Name)
}
fmt.Print("Wähle Asset-Nummer: ")
var choice int
fmt.Scan(&choice)
if choice < 0 || choice >= len(debAssets) {
return errors.New("Ungültige Auswahl")
}
asset := debAssets[choice]
// Datei herunterladen
fmt.Println("Lade herunter:", asset.Name)
out, err := os.Create(asset.Name)
if err != nil {
return err
}
defer out.Close()
dl, err := http.Get(asset.BrowserDownloadURL)
if err != nil {
return err
}
defer dl.Body.Close()
_, err = io.Copy(out, dl.Body)
if err != nil {
return err
}
fmt.Println("Download abgeschlossen.")
// Installation (unterstützt .deb)
if strings.HasSuffix(asset.Name, ".deb") {
fmt.Println("Installiere .deb Datei…")
cmd := exec.Command("sudo", "dpkg", "-i", asset.Name)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
// Run the install command
err := cmd.Run()
if err != nil {
fmt.Println("Package konnte nicht installiert werden.")
}
// Remove debian package file
err = os.Remove(asset.Name)
if err != nil {
fmt.Println("Package konnte nicht gelöscht werden.")
}
if installedVersion == "" {
installedVersion = "unknown"
}
if err := repos.SetInstalledVersion(repo.Name, installedVersion); err != nil {
fmt.Println("Speichern der installierten Version nicht möglich.")
return err
}
return nil
}
return errors.New("Asset-Typ wird noch nicht unterstützt")
}
func Update(pkg string) error {
name := pkg
all, err := repos.LoadRepos()
if err != nil {
return err
}
var repo repos.Repo
found := false
for _, r := range all {
if r.Name == name {
repo = r
found = true
break
}
}
if !found {
return errors.New("Repo nicht gefunden. Nutze: ginie add <URL>")
}
if repo.InstalledVersion == "" {
return errors.New("Package noch nicht installiert. Nutze: ginie install " + name)
}
api := fmt.Sprintf("https://api.github.com/repos/%s/%s/releases/latest", repo.Owner, repo.Name)
resp, err := http.Get(api)
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode != 200 {
return fmt.Errorf("GitHub API Fehler: %s", resp.Status)
}
var release struct {
TagName string `json:"tag_name"`
}
if err := json.NewDecoder(resp.Body).Decode(&release); err != nil {
return err
}
if release.TagName == "" {
return errors.New("Keine Version im Release gefunden")
}
if release.TagName == repo.InstalledVersion {
fmt.Printf("%s ist bereits auf Version %s\n", repo.Name, repo.InstalledVersion)
return nil
}
fmt.Printf("Aktualisiere %s von %s auf %s\n", repo.Name, repo.InstalledVersion, release.TagName)
return Install(fmt.Sprintf("%s:%s", repo.Name, release.TagName))
}
func UpdateAll() error {
reposList, err := repos.LoadRepos()
if err != nil {
return err
}
var installed []repos.Repo
for _, r := range reposList {
if r.InstalledVersion != "" {
installed = append(installed, r)
}
}
if len(installed) == 0 {
return errors.New("Keine installierten Pakete gefunden. Nutze: ginie install <name>")
}
var failed []string
for _, r := range installed {
fmt.Printf("Prüfe Updates für %s...\n", r.Name)
if err := Update(r.Name); err != nil {
failed = append(failed, fmt.Sprintf("%s: %v", r.Name, err))
}
}
if len(failed) > 0 {
return fmt.Errorf("Einige Updates fehlgeschlagen:\n%s", strings.Join(failed, "\n"))
}
return nil
}

View File

@@ -1,108 +1,154 @@
package repos package repos
import ( import (
"encoding/json" "encoding/json"
"errors" "errors"
"os" "os"
"path/filepath" "path/filepath"
"strings" "strings"
) )
type Repo struct { type Repo struct {
URL string `json:"url"` URL string `json:"url"`
Owner string `json:"owner"` Owner string `json:"owner"`
Name string `json:"name"` Name string `json:"name"`
InstalledVersion string `json:"installed_version,omitempty"`
} }
func configPath() (string, error) { func configPath() (string, error) {
home, err := os.UserHomeDir() home, err := os.UserHomeDir()
if err != nil { if err != nil {
return "", err return "", err
} }
dir := filepath.Join(home, ".config", "ginie") dir := filepath.Join(home, ".config", "ginie")
os.MkdirAll(dir, 0755) os.MkdirAll(dir, 0755)
return filepath.Join(dir, "repos.json"), nil return filepath.Join(dir, "repos.json"), nil
} }
func LoadRepos() ([]Repo, error) { func LoadRepos() ([]Repo, error) {
path, err := configPath() path, err := configPath()
if err != nil { if err != nil {
return nil, err return nil, err
} }
data, err := os.ReadFile(path) data, err := os.ReadFile(path)
if os.IsNotExist(err) { if os.IsNotExist(err) {
return []Repo{}, nil return []Repo{}, nil
} }
if err != nil { if err != nil {
return nil, err return nil, err
} }
var repos []Repo var repos []Repo
err = json.Unmarshal(data, &repos) err = json.Unmarshal(data, &repos)
if err != nil { if err != nil {
return nil, err return nil, err
} }
return repos, nil return repos, nil
} }
func SaveRepos(repos []Repo) error { func SaveRepos(repos []Repo) error {
path, err := configPath() path, err := configPath()
if err != nil { if err != nil {
return err return err
} }
data, err := json.MarshalIndent(repos, "", " ") data, err := json.MarshalIndent(repos, "", " ")
if err != nil { if err != nil {
return err return err
} }
return os.WriteFile(path, data, 0644) return os.WriteFile(path, data, 0644)
} }
func ParseGitHubURL(raw string) (Repo, error) { func ParseGitHubURL(raw string) (Repo, error) {
if !strings.HasPrefix(raw, "https://github.com/") { if !strings.HasPrefix(raw, "https://github.com/") {
return Repo{}, errors.New("URL ist kein gültiger GitHub-Link") return Repo{}, errors.New("URL ist kein gültiger GitHub-Link")
} }
parts := strings.Split(strings.TrimPrefix(raw, "https://github.com/"), "/") parts := strings.Split(strings.TrimPrefix(raw, "https://github.com/"), "/")
if len(parts) < 2 { if len(parts) < 2 {
return Repo{}, errors.New("GitHub-Link hat nicht das Format: https://github.com/<owner>/<repo>") return Repo{}, errors.New("GitHub-Link hat nicht das Format: https://github.com/<owner>/<repo>")
} }
return Repo{ return Repo{
URL: raw, URL: raw,
Owner: parts[0], Owner: parts[0],
Name: parts[1], Name: parts[1],
}, nil }, nil
} }
func AddRepo(url string) error { func AddRepo(url string) error {
repo, err := ParseGitHubURL(url) repo, err := ParseGitHubURL(url)
if err != nil { if err != nil {
return err return err
} }
repos, err := LoadRepos() repos, err := LoadRepos()
if err != nil { if err != nil {
return err return err
} }
// prüfen ob bereits vorhanden // prüfen ob bereits vorhanden
for _, r := range repos { for _, r := range repos {
if r.URL == repo.URL { if r.URL == repo.URL {
return errors.New("Repo existiert bereits") return errors.New("Repo existiert bereits")
} }
} }
repos = append(repos, repo) repos = append(repos, repo)
return SaveRepos(repos) return SaveRepos(repos)
}
func RemoveRepo(url string) error {
repos, err := LoadRepos()
if err != nil {
return err
}
index := -1
for i, r := range repos {
if r.URL == url {
index = i
break
}
}
if index == -1 {
return errors.New("Repo nicht gefunden")
}
repos = append(repos[:index], repos[index+1:]...)
return SaveRepos(repos)
} }
func ListRepos() ([]Repo, error) { func ListRepos() ([]Repo, error) {
return LoadRepos() return LoadRepos()
}
func SetInstalledVersion(name string, version string) error {
repos, err := LoadRepos()
if err != nil {
return err
}
updated := false
for i := range repos {
if repos[i].Name == name {
repos[i].InstalledVersion = version
updated = true
break
}
}
if !updated {
return errors.New("Repo nicht gefunden")
}
return SaveRepos(repos)
} }

View File

@@ -1,4 +1,4 @@
package version package version
// Wird vom Build-Prozess überschrieben, falls gewünscht // Wird vom Build-Prozess überschrieben, falls gewünscht
var Version = "0.1.1" var Version = "0.1.6"