Files
packets/cmd/packets/main.go
2025-08-02 11:03:44 -03:00

1527 lines
34 KiB
Go

//go:build linux
package main
import (
"archive/tar"
"bufio"
"crypto/sha256"
"database/sql"
"encoding/hex"
"fmt"
"io"
"io/fs"
"log"
"net"
"net/http"
"net/url"
"os"
"os/exec"
"packets/internal"
"path"
"path/filepath"
"strconv"
"strings"
"syscall"
"time"
"github.com/BurntSushi/toml"
"github.com/klauspost/compress/zstd"
"github.com/schollz/progressbar/v3"
lua "github.com/yuin/gopher-lua"
"golang.org/x/net/ipv4"
_ "modernc.org/sqlite"
)
type ConfigTOML struct {
Config struct {
HttpPort int `toml:"httpPort"`
CacheDir string `toml:"cacheDir"`
AutoDeleteCacheDir bool `toml:"dayToDeleteCacheDir"`
DaysToDelete int `toml:"daysToDelete"`
DataDir string `toml:"dataDir"`
BinDir string `toml:"binDir"`
LastDataDir string `toml:"lastDataDir"`
} `toml:"Config"`
}
type IndexTOML struct {
Name string `toml:"name"`
Version string `toml:"version"`
Author string `toml:"author"`
Description string `toml:"description"`
CreatedAt time.Time `toml:"createdAt"`
}
type CountingReader struct {
R io.Reader
Total int64
}
func (c *CountingReader) Read(p []byte) (int, error) {
n, err := c.R.Read(p)
c.Total += int64(n)
return n, err
}
type Installed struct {
Realname string
Version string
Dependencies []string
Family string
Serial uint
}
type Peer struct {
IP net.IP
Port int
}
type Quer1 struct {
Realname string
Version string
Description string
}
type Manifest struct {
Info struct {
Name string `toml:"name"`
Version string `toml:"version"`
Description string `toml:"description"`
Dependencies []string `toml:"dependencies"`
Author string `toml:"author"`
Family string `toml:"family"`
Serial uint `toml:"serial"`
} `toml:"Info"`
Hooks struct {
Install string `toml:"install"`
Remove string `toml:"remove"`
} `toml:"Hooks"`
}
var serialPass uint
var cfg ConfigTOML
var PacketsDir string
func main() {
PacketsDir = internal.PacketsPackageDir()
_, err := os.Stat(filepath.Join(PacketsDir, "config.toml"))
if os.IsNotExist(err) {
fmt.Println("can't find config.toml, generating a default one")
os.MkdirAll(PacketsDir, 0644)
file, err := os.Create(filepath.Join(PacketsDir, "config.toml"))
if err != nil {
log.Fatal(err)
}
defer file.Close()
cfg := internal.DefaultConfigTOML()
encoder := toml.NewEncoder(file)
if err := encoder.Encode(cfg); err != nil {
log.Fatal(err)
}
file.WriteString("\n\n# BE CAREFULL CHANGING BIN_DIR, BECAUSE THE BINARIES DON'T MOVE AUTOMATICALLY\n#NEVER CHANGE lastDataDir\n")
fmt.Println("Operation Sucess!")
}
_, err = toml.DecodeFile(filepath.Join(PacketsDir, "config.toml"), &cfg)
if err != nil {
log.Fatal(err)
}
if len(os.Args) < 2 {
fmt.Println("invalid syntax")
return
}
cmd := os.Args[1]
switch cmd {
case "install":
if os.Getuid() != 0 {
fmt.Println("please, run as root")
return
}
if len(os.Args) < 3 {
fmt.Println("usage: packets install <name>")
return
}
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
return
}
defer db.Close()
nameToQuery := os.Args[2]
var exist bool
db.QueryRow("SELECT EXISTS(SELECT 1 FROM packages WHERE realname = ? LIMIT 1)", nameToQuery).Scan(&exist)
if exist {
QueryInstall(nameToQuery)
return
}
rows, err := db.Query("SELECT realname, version, description FROM packages WHERE name = ?", nameToQuery)
if err != nil {
if strings.Contains(err.Error(), "file is not a database (26)") {
fmt.Println("index.db corrupted")
return
}
log.Panic(err)
return
}
defer rows.Close()
var pkgs []Quer1
for rows.Next() {
var q Quer1
if err := rows.Scan(&q.Realname, &q.Version, &q.Description); err != nil {
log.Fatal(err)
}
pkgs = append(pkgs, q)
}
switch len(pkgs) {
case 0:
fmt.Printf("can't find any results for %s\n", nameToQuery)
return
case 1:
fmt.Printf("Founded 1 package for %s \n", nameToQuery)
fmt.Printf("Downloading %s \n", pkgs[0].Realname)
QueryInstall(pkgs[0].Realname)
return
default:
fmt.Printf("Found %d versions of %s\n Select 1\n", len(pkgs), nameToQuery)
for i, q := range pkgs {
fmt.Printf("[%d] %s : %s\n %s\n", i, q.Realname, q.Version, q.Description)
}
var choice int
fmt.Fscan(bufio.NewReader(os.Stdin), &choice)
if choice > len(pkgs) || choice < 0 {
fmt.Println("invalid option")
return
}
QueryInstall(pkgs[choice].Realname)
return
}
case "serve":
if os.Getuid() != 0 {
fmt.Println("please, run as root")
return
}
if len(os.Args) < 3 {
fmt.Println("usage: packets serve <option>\navaiable options: init, stop")
return
}
switch os.Args[2] {
case "init":
var sockets [2]string
sockets[0] = filepath.Join(PacketsDir, "udpsocket")
sockets[1] = filepath.Join(PacketsDir, "httpsocket")
for _, v := range sockets {
abs, _ := filepath.Abs(v)
cmd := exec.Command(abs)
cmd.Stdout, cmd.Stderr = os.Stdout, os.Stderr
if err := cmd.Start(); err != nil {
log.Fatalf("failed to start %s: %v", v, err)
}
}
return
case "stop":
var pidfiles [2]string
pidfiles[0] = filepath.Join(PacketsDir, "http.pid")
pidfiles[1] = filepath.Join(PacketsDir, "udp.pid")
for _, v := range pidfiles {
data, err := os.ReadFile(v)
if err != nil {
fmt.Println("cant read PID:", err)
return
}
pid, _ := strconv.Atoi(string(data))
syscall.Kill(pid, syscall.SIGTERM)
}
return
default:
return
}
case "sync":
if os.Getuid() != 0 {
fmt.Println("please, run as root")
return
}
if len(os.Args) < 3 {
fmt.Println("Starting to sync with https://servidordomal.fun/mirror/index.db")
if err := Sync("https://servidordomal.fun/mirror/index.db"); err != nil {
fmt.Println("failed to sync with https://servidordomal.fun/mirror/index.db : ", err)
return
}
fmt.Println("Sucessifully sync!")
return
}
syncurl := os.Args[2]
fmt.Printf("Starting to sync with %s\n", syncurl)
if err := Sync(syncurl); err != nil {
fmt.Printf("failed to sync with %s : %e ", syncurl, err)
return
}
fmt.Println("Sucessifully sync!")
return
case "remove":
if os.Getuid() != 0 {
fmt.Println("please, run as root")
return
}
if len(os.Args) < 3 {
fmt.Println("usage: packets remove <package-name>")
return
}
err := Unninstall(os.Args[2])
if err != nil {
log.Fatal(err)
return
}
return
case "list":
if err := ListPackets(); err != nil {
return
}
return
case "upgrade":
if os.Getuid() != 0 {
fmt.Println("please, run as root")
return
}
if len(os.Args) < 3 {
fmt.Println("usage: packets upgrade <realname>")
return
}
og_realname := os.Args[2]
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
return
}
defer db.Close()
idb, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return
}
defer idb.Close()
var family string
if err := idb.QueryRow("SELECT family FROM packages WHERE realname = ?", og_realname).Scan(&family); err != nil {
log.Fatal("line 239", err)
return
}
var neo_realname string
if err := db.QueryRow("SELECT realname FROM packages WHERE family = ? ORDER BY serial DESC LIMIT 1", family).Scan(&neo_realname); err != nil {
log.Fatal("line 245", err)
return
}
if neo_realname == og_realname {
fmt.Printf("%s is up to date!\n", og_realname)
return
}
if err := db.QueryRow("SELECT serial FROM packages WHERE family = ? ORDER BY serial DESC LIMIT 1", family).Scan(&serialPass); err != nil {
log.Fatal("line 255", err)
return
}
fmt.Println("founded upgrade")
QueryInstall(neo_realname)
default:
fmt.Printf(" %s it's not a command\n", cmd)
return
}
}
func Install(packagepath string, serial uint) error {
manifest, err := internal.ManifestReadXZ(packagepath)
if err != nil {
log.Panic(err)
}
name := manifest.Info.Name
var destDir = filepath.Join(cfg.Config.DataDir, name)
if cfg.Config.LastDataDir != cfg.Config.DataDir {
fmt.Printf("Ooops... Data directory has been changed from (%s), to (%s), do you want to cancel the installation and Sync first?\n y/n? ", cfg.Config.LastDataDir, cfg.Config.DataDir)
var answer string
fmt.Scan(&answer)
if answer == "y" || answer == "Y" {
if err := os.MkdirAll(cfg.Config.DataDir, 0755); err != nil {
return err
}
bar := progressbar.NewOptions64(-1,
progressbar.OptionSetDescription("Moving ..."),
)
err := filepath.WalkDir(cfg.Config.LastDataDir, func(last string, d fs.DirEntry, walkErr error) error {
if walkErr != nil {
return walkErr
}
if last == cfg.Config.LastDataDir {
return nil
}
rel, err := filepath.Rel(cfg.Config.LastDataDir, last)
if err != nil {
return err
}
dest := filepath.Join(cfg.Config.DataDir, rel)
if d.IsDir() {
return os.MkdirAll(dest, 0755)
}
return os.Rename(last, dest)
})
if err != nil {
return err
}
f, err := os.OpenFile(filepath.Join(PacketsDir, "config.toml"), os.O_WRONLY, 0644)
if err != nil {
return err
}
cfg.Config.LastDataDir = cfg.Config.DataDir
encoder := toml.NewEncoder(f)
err = encoder.Encode(cfg)
if err != nil {
bar.Finish()
return err
}
f.WriteString("\n\n# BE CAREFULL CHANGING BIN_DIR, BECAUSE THE BINARIES DON'T MOVE AUTOMATICALLY\n#NEVER CHANGE lastDataDir\n")
os.Remove(cfg.Config.LastDataDir)
bar.Finish()
}
}
f, err := os.Open(packagepath)
if err != nil {
return err
}
defer f.Close()
stat, _ := f.Stat()
totalsize := stat.Size()
counter := &CountingReader{R: f}
zs, err := zstd.NewReader(f)
if err != nil {
return err
}
tr := tar.NewReader(zs)
bar := progressbar.NewOptions64(
totalsize,
progressbar.OptionSetDescription("[2/2] Unpacking ..."),
progressbar.OptionSetWriter(os.Stdout),
)
for {
hdr, err := tr.Next()
if err == io.EOF {
break
}
if err != nil {
return err
}
rel := filepath.Clean(hdr.Name)
if rel == ".." || strings.HasPrefix(rel, ".."+string(os.PathSeparator)) {
continue
}
if err := os.MkdirAll(destDir, 0755); err != nil {
return err
}
absPath := filepath.Join(destDir, rel)
switch hdr.Typeflag {
case tar.TypeDir:
err = os.MkdirAll(absPath, os.FileMode(hdr.Mode))
if err != nil {
return err
}
case tar.TypeReg:
err = os.MkdirAll(filepath.Dir(absPath), 0755)
if err != nil {
return err
}
out, err := os.Create(absPath)
if err != nil {
return err
}
_, err = io.Copy(io.MultiWriter(out), tr)
out.Close()
if err != nil {
return err
}
bar.Set(int(counter.Total))
err = os.Chmod(absPath, os.FileMode(hdr.Mode))
if err != nil {
return err
}
default:
}
}
L := lua.NewState()
defer L.Close()
osObject := L.GetGlobal("os").(*lua.LTable)
ioObject := L.GetGlobal("io").(*lua.LTable)
L.SetGlobal("packets_package_dir", lua.LString(cfg.Config.DataDir))
L.SetGlobal("packets_bin_dir", lua.LString(cfg.Config.BinDir))
L.SetGlobal("script", lua.LString(manifest.Hooks.Install))
L.SetGlobal("data_dir", lua.LString(filepath.Join(cfg.Config.DataDir, name, "data")))
L.SetGlobal("path_join", L.NewFunction(internal.Ljoin))
osObject.RawSetString("execute", lua.LNil)
osObject.RawSetString("exit", lua.LNil)
osObject.RawSetString("getenv", lua.LNil)
osObject.RawSetString("remove", L.NewFunction(internal.SafeRemove))
osObject.RawSetString("rename", L.NewFunction(internal.SafeRename))
osObject.RawSetString("copy", L.NewFunction(internal.SafeCopy))
osObject.RawSetString("symlink", L.NewFunction(internal.SymbolicLua))
ioObject.RawSetString("input", lua.LNil)
ioObject.RawSetString("output", lua.LNil)
ioObject.RawSetString("popen", lua.LNil)
ioObject.RawSetString("tmpfile", lua.LNil)
ioObject.RawSetString("stdout", lua.LNil)
ioObject.RawSetString("stdeer", lua.LNil)
ioObject.RawSetString("stdin", lua.LNil)
ioObject.RawSetString("lines", lua.LNil)
ioObject.RawSetString("open", L.NewFunction(internal.SafeOpen))
if err := L.DoFile(manifest.Hooks.Install); err != nil {
log.Panic(err)
}
bar.Finish()
fmt.Printf("Package %s fully installed\n", name)
var insert = Installed{
Realname: manifest.Info.Name,
Version: manifest.Info.Version,
Dependencies: manifest.Info.Dependencies,
Family: manifest.Info.Family,
Serial: manifest.Info.Serial,
}
if err := AddToInstalledDB(insert); err != nil {
log.Fatal(err)
return err
}
return nil
}
func GetPackageByMirror(mirror string, realname string) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
var serial uint
if err := db.QueryRow("SELECT serial FROM packages WHERE realname = ?", realname).Scan(&serial); err != nil {
log.Fatal(err)
}
u, _ := url.Parse(mirror)
filename := path.Base(u.Path)
resp, err := http.Get(mirror)
if err != nil {
log.Panic("error doing get request, do you really have an internet connection?")
return err
}
var domain = mirror
var link bool
if cont := strings.Contains(mirror, "https"); cont {
link = true
domain = strings.Replace(mirror, "https", "", 1)
} else {
link = true
domain = strings.Replace(mirror, "http", "", 1)
}
if link {
domain = strings.Replace(domain, "://", "", 1)
slice := strings.SplitN(domain, "/", 2)
domain = slice[0]
}
bar := progressbar.NewOptions64(resp.ContentLength,
progressbar.OptionSetDescription(fmt.Sprintf("[1/2] Downloading from %s ...", domain)),
progressbar.OptionShowBytes(true),
progressbar.OptionSetPredictTime(true),
progressbar.OptionShowCount(),
progressbar.OptionClearOnFinish(),
)
defer resp.Body.Close()
if resp.StatusCode != 200 {
err := os.Remove(filepath.Join(cfg.Config.CacheDir, filename))
if os.IsNotExist(err) {
return fmt.Errorf("failed to download, status code not 200OK")
} else if err != nil {
return err
}
return fmt.Errorf("failed to download, status code not 200OK")
}
if err := os.MkdirAll(cfg.Config.CacheDir, 0755); err != nil {
log.Fatal("error creating file for package ", err)
return err
}
out, err := os.Create(filepath.Join(cfg.Config.CacheDir, filename))
if err != nil {
log.Fatal("error creating package ", err)
return err
}
defer out.Close()
_, err = io.Copy(io.MultiWriter(out, bar), resp.Body)
if err != nil {
err := os.Remove(filepath.Join(cfg.Config.CacheDir, filename))
if err != nil {
return err
}
return err
}
bar.Finish()
err = Validate(filename, realname)
if err != nil {
return err
}
if os.Args[1] == "upgrade" {
if err := Upgrade(filepath.Join(cfg.Config.CacheDir, filename), os.Args[2], serialPass); err != nil {
return err
}
return nil
}
err = Install(filepath.Join(cfg.Config.CacheDir, filename), serial)
if err != nil {
return err
}
return nil
}
func ResolvDependencies(realname string) {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
}
defer db.Close()
var dependencies *string
err = db.QueryRow("SELECT dependencies FROM packages WHERE realname = ?", realname).Scan(&dependencies)
if err != nil {
log.Panic(err)
return
}
if dependencies == nil {
return
}
dependencie := strings.Fields(*dependencies)
for _, v := range dependencie {
err := AlredySatisfied(v)
if err != nil {
fmt.Printf("error installing %v : %s", v, err.Error())
continue
}
QueryInstall(v)
}
}
func QueryInstall(realname string) {
_, err := os.Stat(filepath.Join(PacketsDir, "index.db"))
if os.IsNotExist(err) {
fmt.Println("cant find index.db, please use sync first")
}
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal("cant find index.db, please use sync first")
}
defer db.Close()
simplecheck, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err == nil {
var exist bool
simplecheck.QueryRow("SELECT EXISTS(SELECT 1 FROM packages WHERE realname = ? LIMIT 1)", realname).Scan(&exist)
if exist {
fmt.Println("Alredy installed!")
simplecheck.Close()
return
}
simplecheck.Close()
}
var mirrors string
ResolvDependencies(realname)
err = db.QueryRow("SELECT mirrors FROM packages WHERE realname = ?", realname).Scan(&mirrors)
if err != nil {
log.Panic(err)
return
}
var serial uint
err = db.QueryRow("SELECT serial FROM packages WHERE realname = ?", realname).Scan(&serial)
if err != nil {
log.Panic(err)
return
}
if !strings.Contains(mirrors, " ") {
u, _ := url.Parse(mirrors)
filename := path.Base(u.Path)
fmt.Println("Checking if the package exists")
if CheckDownloaded(filename) {
err := Validate(filename, realname)
if err != nil {
return
}
if os.Args[1] == "upgrade" {
if err := Upgrade(filepath.Join(cfg.Config.CacheDir, filename), os.Args[2], serialPass); err != nil {
log.Fatal(err)
return
}
return
}
Install(filepath.Join(cfg.Config.CacheDir, filename), serial)
return
}
fmt.Println("Asking in LAN for the package")
peers := AskLAN(filename)
answers := len(peers)
if answers != 0 {
for _, p := range peers {
fmt.Printf("Downloading from %s\n", p.IP)
if err := GetPackageByMirror(fmt.Sprintf("http://%s:%d/%s", p.IP, p.Port, filename), realname); err != nil {
log.Println(err)
break
}
fmt.Printf("Download failed!\n")
}
}
if err := GetPackageByMirror(mirrors, realname); err != nil {
log.Println(err)
return
}
} else {
fmt.Println("A mirror list was found")
mirrorlist := strings.Fields(mirrors)
for _, v := range mirrorlist {
u, _ := url.Parse(v)
filename := path.Base(u.Path)
fmt.Printf("Checking for %s", filename)
if CheckDownloaded(filename) {
err := Validate(filename, realname)
if err != nil {
continue
} else {
if os.Args[1] == "upgrade" {
if err := Upgrade(filepath.Join(cfg.Config.CacheDir, filename), os.Args[2], serialPass); err != nil {
log.Fatal(err)
return
}
break
}
Install(filepath.Join(cfg.Config.CacheDir, filename), serial)
break
}
}
fmt.Println("Checking for package in LAN")
peers := AskLAN(filename)
answers := len(peers)
if answers != 0 {
for _, p := range peers {
fmt.Printf("Downloading from %s\n", v)
if err := GetPackageByMirror(fmt.Sprintf("http://%s:%d/%s", p.IP, p.Port, filename), realname); err == nil {
break
}
fmt.Printf("Failed!\n")
}
}
fmt.Printf("Downloading from %s\n", v)
if err := GetPackageByMirror(v, realname); err != nil {
log.Println(err)
break
}
}
}
}
func CheckDownloaded(filename string) bool {
_, err := os.Stat(filepath.Join(cfg.Config.CacheDir, filename))
if os.IsNotExist(err) {
return false
} else {
return true
}
}
func Validate(filename string, realname string) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
downloaded, err := os.Open(filepath.Join(cfg.Config.CacheDir, filename))
if err != nil {
log.Fatal("error reading new file")
return err
}
defer downloaded.Close()
h := sha256.New()
if _, err := io.Copy(h, downloaded); err != nil {
fmt.Println("error doing sha256sum")
return err
}
sum := h.Sum(nil)
hashString := hex.EncodeToString(sum)
var hashStringDB string
err = db.QueryRow("SELECT hash FROM packages WHERE realname = ?", realname).Scan(&hashStringDB)
if err != nil {
log.Panic(err)
return err
}
if hashString != hashStringDB {
fmt.Println("tampered package, removing it...\nplease run the command again")
err := os.Remove(filepath.Join(cfg.Config.CacheDir, filename))
if err != nil {
return err
}
return fmt.Errorf("the package isn't safe, alredy removed")
}
return nil
}
func AskLAN(filename string) []Peer {
var peers []Peer
query := []byte("Q:" + filename)
pc, err := net.ListenPacket("udp", ":0")
if err != nil {
log.Panic("error starting udp socket:", err)
}
defer pc.Close()
if pconn := ipv4.NewPacketConn(pc); pconn != nil {
_ = pconn.SetTTL(1)
}
ifaces, _ := net.Interfaces()
for _, ifc := range ifaces {
if ifc.Flags&net.FlagUp == 0 || ifc.Flags&net.FlagLoopback != 0 {
continue
}
addrs, _ := ifc.Addrs()
for _, a := range addrs {
ipnet, ok := a.(*net.IPNet)
if !ok || ipnet.IP.To4() == nil {
continue
}
bcast := broadcastAddr(ipnet.IP.To4(), ipnet.Mask)
dst := &net.UDPAddr{IP: bcast, Port: 1333}
_, err = pc.WriteTo(query, dst)
if err != nil {
log.Printf("[%s] can't send to %s: %v", ifc.Name, bcast, err)
}
}
}
_ = pc.SetDeadline(time.Now().Add(2 * time.Second))
buf := make([]byte, 1500)
for {
n, addr, err := pc.ReadFrom(buf)
if err != nil {
break
}
msg := string(buf[:n])
if strings.HasPrefix(msg, "H:"+filename) {
parts := strings.Split(msg, ":")
port, _ := strconv.Atoi(parts[2])
fmt.Printf("%s have the package\n", addr)
peers = append(peers, Peer{IP: addr.(*net.UDPAddr).IP, Port: port})
}
}
return peers
}
func broadcastAddr(ip net.IP, mask net.IPMask) net.IP {
b := make(net.IP, len(ip))
for i := range ip {
b[i] = ip[i] | ^mask[i]
}
return b
}
func Sync(url string) error {
resp, err := http.Get(url)
if err != nil {
return err
}
defer resp.Body.Close()
_, err = os.Stat(filepath.Join(PacketsDir, "index.db"))
if os.IsNotExist(err) {
os.MkdirAll(PacketsDir, 0755)
}
f, err := os.Create(filepath.Join(PacketsDir, "index.db"))
if err != nil {
return err
}
if _, err = io.Copy(f, resp.Body); err != nil {
return err
}
if cfg.Config.LastDataDir != cfg.Config.DataDir {
fmt.Printf("Ooops... Data directory has been changed on %s do you want to move the packages from (%s), to (%s)\n", filepath.Join(PacketsDir, "config.toml"), cfg.Config.LastDataDir, cfg.Config.DataDir)
fmt.Println("What you want to do?")
fmt.Println("[y] Yes [n] No, [x] Ignore it and stop to show this message (not recommended)")
}
var answer string
fmt.Scanln(&answer)
switch answer {
case "n":
fmt.Println("Just ignoring...")
case "x":
f, err := os.OpenFile(filepath.Join(PacketsDir, "config.toml"), os.O_WRONLY, 0644)
if err != nil {
return err
}
cfg.Config.LastDataDir = cfg.Config.DataDir
encoder := toml.NewEncoder(f)
err = encoder.Encode(cfg)
if err != nil {
return err
}
f.WriteString("\n\n# BE CAREFULL CHANGING BIN_DIR, BECAUSE THE BINARIES DON'T MOVE AUTOMATICALLY\n#NEVER CHANGE lastDataDir\n")
os.Remove(cfg.Config.LastDataDir)
case "y":
if err := os.MkdirAll(cfg.Config.DataDir, 0755); err != nil {
return err
}
bar := progressbar.NewOptions64(-1,
progressbar.OptionSetDescription("Moving ..."),
)
err := filepath.WalkDir(cfg.Config.LastDataDir, func(last string, d fs.DirEntry, walkErr error) error {
if walkErr != nil {
return walkErr
}
if last == cfg.Config.LastDataDir {
return nil
}
rel, err := filepath.Rel(cfg.Config.LastDataDir, last)
if err != nil {
return err
}
dest := filepath.Join(cfg.Config.DataDir, rel)
if d.IsDir() {
return os.MkdirAll(dest, 0755)
}
return os.Rename(last, dest)
})
if err != nil {
return err
}
f, err := os.OpenFile(filepath.Join(PacketsDir, "config.toml"), os.O_WRONLY, 0644)
if err != nil {
return err
}
cfg.Config.LastDataDir = cfg.Config.DataDir
encoder := toml.NewEncoder(f)
err = encoder.Encode(cfg)
if err != nil {
bar.Finish()
return err
}
f.WriteString("\n\n# BE CAREFULL CHANGING BIN_DIR, BECAUSE THE BINARIES DON'T MOVE AUTOMATICALLY\n#NEVER CHANGE lastDataDir\n")
os.Remove(cfg.Config.LastDataDir)
bar.Finish()
default:
if err := os.MkdirAll(cfg.Config.DataDir, 0755); err != nil {
return err
}
bar := progressbar.NewOptions64(-1,
progressbar.OptionSetDescription("Moving ..."),
)
err := filepath.WalkDir(cfg.Config.LastDataDir, func(last string, d fs.DirEntry, walkErr error) error {
if walkErr != nil {
return walkErr
}
if last == cfg.Config.LastDataDir {
return nil
}
rel, err := filepath.Rel(cfg.Config.LastDataDir, last)
if err != nil {
return err
}
dest := filepath.Join(cfg.Config.DataDir, rel)
if d.IsDir() {
return os.MkdirAll(dest, 0755)
}
return os.Rename(last, dest)
})
if err != nil {
return err
}
bar.Finish()
}
return nil
}
func AddToInstalledDB(insert Installed) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
_, err = db.Exec("CREATE TABLE IF NOT EXISTS packages (realname TEXT NOT NULL UNIQUE PRIMARY KEY, version TEXT NOT NULL, dependencies TEXT, name TEXT, family TEXT NOT NULL, serial INTEGER)")
if err != nil {
return err
}
if len(insert.Dependencies) == 0 {
_, err = db.Exec("INSERT INTO packages (realname, version, family, serial) VALUES (?, ?, ?, ?)", insert.Realname, insert.Version, insert.Family, insert.Serial)
if err != nil {
return err
}
return nil
}
var query string
for _, v := range insert.Dependencies {
query = query + v + " "
}
query = query[:len(query)-1]
_, err = db.Exec("INSERT INTO packages (realname, version, dependencies, family, serial) VALUES (?, ?, ?, ?, ?)", insert.Realname, insert.Version, query, insert.Family, insert.Serial)
if err != nil {
return err
}
return nil
}
func Unninstall(realname string) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
var exist bool
err = db.QueryRow("SELECT EXISTS(SELECT 1 FROM packages WHERE realname = ? LIMIT 1)", realname).Scan(&exist)
if err != nil {
return err
}
if !exist {
return fmt.Errorf("this package isn't installed")
}
fmt.Printf("Sure you will remove %s ? [Y/n] ", realname)
var answer string
fmt.Scanf("%s", &answer)
if answer != "y" && answer != "Y" {
return fmt.Errorf("operation cancelled")
}
var manifest Manifest
toml.DecodeFile(filepath.Join(cfg.Config.DataDir, realname, "manifest.toml"), &manifest)
L := lua.NewState()
defer L.Close()
osObject := L.GetGlobal("os").(*lua.LTable)
ioObject := L.GetGlobal("io").(*lua.LTable)
L.SetGlobal("packets_package_dir", lua.LString(cfg.Config.DataDir))
L.SetGlobal("packets_bin_dir", lua.LString(cfg.Config.BinDir))
L.SetGlobal("script", lua.LString(manifest.Hooks.Remove))
L.SetGlobal("data_dir", lua.LString(filepath.Join(cfg.Config.DataDir, realname, "data")))
L.SetGlobal("path_join", L.NewFunction(internal.Ljoin))
osObject.RawSetString("execute", lua.LNil)
osObject.RawSetString("exit", lua.LNil)
osObject.RawSetString("getenv", lua.LNil)
osObject.RawSetString("remove", L.NewFunction(internal.SafeRemove))
osObject.RawSetString("rename", L.NewFunction(internal.SafeRename))
osObject.RawSetString("copy", L.NewFunction(internal.SafeCopy))
osObject.RawSetString("symlink", L.NewFunction(internal.SymbolicLua))
ioObject.RawSetString("input", lua.LNil)
ioObject.RawSetString("output", lua.LNil)
ioObject.RawSetString("popen", lua.LNil)
ioObject.RawSetString("tmpfile", lua.LNil)
ioObject.RawSetString("stdout", lua.LNil)
ioObject.RawSetString("stdeer", lua.LNil)
ioObject.RawSetString("stdin", lua.LNil)
ioObject.RawSetString("lines", lua.LNil)
ioObject.RawSetString("open", L.NewFunction(internal.SafeOpen))
if err := L.DoFile(manifest.Hooks.Remove); err != nil {
log.Panic(err)
}
if err := os.RemoveAll(filepath.Join(cfg.Config.DataDir, realname)); err != nil {
return err
}
_, err = db.Exec("DELETE FROM packages WHERE realname = ?", realname)
if err != nil {
return err
}
fmt.Println("Sucessifully removed")
return nil
}
func AlredySatisfied(realname string) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
var exist bool
err = db.QueryRow("SELECT EXISTS(SELECT 1 FROM packages WHERE realname = ? LIMIT 1)", realname).Scan(&exist)
if err != nil {
return err
}
if !exist {
return nil
}
return fmt.Errorf("conflict")
}
func ListPackets() error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
rows, err := db.Query("SELECT realname, version FROM packages")
if err != nil {
return err
}
var realname string
var version string
defer rows.Close()
fmt.Println("Installed packages:")
for rows.Next() {
rows.Scan(&realname, &version)
fmt.Printf("%s %s\n", realname, version)
}
return nil
}
func Upgrade(packagepath string, og_realname string, serial uint) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "installed.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
var exist bool
err = db.QueryRow("SELECT EXISTS(SELECT 1 FROM packages WHERE realname = ? LIMIT 1)", og_realname).Scan(&exist)
if err != nil {
return err
}
if !exist {
return fmt.Errorf("this package isn't installed")
}
if cfg.Config.LastDataDir != cfg.Config.DataDir {
fmt.Printf("Ooops... Data directory has been changed from (%s), to (%s), do you want to cancel the installation and Sync first?\n y/n? ", cfg.Config.LastDataDir, cfg.Config.DataDir)
var answer string
fmt.Scan(&answer)
if answer == "y" || answer == "Y" {
if err := os.MkdirAll(cfg.Config.DataDir, 0755); err != nil {
return err
}
bar := progressbar.NewOptions64(-1,
progressbar.OptionSetDescription("Moving ..."),
)
err := filepath.WalkDir(cfg.Config.LastDataDir, func(last string, d fs.DirEntry, walkErr error) error {
if walkErr != nil {
return walkErr
}
if last == cfg.Config.LastDataDir {
return nil
}
rel, err := filepath.Rel(cfg.Config.LastDataDir, last)
if err != nil {
return err
}
dest := filepath.Join(cfg.Config.DataDir, rel)
if d.IsDir() {
return os.MkdirAll(dest, 0755)
}
return os.Rename(last, dest)
})
if err != nil {
return err
}
f, err := os.OpenFile(filepath.Join(PacketsDir, "config.toml"), os.O_WRONLY, 0644)
if err != nil {
return err
}
cfg.Config.LastDataDir = cfg.Config.DataDir
encoder := toml.NewEncoder(f)
err = encoder.Encode(cfg)
if err != nil {
bar.Finish()
return err
}
f.WriteString("\n\n# BE CAREFULL CHANGING BIN_DIR, BECAUSE THE BINARIES DON'T MOVE AUTOMATICALLY\n#NEVER CHANGE lastDataDir\n")
os.Remove(cfg.Config.LastDataDir)
bar.Finish()
}
}
manifest, err := internal.ManifestReadXZ(packagepath)
if err != nil {
log.Panic(err)
}
name := manifest.Info.Name
fmt.Printf("Unpacking (%s) above (%s)\n", name, og_realname)
var destDir = filepath.Join(cfg.Config.DataDir, og_realname)
f, err := os.Open(packagepath)
if err != nil {
return err
}
stats, _ := f.Stat()
totalSize := stats.Size()
defer f.Close()
counter := &CountingReader{R: f}
zs, err := zstd.NewReader(f)
if err != nil {
return err
}
tr := tar.NewReader(zs)
bar := progressbar.NewOptions64(
totalSize,
progressbar.OptionSetDescription("[2/2] Upgrading ..."),
progressbar.OptionSetWriter(os.Stdout),
)
for {
hdr, err := tr.Next()
if err == io.EOF {
break
}
if err != nil {
return err
}
rel := filepath.Clean(hdr.Name)
if rel == ".." || strings.HasPrefix(rel, ".."+string(os.PathSeparator)) {
continue
}
if err := os.MkdirAll(destDir, 0755); err != nil {
return err
}
absPath := filepath.Join(destDir, rel)
switch hdr.Typeflag {
case tar.TypeDir:
err = os.MkdirAll(absPath, os.FileMode(hdr.Mode))
if err != nil {
return err
}
case tar.TypeReg:
err = os.MkdirAll(filepath.Dir(absPath), 0755)
if err != nil {
return err
}
out, err := os.Create(absPath)
if err != nil {
return err
}
_, err = io.Copy(out, tr)
out.Close()
if err != nil {
return err
}
bar.Set(int(counter.Total))
err = os.Chmod(absPath, os.FileMode(hdr.Mode))
if err != nil {
return err
}
default:
}
}
bar.Finish()
os.Rename(destDir, filepath.Join(cfg.Config.DataDir, name))
L := lua.NewState()
defer L.Close()
osObject := L.GetGlobal("os").(*lua.LTable)
ioObject := L.GetGlobal("io").(*lua.LTable)
L.SetGlobal("packets_package_dir", lua.LString(cfg.Config.DataDir))
L.SetGlobal("packets_bin_dir", lua.LString(cfg.Config.BinDir))
L.SetGlobal("script", lua.LString(manifest.Hooks.Install))
L.SetGlobal("data_dir", lua.LString(filepath.Join(cfg.Config.DataDir, name, "data")))
L.SetGlobal("path_join", L.NewFunction(internal.Ljoin))
osObject.RawSetString("execute", lua.LNil)
osObject.RawSetString("exit", lua.LNil)
osObject.RawSetString("getenv", lua.LNil)
osObject.RawSetString("remove", L.NewFunction(internal.SafeRemove))
osObject.RawSetString("rename", L.NewFunction(internal.SafeRename))
osObject.RawSetString("copy", L.NewFunction(internal.SafeCopy))
osObject.RawSetString("symlink", L.NewFunction(internal.SymbolicLua))
ioObject.RawSetString("input", lua.LNil)
ioObject.RawSetString("output", lua.LNil)
ioObject.RawSetString("popen", lua.LNil)
ioObject.RawSetString("tmpfile", lua.LNil)
ioObject.RawSetString("stdout", lua.LNil)
ioObject.RawSetString("stdeer", lua.LNil)
ioObject.RawSetString("stdin", lua.LNil)
ioObject.RawSetString("lines", lua.LNil)
ioObject.RawSetString("open", L.NewFunction(internal.SafeOpen))
if err := L.DoFile(manifest.Hooks.Install); err != nil {
log.Panic(err)
}
fmt.Printf("Package %s fully installed", name)
var insert = Installed{
Realname: manifest.Info.Name,
Version: manifest.Info.Version,
Dependencies: manifest.Info.Dependencies,
Family: manifest.Info.Family,
Serial: manifest.Info.Serial,
}
_, err = db.Exec("DELETE FROM packages WHERE realname = ?", og_realname)
if err != nil {
return err
}
if err := AddToInstalledDB(insert); err != nil {
return err
}
return nil
}
func SearchUpgrades(name string) error {
db, err := sql.Open("sqlite", filepath.Join(PacketsDir, "index.db"))
if err != nil {
log.Fatal(err)
return err
}
defer db.Close()
return nil
}