yay/pacman/pacman.go

528 lines
11 KiB
Go
Raw Normal View History

2016-12-01 00:11:13 +01:00
package pacman
import (
"fmt"
"os"
"os/exec"
"strings"
2016-12-01 02:29:09 +01:00
"github.com/jguer/go-alpm"
2017-01-05 17:13:52 +01:00
"github.com/jguer/yay/util"
2016-12-01 00:11:13 +01:00
)
// Query describes a Repository search.
type Query []Result
2016-12-01 00:11:13 +01:00
// Result describes a pkg.
type Result struct {
Name string
Repository string
Version string
Description string
2016-12-02 13:19:03 +01:00
Group string
2016-12-01 00:11:13 +01:00
Installed bool
}
// PacmanConf describes the default pacman config file
const PacmanConf string = "/etc/pacman.conf"
var conf alpm.PacmanConfig
func init() {
conf, _ = readConfig(PacmanConf)
}
func readConfig(pacmanconf string) (conf alpm.PacmanConfig, err error) {
file, err := os.Open(pacmanconf)
if err != nil {
return
}
conf, err = alpm.ParseConfig(file)
if err != nil {
return
}
return
}
// UpdatePackages handles cache update and upgrade
func UpdatePackages(flags []string) error {
var cmd *exec.Cmd
var args []string
args = append(args, "pacman", "-Syu")
args = append(args, flags...)
cmd = exec.Command("sudo", args...)
cmd.Stdout = os.Stdout
cmd.Stdin = os.Stdin
cmd.Stderr = os.Stderr
err := cmd.Run()
return err
}
2016-12-02 13:19:03 +01:00
// Search handles repo searches. Creates a RepoSearch struct.
func Search(pkgName string) (s Query, n int, err error) {
2016-12-01 00:11:13 +01:00
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
}
2016-12-02 13:19:03 +01:00
localDb, err := h.LocalDb()
if err != nil {
return
2016-12-01 00:11:13 +01:00
}
2016-12-02 13:19:03 +01:00
dbList, err := h.SyncDbs()
2016-12-01 00:11:13 +01:00
if err != nil {
2016-12-02 13:19:03 +01:00
return
2016-12-01 00:11:13 +01:00
}
var installed bool
dbS := dbList.Slice()
// BottomUp functions
initL := func(len int) int {
return len - 1
}
compL := func(len int, i int) bool {
if i > 0 {
return true
}
return false
}
finalL := func(i int) int {
return i - 1
}
// TopDown functions
if util.SortMode == util.TopDown {
initL = func(len int) int {
return 0
}
compL = func(len int, i int) bool {
if i < len {
return true
}
return false
}
finalL = func(i int) int {
return i + 1
}
}
lenDbs := len(dbS)
for f := initL(lenDbs); compL(lenDbs, f); f = finalL(f) {
pkgS := dbS[f].PkgCache().Slice()
lenPkgs := len(pkgS)
for i := initL(lenPkgs); compL(lenPkgs, i); i = finalL(i) {
if strings.Contains(pkgS[i].Name(), pkgName) || strings.Contains(strings.ToLower(pkgS[i].Description()), pkgName) {
if r, _ := localDb.PkgByName(pkgS[i].Name()); r != nil {
2016-12-01 00:11:13 +01:00
installed = true
} else {
installed = false
}
2016-12-02 13:19:03 +01:00
s = append(s, Result{
Name: pkgS[i].Name(),
Description: pkgS[i].Description(),
Version: pkgS[i].Version(),
Repository: dbS[f].Name(),
Group: strings.Join(pkgS[i].Groups().Slice(), ","),
2016-12-01 00:11:13 +01:00
Installed: installed,
})
2016-12-02 13:19:03 +01:00
n++
2016-12-01 00:11:13 +01:00
}
}
2016-12-01 00:11:13 +01:00
}
return
}
//PrintSearch receives a RepoSearch type and outputs pretty text.
func (s Query) PrintSearch() {
2016-12-02 13:19:03 +01:00
for i, res := range s {
var toprint string
2017-01-05 17:13:52 +01:00
if util.SearchVerbosity == util.NumberMenu {
if util.SortMode == util.BottomUp {
2016-12-02 13:19:03 +01:00
toprint += fmt.Sprintf("%d ", len(s)-i-1)
} else {
toprint += fmt.Sprintf("%d ", i)
}
2017-01-05 17:13:52 +01:00
} else if util.SearchVerbosity == util.Minimal {
fmt.Println(res.Name)
continue
2016-12-01 00:11:13 +01:00
}
2016-12-02 13:19:03 +01:00
toprint += fmt.Sprintf("\x1b[1m%s/\x1b[33m%s \x1b[36m%s \x1b[0m",
res.Repository, res.Name, res.Version)
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
if len(res.Group) != 0 {
toprint += fmt.Sprintf("(%s) ", res.Group)
}
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
if res.Installed == true {
toprint += fmt.Sprintf("\x1b[32;40mInstalled\x1b[0m")
}
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
toprint += "\n" + res.Description
fmt.Println(toprint)
2016-12-01 00:11:13 +01:00
}
2016-12-02 13:19:03 +01:00
}
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
// PFactory execute an action over a series of packages without reopening the handle everytime.
// Everybody told me it wouln't work. It does. It's just not pretty.
// When it worked: https://youtu.be/a4Z5BdEL0Ag?t=1m11s
func PFactory(action func(interface{})) func(name string, object interface{}, rel bool) {
h, _ := conf.CreateHandle()
localDb, _ := h.LocalDb()
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
return func(name string, object interface{}, rel bool) {
_, err := localDb.PkgByName(name)
if err == nil {
action(object)
}
2016-12-01 00:11:13 +01:00
2016-12-02 13:19:03 +01:00
if rel {
h.Release()
}
}
2016-12-01 00:11:13 +01:00
}
2016-12-01 01:36:58 +01:00
// PackageSlices separates an input slice into aur and repo slices
func PackageSlices(toCheck []string) (aur []string, repo []string, err error) {
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
dbList, err := h.SyncDbs()
if err != nil {
return
}
for _, pkg := range toCheck {
found := false
for _, db := range dbList.Slice() {
_, err = db.PkgByName(pkg)
if err == nil {
found = true
repo = append(repo, pkg)
break
}
}
if !found {
if _, err := dbList.PkgCachebyGroup(pkg); err == nil {
repo = append(repo, pkg)
} else {
aur = append(aur, pkg)
}
2016-12-01 01:36:58 +01:00
}
}
err = nil
2016-12-01 01:36:58 +01:00
return
}
// BuildDependencies finds packages, on the second run
// compares with a baselist and avoids searching those
func BuildDependencies(baselist []string) func(toCheck []string, isBaseList bool, last bool) (repo []string, notFound []string) {
h, _ := conf.CreateHandle()
localDb, _ := h.LocalDb()
dbList, _ := h.SyncDbs()
f := func(c rune) bool {
return c == '>' || c == '<' || c == '=' || c == ' '
}
return func(toCheck []string, isBaseList bool, close bool) (repo []string, notFound []string) {
if close {
h.Release()
return
}
Loop:
for _, dep := range toCheck {
if !isBaseList {
for _, base := range baselist {
if base == dep {
continue Loop
}
}
}
if _, erp := localDb.PkgCache().FindSatisfier(dep); erp == nil {
continue
} else if pkg, erp := dbList.FindSatisfier(dep); erp == nil {
repo = append(repo, pkg.Name())
} else {
field := strings.FieldsFunc(dep, f)
notFound = append(notFound, field[0])
}
}
return
}
}
// DepSatisfier receives a string slice, returns a slice of packages found in
// repos and one of packages not found in repos. Leaves out installed packages.
func DepSatisfier(toCheck []string) (repo []string, notFound []string, err error) {
2016-12-01 00:11:13 +01:00
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
dbList, err := h.SyncDbs()
if err != nil {
return
}
f := func(c rune) bool {
return c == '>' || c == '<' || c == '=' || c == ' '
}
for _, dep := range toCheck {
if _, erp := localDb.PkgCache().FindSatisfier(dep); erp == nil {
2016-12-01 00:11:13 +01:00
continue
} else if pkg, erp := dbList.FindSatisfier(dep); erp == nil {
repo = append(repo, pkg.Name())
} else {
field := strings.FieldsFunc(dep, f)
notFound = append(notFound, field[0])
2016-12-01 00:11:13 +01:00
}
}
err = nil
2016-12-01 00:11:13 +01:00
return
}
2016-12-02 13:19:03 +01:00
// Install sends an install command to pacman with the pkgName slice
func Install(pkgName []string, flags []string) (err error) {
if len(pkgName) == 0 {
return nil
}
var cmd *exec.Cmd
var args []string
args = append(args, "pacman", "-S")
args = append(args, pkgName...)
if len(flags) != 0 {
args = append(args, flags...)
}
cmd = exec.Command("sudo", args...)
cmd.Stdin, cmd.Stdout, cmd.Stderr = os.Stdin, os.Stdout, os.Stderr
cmd.Run()
return nil
}
// CleanRemove sends a full removal command to pacman with the pkgName slice
func CleanRemove(pkgName []string) (err error) {
if len(pkgName) == 0 {
return nil
}
var cmd *exec.Cmd
var args []string
args = append(args, "pacman", "-Rnsc")
2016-12-02 13:19:03 +01:00
args = append(args, pkgName...)
2016-12-17 03:29:52 +01:00
args = append(args, "--noconfirm")
2016-12-02 13:19:03 +01:00
cmd = exec.Command("sudo", args...)
cmd.Stdin, cmd.Stdout, cmd.Stderr = os.Stdin, os.Stdout, os.Stderr
2016-12-02 13:19:03 +01:00
cmd.Run()
return nil
}
2016-12-01 00:11:13 +01:00
// ForeignPackages returns a map of foreign packages, with their version and date as values.
func ForeignPackages() (foreign map[string]*struct {
Version string
Date int64
}, n int, err error) {
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
dbList, err := h.SyncDbs()
if err != nil {
return
}
foreign = make(map[string]*struct {
Version string
Date int64
})
// Find foreign packages in system
for _, pkg := range localDb.PkgCache().Slice() {
// Change to more effective method
found := false
for _, db := range dbList.Slice() {
_, err = db.PkgByName(pkg.Name())
if err == nil {
found = true
break
}
}
if !found {
foreign[pkg.Name()] = &struct {
Version string
Date int64
}{pkg.Version(), pkg.InstallDate().Unix()}
n++
}
}
return
}
2016-12-01 01:36:58 +01:00
// Statistics returns statistics about packages installed in system
2016-12-15 02:18:07 +01:00
func Statistics() (info struct {
2016-12-01 01:36:58 +01:00
Totaln int
Expln int
TotalSize int64
}, err error) {
var tS int64 // TotalSize
var nPkg int
var ePkg int
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
2016-12-15 02:18:07 +01:00
for _, pkg := range localDb.PkgCache().Slice() {
2016-12-01 01:36:58 +01:00
tS += pkg.ISize()
nPkg++
if pkg.Reason() == 0 {
ePkg++
}
}
info = struct {
Totaln int
Expln int
TotalSize int64
}{
nPkg, ePkg, tS,
}
return
}
2016-12-15 02:18:07 +01:00
// BiggestPackages prints the name of the ten biggest packages in the system.
func BiggestPackages() {
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
pkgCache := localDb.PkgCache()
pkgS := pkgCache.SortBySize().Slice()
if len(pkgS) < 10 {
return
}
for i := 0; i < 10; i++ {
fmt.Printf("%s: \x1B[0;33m%dMB\x1B[0m\n", pkgS[i].Name(), pkgS[i].ISize()/(1024*1024))
}
// Could implement size here as well, but we just want the general idea
}
2016-12-17 03:29:52 +01:00
// HangingPackages returns a list of packages installed as deps
// and unneeded by the system
func HangingPackages() (hanging []string, err error) {
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
f := func(pkg alpm.Package) error {
if pkg.Reason() != alpm.PkgReasonDepend {
return nil
}
requiredby := pkg.ComputeRequiredBy()
if len(requiredby) == 0 {
hanging = append(hanging, pkg.Name())
fmt.Printf("%s: \x1B[0;33m%dMB\x1B[0m\n", pkg.Name(), pkg.ISize()/(1024*1024))
}
return nil
}
err = localDb.PkgCache().ForEach(f)
return
}
// SliceHangingPackages returns a list of packages installed as deps
// and unneeded by the system from a provided list of package names.
func SliceHangingPackages(pkgS []string) (hanging []string) {
h, err := conf.CreateHandle()
defer h.Release()
if err != nil {
return
}
localDb, err := h.LocalDb()
if err != nil {
return
}
big:
for _, pkgName := range pkgS {
for _, hangN := range hanging {
if hangN == pkgName {
continue big
}
}
pkg, err := localDb.PkgByName(pkgName)
if err == nil {
if pkg.Reason() != alpm.PkgReasonDepend {
continue
}
requiredby := pkg.ComputeRequiredBy()
if len(requiredby) == 0 {
hanging = append(hanging, pkgName)
fmt.Printf("%s: \x1B[0;33m%dMB\x1B[0m\n", pkg.Name(), pkg.ISize()/(1024*1024))
}
}
}
return
}