Compare commits
14 Commits
f89e7112a4
...
c_rewrite
Author | SHA1 | Date | |
---|---|---|---|
8024e86962
|
|||
88742c2ea5
|
|||
d101e186c4
|
|||
2d881a0d34
|
|||
cb735a01a1
|
|||
e3c9924fbc
|
|||
3645f7f30c
|
|||
66ca74441d
|
|||
7bca34cc89
|
|||
e4ab48fb22
|
|||
67c7f48cd7
|
|||
e2d4253050
|
|||
81b07991aa
|
|||
452b0e92e2
|
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
# Build artifacts
|
||||
update
|
27
README.md
27
README.md
@ -1,28 +1,33 @@
|
||||
# Update
|
||||
|
||||
Update script written in Bash for Arch Linux only. Keeps all your official, aur packages and your flatpaks up to date with one simple script. It also backups a list of all your installed packages and flatpaks.
|
||||
Alpm hooks for the pacman package manager.<br>
|
||||
This project consists of two hooks, one for pre-transaction and the other for
|
||||
post-transaction.
|
||||
|
||||
## Installation
|
||||
|
||||
```bash
|
||||
git clone https://gitea.hopeless-cloud.xyz/AustrianToast/update.git && cd update
|
||||
git clone https://gitea.hopeless-cloud.xyz/AustrianToast/update.git
|
||||
cd update
|
||||
```
|
||||
|
||||
Before installing, please edit the config and configure it to your liking.
|
||||
|
||||
Then install using
|
||||
```bash
|
||||
make
|
||||
make install
|
||||
```
|
||||
|
||||
## Usage
|
||||
If your require the previous functionality of being able to update flatpak and
|
||||
having the ability to preview the updateable packages and flatpaks.<br>
|
||||
This functionality has been re-packaged into `update.zsh`.<br>
|
||||
If you use oh-my-zsh, then you just throw this file into your `$ZSH_CUSTOM`.<br>
|
||||
This path is by default `~/.oh-my-zsh/custom`. If you use only plain zsh or any other
|
||||
shell, then you can just take the contained functions and put them inside the
|
||||
according shell config file. For example your `.bashrc`.
|
||||
|
||||
```
|
||||
Usage: update [OPTION]
|
||||
options:
|
||||
--help displays this message
|
||||
--preview shows a preview of which pkg's will be updates
|
||||
--version prints out the version number
|
||||
```
|
||||
The function `flatpak-update` contains a variable called `BACKUP_LOCATION`.<br>
|
||||
Please change this path to your desired backup location.
|
||||
|
||||
## Contributing
|
||||
Contributions are always welcome!
|
||||
|
164
backup.go
164
backup.go
@ -1,164 +0,0 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"archive/zip"
|
||||
"bufio"
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"os/exec"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/pelletier/go-toml"
|
||||
)
|
||||
|
||||
var backup_locations []string
|
||||
|
||||
func parsePacmanConf() (string, error) {
|
||||
var err error
|
||||
var path string
|
||||
|
||||
file, err := os.Open("/etc/pacman.conf")
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
scanner := bufio.NewScanner(file)
|
||||
for scanner.Scan() {
|
||||
line := scanner.Text()
|
||||
if !strings.Contains(line, "DBPath") {
|
||||
continue
|
||||
}
|
||||
path = strings.TrimSpace(strings.Split(line, "=")[1])
|
||||
break
|
||||
}
|
||||
|
||||
err = file.Close()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return fmt.Sprint(path, "local"), nil
|
||||
}
|
||||
|
||||
func zipIt(pathToZip string, pathToFiles string) error {
|
||||
var err error
|
||||
|
||||
zipFile, err := os.Create(pathToZip)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
zipWriter := zip.NewWriter(zipFile)
|
||||
|
||||
walker := func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if info.IsDir() {
|
||||
return nil
|
||||
}
|
||||
file, err := os.Open(path)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
f, err := zipWriter.Create(path[len(pathToFiles):])
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
_, err = io.Copy(f, file)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
err = filepath.Walk(pathToFiles, walker)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = zipWriter.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = zipFile.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func backup(when string) error {
|
||||
var err error
|
||||
tmpPath := fmt.Sprint("/tmp/update/", when, "_backup")
|
||||
pacmanDb, err := parsePacmanConf()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = os.MkdirAll(tmpPath, 0755)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer func() {
|
||||
os.RemoveAll("/tmp/update")
|
||||
}()
|
||||
|
||||
output, err := exec.Command("pacman", "--verbose", "--query").Output()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = os.WriteFile(fmt.Sprint(tmpPath, fmt.Sprint(when, "_pacman.txt")), output, 0644)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if when != "post" {
|
||||
err = CopyDir(pacmanDb, fmt.Sprint(tmpPath, "/", pacmanDb))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
current_time := time.Now().Format(time.RFC3339)
|
||||
pathToZip := fmt.Sprint(tmpPath, "_", current_time, ".zip")
|
||||
err = zipIt(pathToZip, tmpPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
for _, backup_location := range backup_locations {
|
||||
err = CopyFile(pathToZip, fmt.Sprint(backup_location, "/", when, "_backup_", current_time, ".zip"))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func main() {
|
||||
os.RemoveAll("/tmp/update")
|
||||
|
||||
tree, err := toml.LoadFile("/etc/update.toml")
|
||||
if err != nil {
|
||||
fmt.Println("backup failed for the following reason: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
|
||||
backup_locations = tree.GetArray("backup.locations").([]string)
|
||||
|
||||
err = backup(os.Args[1])
|
||||
if err != nil {
|
||||
fmt.Println("backup failed for the following reason: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
}
|
134
copy.go
134
copy.go
@ -1,134 +0,0 @@
|
||||
/* MIT License
|
||||
*
|
||||
* Copyright (c) 2017 Roland Singer [roland.singer@desertbit.com]
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
// CopyFile copies the contents of the file named src to the file named
|
||||
// by dst. The file will be created if it does not already exist. If the
|
||||
// destination file exists, all it's contents will be replaced by the contents
|
||||
// of the source file. The file mode will be copied from the source and
|
||||
// the copied data is synced/flushed to stable storage.
|
||||
func CopyFile(src, dst string) (err error) {
|
||||
in, err := os.Open(src)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
defer in.Close()
|
||||
|
||||
out, err := os.Create(dst)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
defer func() {
|
||||
if e := out.Close(); e != nil {
|
||||
err = e
|
||||
}
|
||||
}()
|
||||
|
||||
_, err = io.Copy(out, in)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
err = out.Sync()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
si, err := os.Stat(src)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
err = os.Chmod(dst, si.Mode())
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
// CopyDir recursively copies a directory tree, attempting to preserve permissions.
|
||||
// Source directory must exist, destination directory must *not* exist.
|
||||
// Symlinks are ignored and skipped.
|
||||
func CopyDir(src string, dst string) (err error) {
|
||||
src = filepath.Clean(src)
|
||||
dst = filepath.Clean(dst)
|
||||
|
||||
si, err := os.Stat(src)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if !si.IsDir() {
|
||||
return fmt.Errorf("source is not a directory")
|
||||
}
|
||||
|
||||
_, err = os.Stat(dst)
|
||||
if err != nil && !os.IsNotExist(err) {
|
||||
return
|
||||
}
|
||||
if err == nil {
|
||||
return fmt.Errorf("destination already exists")
|
||||
}
|
||||
|
||||
err = os.MkdirAll(dst, si.Mode())
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
entries, err := ioutil.ReadDir(src)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
for _, entry := range entries {
|
||||
srcPath := filepath.Join(src, entry.Name())
|
||||
dstPath := filepath.Join(dst, entry.Name())
|
||||
|
||||
if entry.IsDir() {
|
||||
err = CopyDir(srcPath, dstPath)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
} else {
|
||||
// Skip symlinks.
|
||||
if entry.Mode()&os.ModeSymlink != 0 {
|
||||
continue
|
||||
}
|
||||
|
||||
err = CopyFile(srcPath, dstPath)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return
|
||||
}
|
5
go.mod
5
go.mod
@ -1,5 +0,0 @@
|
||||
module backup
|
||||
|
||||
go 1.23.0
|
||||
|
||||
require github.com/pelletier/go-toml v1.9.5
|
2
go.sum
2
go.sum
@ -1,2 +0,0 @@
|
||||
github.com/pelletier/go-toml v1.9.5 h1:4yBQzkHv+7BHq2PQUZF3Mx0IYxG7LsP222s7Agd3ve8=
|
||||
github.com/pelletier/go-toml v1.9.5/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c=
|
10
justfile
10
justfile
@ -1,10 +0,0 @@
|
||||
install:
|
||||
@echo "==> Installing update into /usr/local/bin"
|
||||
@sudo install -Dm755 update /usr/local/bin/update
|
||||
@[[ -f ${HOME}/.config/update.conf ]] || cp update.conf ${HOME}/.config/
|
||||
@echo "==> Finished."
|
||||
|
||||
uninstall:
|
||||
@echo "==> Uninstalling update from /usr/local/bin"
|
||||
@sudo rm /usr/local/bin/update ${HOME}/.config/update.conf
|
||||
@echo "==> Finished."
|
28
makefile
28
makefile
@ -1,28 +0,0 @@
|
||||
help:
|
||||
@echo "Usage: make [OPTION]"
|
||||
@echo "Available options are:"
|
||||
@echo "help"
|
||||
@echo "install"
|
||||
@echo "uninstall"
|
||||
|
||||
compile:
|
||||
@echo "==> Compiling backup"
|
||||
go build .
|
||||
|
||||
install: compile
|
||||
@echo "==> Copying the hooks into /etc/pacman.d/hooks"
|
||||
[[ -d /etc/pacman.d/hooks ]] || sudo mkdir /etc/pacman.d/hooks
|
||||
sudo cp pre_backup.hook post_backup.hook /etc/pacman.d/hooks
|
||||
@echo "==> Copying backup into /usr/local/bin"
|
||||
sudo install -Dm755 backup /usr/local/bin/backup
|
||||
@echo "==> Copying the config into /etc"
|
||||
[[ -f /etc/update.toml ]] || sudo cp update.toml /etc
|
||||
@echo "==> Finished."
|
||||
|
||||
uninstall:
|
||||
@echo "==> Removing the hooks from /etc/pacman.d/hooks"
|
||||
sudo rm /etc/pacman.d/hooks/pre_backup.hook /etc/pacman.d/hooks/post_backup.hook
|
||||
@echo "==> Removing backup into /usr/local/bin"
|
||||
sudo rm /usr/local/bin/backup
|
||||
@echo "==> /etc/update.toml will remain"
|
||||
@echo "==> Finished."
|
@ -1,11 +0,0 @@
|
||||
[Trigger]
|
||||
Operation = Install
|
||||
Operation = Upgrade
|
||||
Operation = Remove
|
||||
Type = Package
|
||||
Target = *
|
||||
|
||||
[Action]
|
||||
Description = Backing up the pacman db...
|
||||
When = PostTransaction
|
||||
Exec = /usr/local/bin/backup post
|
@ -1,12 +0,0 @@
|
||||
[Trigger]
|
||||
Operation = Install
|
||||
Operation = Upgrade
|
||||
Operation = Remove
|
||||
Type = Package
|
||||
Target = *
|
||||
|
||||
[Action]
|
||||
Description = Backing up the pacman db...
|
||||
When = PreTransaction
|
||||
Exec = /usr/local/bin/backup pre
|
||||
AbortOnFail
|
141
update
141
update
@ -1,141 +0,0 @@
|
||||
#!/usr/bin/bash
|
||||
|
||||
VERSION="3.2"
|
||||
IGREEN="\033[0;92m" # Intense Green
|
||||
IYELLOW="\033[0;93m" # Intense Yellow
|
||||
IRED='\033[0;91m' # Red
|
||||
NO_COLOR="\033[0m" # Text Reset
|
||||
TMP="/tmp/update"
|
||||
DATE="$(date +"%Y-%m-%dT%H:%M:%S%:z")" # RFC 3339 date-time https://datatracker.ietf.org/doc/html/rfc3339#section-5.6
|
||||
|
||||
trap interrupt_function INT
|
||||
|
||||
interrupt_function() {
|
||||
echo -e "${IRED}\nInterrupt has been detected${NO_COLOR}"
|
||||
if [ -f /var/lib/pacman/db.lck ]; then
|
||||
echo -e "${IRED}Trying to unlock the pacman db. Please enter your password.${NO_COLOR}"
|
||||
su --command="rm --force /var/lib/pacman/db.lck"
|
||||
fi
|
||||
[[ -d $TMP ]] && rm --recursive --force "$TMP"
|
||||
[[ -f "$BACKUP_LOCATION"/before-backup_"$DATE".tar.zst.new ]] && rm --force "$BACKUP_LOCATION"/before-backup_"$DATE".tar.zst.new
|
||||
[[ -f "$BACKUP_LOCATION"/after-backup_"$DATE".tar.zst.new ]] && rm --force "$BACKUP_LOCATION"/after-backup_"$DATE".tar.zst.new
|
||||
exit 1
|
||||
}
|
||||
|
||||
help() {
|
||||
echo "Usage: update [OPTION]"
|
||||
echo "options:"
|
||||
echo "--help displays this message"
|
||||
echo "--preview shows a preview of pkg's and flatpaks which can be updated"
|
||||
echo "--version prints out the version number"
|
||||
}
|
||||
|
||||
source "$HOME"/.config/update.conf || echo -e "${IRED}No config found${NO_COLOR}" || exit 1
|
||||
|
||||
if [[ ${1:0:2} == \-\- ]]; then
|
||||
case "${1:2}" in
|
||||
help)
|
||||
help ;;
|
||||
preview)
|
||||
"$PACMAN_WRAPPER" -Syy
|
||||
"$PACMAN_WRAPPER" --query --upgrades
|
||||
[[ -x /usr/bin/flatpak ]] && flatpak remote-ls --updates ;;
|
||||
version)
|
||||
echo "$VERSION" ;;
|
||||
?)
|
||||
help; exit 1;;
|
||||
esac
|
||||
exit 0
|
||||
fi
|
||||
|
||||
check_for_valid_backup_location() {
|
||||
if [ ! -d $1 ]; then
|
||||
echo -e "${IYELLOW}$1 doesn't exist${NO_COLOR}"
|
||||
read -p "Do you want to create the path and continue? [y/N]" input
|
||||
case $input in
|
||||
[Yy]) mkdir --parents "$1" ;;
|
||||
[Nn]) exit 0;;
|
||||
* ) exit 0 ;;
|
||||
esac
|
||||
fi
|
||||
}
|
||||
|
||||
if [ $BACKUP_LOCATION ]; then
|
||||
check_for_valid_backup_location "$BACKUP_LOCATION"
|
||||
else
|
||||
echo -e "${IRED}No BACKUP_LOCATION in $HOME/.config/update.conf specified${NO_COLOR}"
|
||||
exit 1
|
||||
fi
|
||||
[[ $SECONDARY_BACKUP_LOCATION ]] && check_for_valid_backup_location "$SECONDARY_BACKUP_LOCATION"
|
||||
|
||||
lock_pacman_db() {
|
||||
if [ -f /var/lib/pacman/db.lck ]; then
|
||||
echo -e "${IYELLOW}->${NO_COLOR} /var/lib/pacman/db.lck exists"
|
||||
echo -e "${IYELLOW}->${NO_COLOR} there might be another instance of pacman running. exiting..."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo -e "${IYELLOW}Trying to lock the pacman db. Please enter your password.${NO_COLOR}"
|
||||
su --command="touch /var/lib/pacman/db.lck"
|
||||
}
|
||||
|
||||
unlock_pacman_db() {
|
||||
echo -e "${IYELLOW}Trying to unlock the pacman db. Please enter your password.${NO_COLOR}"
|
||||
su --command="rm --force /var/lib/pacman/db.lck"
|
||||
}
|
||||
|
||||
delete_oldest_backup() {
|
||||
if [[ $(find $BACKUP_LOCATION -name '*$1*' -exec printf %c {} + | wc -c) -ge $BACKUP_AMOUNT ]]; then
|
||||
rm --force $(find "$BACKUP_LOCATION" -name '*$1*' | sort -rn | head -1)
|
||||
fi
|
||||
}
|
||||
|
||||
before_backup() {
|
||||
lock_pacman_db
|
||||
|
||||
[[ -d $TMP ]] && rm --recursive --force "$TMP"
|
||||
mkdir --parents "$TMP"/before-backup_"$DATE" "$TMP"/after-backup_"$DATE"
|
||||
|
||||
pacman --verbose --query > "$TMP"/before-backup_"$DATE"/pacman-before.txt
|
||||
[[ -x /usr/bin/flatpak ]] && flatpak list --all --show-details > "$TMP"/before-backup_"$DATE"/flatpak-before.txt
|
||||
|
||||
tar --create --zstd --file "$TMP"/before-backup_"$DATE".tar.zst.new "$TMP"/before-backup_"$DATE" /var/lib/pacman/local
|
||||
cp "$TMP"/before-backup_"$DATE".tar.zst.new "$BACKUP_LOCATION"
|
||||
[[ $SECONDARY_BACKUP_LOCATION ]] && cp "$TMP"/before-backup_"$DATE".tar.zst.new "$SECONDARY_BACKUP_LOCATION"
|
||||
|
||||
unlock_pacman_db
|
||||
}
|
||||
|
||||
after_backup() {
|
||||
lock_pacman_db
|
||||
|
||||
pacman --verbose --query > "$TMP"/after-backup_"$DATE"/pacman-after.txt
|
||||
[[ -x /usr/bin/flatpak ]] && flatpak list --all --show-details > "$TMP"/after-backup_"$DATE"/flatpak-after.txt
|
||||
tar --create --zstd --file "$TMP"/after-backup_"$DATE".tar.zst.new "$TMP"/after-backup_"$DATE"
|
||||
|
||||
cp "$TMP"/after-backup_"$DATE".tar.zst.new "$BACKUP_LOCATION"
|
||||
[[ $SECONDARY_BACKUP_LOCATION ]] && cp "$TMP"/after-backup_"$DATE".tar.zst.new "$SECONDARY_BACKUP_LOCATION"
|
||||
|
||||
delete_oldest_backup after-backup
|
||||
mv "$BACKUP_LOCATION"/after-backup_"$DATE".tar.zst.new "$BACKUP_LOCATION"/after-backup_"$DATE".tar.zst
|
||||
[[ $SECONDARY_BACKUP_LOCATION ]] && mv "$SECONDARY_BACKUP_LOCATION"/after-backup_"$DATE".tar.zst.new "$SECONDARY_BACKUP_LOCATION"/after-backup_"$DATE".tar.zst
|
||||
|
||||
delete_oldest_backup before-backup
|
||||
mv "$BACKUP_LOCATION"/before-backup_"$DATE".tar.zst.new "$BACKUP_LOCATION"/before-backup_"$DATE".tar.zst
|
||||
[[ $SECONDARY_BACKUP_LOCATION ]] && mv "$SECONDARY_BACKUP_LOCATION"/before-backup_"$DATE".tar.zst.new "$SECONDARY_BACKUP_LOCATION"/before-backup_"$DATE".tar.zst
|
||||
|
||||
unlock_pacman_db
|
||||
rm --recursive --force "$TMP"
|
||||
}
|
||||
|
||||
update() {
|
||||
"$PACMAN_WRAPPER"
|
||||
if [[ -x /usr/bin/flatpak ]]; then
|
||||
flatpak update --assumeyes
|
||||
flatpak list --all --show-details > "$TMP"/after-backup_"$DATE"/flatpak-after.txt
|
||||
fi
|
||||
}
|
||||
|
||||
before_backup && echo -e "${IGREEN}before-backup complete${NO_COLOR}"
|
||||
update
|
||||
after_backup && echo -e "${IGREEN}after-backup complete${NO_COLOR}"
|
83
update.c
Normal file
83
update.c
Normal file
@ -0,0 +1,83 @@
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
|
||||
int command_builder(char *command, char **args) {
|
||||
char *tmp_command = malloc(strlen(command));
|
||||
memcpy(tmp_command, command, strlen(command));
|
||||
char *tmp_token;
|
||||
size_t index = 0;
|
||||
|
||||
if (tmp_command == NULL) { return 1; }
|
||||
|
||||
tmp_token = strtok(tmp_command, " ");
|
||||
while(tmp_token != NULL) {
|
||||
args[index] = malloc(strlen(tmp_token));
|
||||
memcpy(args[index], tmp_token, strlen(tmp_token));
|
||||
tmp_token = strtok(NULL, " ");
|
||||
index++;
|
||||
}
|
||||
free(tmp_command);
|
||||
args[index++] = NULL;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int run_cmd_and_wait(char *command) {
|
||||
char **args = malloc(strlen(command));
|
||||
int ret = command_builder(command, args);
|
||||
if (ret != 0) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
signed int pid = fork();
|
||||
int status;
|
||||
|
||||
if (pid == -1) {
|
||||
fprintf(stderr, "Couldn't fork. Error = %s\n", strerror(errno));
|
||||
return 1;
|
||||
} else if (pid > 0) {
|
||||
waitpid(pid, &status, 0);
|
||||
} else {
|
||||
execvp(args[0], args);
|
||||
}
|
||||
free(args);
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
int main() {
|
||||
FILE *os_release = fopen("/etc/os-release", "r");
|
||||
if (!os_release) {
|
||||
fprintf(stderr, "Couldn't open os-release. Error = %s\n", strerror(errno));
|
||||
}
|
||||
char buffer[1024];
|
||||
while (fgets(buffer, sizeof buffer, os_release)) {
|
||||
if (strncmp(buffer, "NAME=", 5) == 0) {
|
||||
memmove(buffer, buffer + 6, (sizeof buffer) - 6);
|
||||
buffer[strcspn(buffer, "\"")] = '\0';
|
||||
break;
|
||||
}
|
||||
}
|
||||
int ret;
|
||||
char *args[100];
|
||||
if (strcmp(buffer, "EndeavourOS") == 0) {
|
||||
ret = run_cmd_and_wait("yay");
|
||||
} else if (strcmp(buffer, "Debian") == 0) {
|
||||
ret = run_cmd_and_wait("apt update");
|
||||
if (ret != 0) {
|
||||
return ret;
|
||||
}
|
||||
ret = run_cmd_and_wait("apt upgrade");
|
||||
} else if (strcmp(buffer, "FreeBSD") == 0) {
|
||||
ret = run_cmd_and_wait("pkg update");
|
||||
if (ret != 0) {
|
||||
return ret;
|
||||
}
|
||||
ret = run_cmd_and_wait("pkg upgrade");
|
||||
} else {
|
||||
fprintf(stderr, "OS %s not suported\n", buffer);
|
||||
return 1;
|
||||
}
|
||||
return ret;
|
||||
}
|
11
update.conf
11
update.conf
@ -1,11 +0,0 @@
|
||||
# backup to this folder
|
||||
#BACKUP_LOCATION=~
|
||||
|
||||
# optional secondary backup loaction
|
||||
#SECONDARY_BACKUP_LOCATION=~
|
||||
|
||||
# use this pacman-wrapper
|
||||
PACMAN_WRAPPER=yay
|
||||
|
||||
# how many different backups to keep
|
||||
BACKUP_AMOUNT=10
|
@ -1,4 +0,0 @@
|
||||
# Config for update
|
||||
|
||||
[backup]
|
||||
locations = ['/opt'] # All locations need to be an absolute path
|
Reference in New Issue
Block a user