mirror of
https://github.com/cupcakearmy/autorestic.git
synced 2024-12-24 09:06:27 +00:00
Compare commits
8 Commits
67bacd499b
...
1ad8935e5a
Author | SHA1 | Date | |
---|---|---|---|
|
1ad8935e5a | ||
ced20801c1 | |||
|
ce9140fa1e | ||
|
046c79fd15 | ||
|
f8603425d1 | ||
|
8b60120342 | ||
|
ba51d1d062 | ||
|
43efd1db1a |
@ -7,7 +7,7 @@ COPY . .
|
|||||||
RUN go build
|
RUN go build
|
||||||
|
|
||||||
FROM restic/restic:0.16.0
|
FROM restic/restic:0.16.0
|
||||||
RUN apk add --no-cache rclone bash
|
RUN apk add --no-cache rclone bash curl
|
||||||
COPY --from=builder /app/autorestic /usr/bin/autorestic
|
COPY --from=builder /app/autorestic /usr/bin/autorestic
|
||||||
ENTRYPOINT []
|
ENTRYPOINT []
|
||||||
CMD [ "autorestic" ]
|
CMD [ "autorestic" ]
|
||||||
|
38
cmd/root.go
38
cmd/root.go
@ -59,24 +59,7 @@ func initConfig() {
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
configPaths := []string{"."}
|
configPaths := getConfigPaths()
|
||||||
|
|
||||||
// Home
|
|
||||||
if home, err := homedir.Dir(); err == nil {
|
|
||||||
configPaths = append(configPaths, home)
|
|
||||||
}
|
|
||||||
|
|
||||||
// XDG_CONFIG_HOME
|
|
||||||
{
|
|
||||||
prefix, found := os.LookupEnv("XDG_CONFIG_HOME")
|
|
||||||
if !found {
|
|
||||||
if home, err := homedir.Dir(); err != nil {
|
|
||||||
prefix = filepath.Join(home, ".config")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
xdgConfig := filepath.Join(prefix, "autorestic")
|
|
||||||
configPaths = append(configPaths, xdgConfig)
|
|
||||||
}
|
|
||||||
for _, cfgPath := range configPaths {
|
for _, cfgPath := range configPaths {
|
||||||
viper.AddConfigPath(cfgPath)
|
viper.AddConfigPath(cfgPath)
|
||||||
}
|
}
|
||||||
@ -88,3 +71,22 @@ func initConfig() {
|
|||||||
viper.AutomaticEnv()
|
viper.AutomaticEnv()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func getConfigPaths() []string {
|
||||||
|
result := []string{"."}
|
||||||
|
if home, err := homedir.Dir(); err == nil {
|
||||||
|
result = append(result, home)
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
xdgConfigHome, found := os.LookupEnv("XDG_CONFIG_HOME")
|
||||||
|
if !found {
|
||||||
|
if home, err := homedir.Dir(); err == nil {
|
||||||
|
xdgConfigHome = filepath.Join(home, ".config")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
xdgConfig := filepath.Join(xdgConfigHome, "autorestic")
|
||||||
|
result = append(result, xdgConfig)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
36
cmd/root_test.go
Normal file
36
cmd/root_test.go
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/mitchellh/go-homedir"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"slices"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
const xdgConfigHome = "XDG_CONFIG_HOME"
|
||||||
|
|
||||||
|
func assertContains(t *testing.T, array []string, element string) {
|
||||||
|
if !slices.Contains(array, element) {
|
||||||
|
t.Errorf("Expected %s to be contained in %s", element, array)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestConfigResolving(t *testing.T) {
|
||||||
|
t.Run("~/.config/autorestic is used if XDG_CONFIG_HOME is not set", func(t *testing.T) {
|
||||||
|
// Override env using testing so that env gets restored after test
|
||||||
|
t.Setenv(xdgConfigHome, "")
|
||||||
|
_ = os.Unsetenv("XDG_CONFIG_HOME")
|
||||||
|
configPaths := getConfigPaths()
|
||||||
|
homeDir, _ := homedir.Dir()
|
||||||
|
expectedConfigPath := filepath.Join(homeDir, ".config/autorestic")
|
||||||
|
assertContains(t, configPaths, expectedConfigPath)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("XDG_CONFIG_HOME is respected if set", func(t *testing.T) {
|
||||||
|
t.Setenv(xdgConfigHome, "/foo/bar")
|
||||||
|
|
||||||
|
configPaths := getConfigPaths()
|
||||||
|
assertContains(t, configPaths, filepath.Join("/", "foo", "bar", "autorestic"))
|
||||||
|
})
|
||||||
|
}
|
79
cmd/unlock.go
Normal file
79
cmd/unlock.go
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/cupcakearmy/autorestic/internal"
|
||||||
|
"github.com/cupcakearmy/autorestic/internal/colors"
|
||||||
|
"github.com/cupcakearmy/autorestic/internal/lock"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
var unlockCmd = &cobra.Command{
|
||||||
|
Use: "unlock",
|
||||||
|
Short: "Unlock autorestic only if you are sure that no other instance is running",
|
||||||
|
Long: `Unlock autorestic only if you are sure that no other instance is running.
|
||||||
|
To check you can run "ps aux | grep autorestic".`,
|
||||||
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
|
internal.GetConfig()
|
||||||
|
|
||||||
|
force, _ := cmd.Flags().GetBool("force")
|
||||||
|
|
||||||
|
if !force && isAutoresticRunning() {
|
||||||
|
colors.Error.Print("Another autorestic instance is running. Are you sure you want to unlock? (yes/no): ")
|
||||||
|
var response string
|
||||||
|
fmt.Scanln(&response)
|
||||||
|
if strings.ToLower(response) != "yes" {
|
||||||
|
colors.Primary.Println("Unlocking aborted.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err := lock.Unlock()
|
||||||
|
if err != nil {
|
||||||
|
colors.Error.Println("Could not unlock:", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
colors.Success.Println("Unlock successful")
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
rootCmd.AddCommand(unlockCmd)
|
||||||
|
unlockCmd.Flags().Bool("force", false, "force unlock")
|
||||||
|
}
|
||||||
|
|
||||||
|
// isAutoresticRunning checks if autorestic is running
|
||||||
|
// and returns true if it is.
|
||||||
|
// It also prints the processes to stdout.
|
||||||
|
func isAutoresticRunning() bool {
|
||||||
|
cmd := exec.Command("sh", "-c", "ps aux | grep autorestic")
|
||||||
|
var out bytes.Buffer
|
||||||
|
cmd.Stdout = &out
|
||||||
|
err := cmd.Run()
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
lines := strings.Split(out.String(), "\n")
|
||||||
|
autoresticProcesses := []string{}
|
||||||
|
|
||||||
|
for _, line := range lines {
|
||||||
|
if strings.Contains(line, "autorestic") && !strings.Contains(line, "grep autorestic") {
|
||||||
|
autoresticProcesses = append(autoresticProcesses, line)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(autoresticProcesses) > 0 {
|
||||||
|
colors.Faint.Println("Found autorestic processes:")
|
||||||
|
for _, proc := range autoresticProcesses {
|
||||||
|
colors.Faint.Println(proc)
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
32
docs/pages/cli/unlock.md
Normal file
32
docs/pages/cli/unlock.md
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# Unlock
|
||||||
|
|
||||||
|
In case autorestic throws the error message `an instance is already running. exiting`, but there is no instance running you can unlock the lock.
|
||||||
|
|
||||||
|
To verify that there is no instance running you can use `ps aux | grep autorestic`.
|
||||||
|
|
||||||
|
Example with no instance running:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
> ps aux | grep autorestic
|
||||||
|
root 39260 0.0 0.0 6976 2696 pts/11 S+ 19:41 0:00 grep autorestic
|
||||||
|
```
|
||||||
|
|
||||||
|
Example with an instance running:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
> ps aux | grep autorestic
|
||||||
|
root 29465 0.0 0.0 1162068 7380 pts/7 Sl+ 19:28 0:00 autorestic --ci backup -a
|
||||||
|
root 39260 0.0 0.0 6976 2696 pts/11 S+ 19:41 0:00 grep autorestic
|
||||||
|
```
|
||||||
|
|
||||||
|
**If an instance is running you should not unlock as it could lead to data loss!**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
autorestic unlock
|
||||||
|
```
|
||||||
|
|
||||||
|
Use the `--force` to prevent the confirmation prompt if an instance is running.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
autorestic unlock --force
|
||||||
|
```
|
@ -2,10 +2,11 @@
|
|||||||
|
|
||||||
## Path
|
## Path
|
||||||
|
|
||||||
By default autorestic searches for a `.autorestic.yml` file in the current directory and your home folder.
|
By default autorestic searches for a `.autorestic.yml` file in the current directory, your home folder and your XDG config folder (`~/.config/` by default):
|
||||||
|
|
||||||
- `./.autorestic.yml`
|
- `./.autorestic.yml`
|
||||||
- `~/.autorestic.yml`
|
- `~/.autorestic.yml`
|
||||||
|
- `~/.config/autorestic/.autorestic.yml`
|
||||||
|
|
||||||
You can also specify a custom file with the `-c path/to/some/config.yml`
|
You can also specify a custom file with the `-c path/to/some/config.yml`
|
||||||
|
|
||||||
|
@ -30,4 +30,4 @@ Fedora users can install the [autorestic](https://src.fedoraproject.org/rpms/aut
|
|||||||
|
|
||||||
### AUR
|
### AUR
|
||||||
|
|
||||||
~~If you are on Arch there is an [AUR Package](https://aur.archlinux.org/packages/autorestic-bin/) (looking for maintainers).~~ - Deprecated
|
If you are on Arch there is an [AUR Package](https://aur.archlinux.org/packages/autorestic-bin/)
|
||||||
|
@ -17,7 +17,7 @@ import (
|
|||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
)
|
)
|
||||||
|
|
||||||
const VERSION = "1.7.9"
|
const VERSION = "1.7.10"
|
||||||
|
|
||||||
type OptionMap map[string][]interface{}
|
type OptionMap map[string][]interface{}
|
||||||
type Options map[string]OptionMap
|
type Options map[string]OptionMap
|
||||||
|
Loading…
Reference in New Issue
Block a user