feat(stores): adds mvs, and flags to bring store commands on par with key commands
This commit is contained in:
parent
b89db8dc48
commit
4e5064d07a
25 changed files with 247 additions and 9 deletions
27
README.md
27
README.md
|
|
@ -89,6 +89,7 @@ Store commands:
|
||||||
export Export store as NDJSON (alias for list --format ndjson)
|
export Export store as NDJSON (alias for list --format ndjson)
|
||||||
import Restore key/value pairs from an NDJSON dump
|
import Restore key/value pairs from an NDJSON dump
|
||||||
list-stores List all stores
|
list-stores List all stores
|
||||||
|
move-store Rename a store
|
||||||
remove-store Delete a store
|
remove-store Delete a store
|
||||||
|
|
||||||
Git commands:
|
Git commands:
|
||||||
|
|
@ -171,7 +172,14 @@ pda get name --exists
|
||||||
`pda mv` to move it.
|
`pda mv` to move it.
|
||||||
```bash
|
```bash
|
||||||
pda mv name name2
|
pda mv name name2
|
||||||
# renamed name to name2
|
# ok renamed name to name2
|
||||||
|
|
||||||
|
# --safe to skip if the destination already exists.
|
||||||
|
pda mv name name2 --safe
|
||||||
|
# info skipped 'name2': already exists
|
||||||
|
|
||||||
|
# --yes/-y to skip confirmation prompts.
|
||||||
|
pda mv name name2 -y
|
||||||
```
|
```
|
||||||
|
|
||||||
`pda cp` to make a copy.
|
`pda cp` to make a copy.
|
||||||
|
|
@ -277,7 +285,7 @@ pda import --drop -f my_backup
|
||||||
|
|
||||||
<p align="center"></p><!-- spacer -->
|
<p align="center"></p><!-- spacer -->
|
||||||
|
|
||||||
You can have as many stores as you want.
|
You can have as many stores as you want. All the store commands have shorthands, like `mv` to move a key, or `mvs` to move a store.
|
||||||
```bash
|
```bash
|
||||||
# Save to a specific store.
|
# Save to a specific store.
|
||||||
pda set alice@birthdays 11/11/1998
|
pda set alice@birthdays 11/11/1998
|
||||||
|
|
@ -298,8 +306,20 @@ pda export birthdays > friends_birthdays
|
||||||
# Import it.
|
# Import it.
|
||||||
pda import birthdays < friends_birthdays
|
pda import birthdays < friends_birthdays
|
||||||
|
|
||||||
|
# Rename it.
|
||||||
|
pda move-store birthdays bdays
|
||||||
|
|
||||||
|
# Or copy it.
|
||||||
|
pda move-store birthdays bdays --copy
|
||||||
|
|
||||||
|
# --safe to skip if the destination already exists.
|
||||||
|
pda move-store birthdays bdays --safe
|
||||||
|
|
||||||
# Delete it.
|
# Delete it.
|
||||||
pda rm-store birthdays
|
pda remove-store birthdays
|
||||||
|
|
||||||
|
# --yes/-y to skip confirmation prompts on delete or overwrite.
|
||||||
|
pda remove-store birthdays -y
|
||||||
```
|
```
|
||||||
|
|
||||||
<p align="center"></p><!-- spacer -->
|
<p align="center"></p><!-- spacer -->
|
||||||
|
|
@ -714,6 +734,7 @@ always_prompt_overwrite = false
|
||||||
[store]
|
[store]
|
||||||
default_store_name = "default"
|
default_store_name = "default"
|
||||||
always_prompt_delete = true
|
always_prompt_delete = true
|
||||||
|
always_prompt_overwrite = true
|
||||||
|
|
||||||
[git]
|
[git]
|
||||||
auto_fetch = false
|
auto_fetch = false
|
||||||
|
|
|
||||||
|
|
@ -44,8 +44,9 @@ type KeyConfig struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type StoreConfig struct {
|
type StoreConfig struct {
|
||||||
DefaultStoreName string `toml:"default_store_name"`
|
DefaultStoreName string `toml:"default_store_name"`
|
||||||
AlwaysPromptDelete bool `toml:"always_prompt_delete"`
|
AlwaysPromptDelete bool `toml:"always_prompt_delete"`
|
||||||
|
AlwaysPromptOverwrite bool `toml:"always_prompt_overwrite"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GitConfig struct {
|
type GitConfig struct {
|
||||||
|
|
@ -80,8 +81,9 @@ func defaultConfig() Config {
|
||||||
AlwaysPromptOverwrite: false,
|
AlwaysPromptOverwrite: false,
|
||||||
},
|
},
|
||||||
Store: StoreConfig{
|
Store: StoreConfig{
|
||||||
DefaultStoreName: "default",
|
DefaultStoreName: "default",
|
||||||
AlwaysPromptDelete: true,
|
AlwaysPromptDelete: true,
|
||||||
|
AlwaysPromptOverwrite: true,
|
||||||
},
|
},
|
||||||
Git: GitConfig{
|
Git: GitConfig{
|
||||||
AutoFetch: false,
|
AutoFetch: false,
|
||||||
|
|
|
||||||
|
|
@ -60,8 +60,12 @@ func delStore(cmd *cobra.Command, args []string) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("cannot delete store '%s': %v", dbName, err)
|
return fmt.Errorf("cannot delete store '%s': %v", dbName, err)
|
||||||
}
|
}
|
||||||
|
yes, err := cmd.Flags().GetBool("yes")
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot delete store '%s': %v", dbName, err)
|
||||||
|
}
|
||||||
|
|
||||||
if interactive || config.Store.AlwaysPromptDelete {
|
if !yes && (interactive || config.Store.AlwaysPromptDelete) {
|
||||||
promptf("delete store '%s'? (y/n)", args[0])
|
promptf("delete store '%s'? (y/n)", args[0])
|
||||||
|
|
||||||
var confirm string
|
var confirm string
|
||||||
|
|
@ -87,5 +91,6 @@ func executeDeletion(path string) error {
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
delStoreCmd.Flags().BoolP("interactive", "i", false, "Prompt yes/no for each deletion")
|
delStoreCmd.Flags().BoolP("interactive", "i", false, "Prompt yes/no for each deletion")
|
||||||
|
delStoreCmd.Flags().BoolP("yes", "y", false, "Skip all confirmation prompts")
|
||||||
rootCmd.AddCommand(delStoreCmd)
|
rootCmd.AddCommand(delStoreCmd)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -35,6 +35,7 @@ func stdoutIsTerminal() bool {
|
||||||
// FAIL red (stderr)
|
// FAIL red (stderr)
|
||||||
// hint dim (stderr)
|
// hint dim (stderr)
|
||||||
// WARN yellow (stderr)
|
// WARN yellow (stderr)
|
||||||
|
// info blue (stderr)
|
||||||
// ok green (stderr)
|
// ok green (stderr)
|
||||||
// ? cyan (stdout)
|
// ? cyan (stdout)
|
||||||
// > dim (stdout)
|
// > dim (stdout)
|
||||||
|
|
@ -60,6 +61,11 @@ func warnf(format string, args ...any) {
|
||||||
fmt.Fprintf(os.Stderr, "%s %s\n", keyword("33", "WARN", stderrIsTerminal()), msg)
|
fmt.Fprintf(os.Stderr, "%s %s\n", keyword("33", "WARN", stderrIsTerminal()), msg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func infof(format string, args ...any) {
|
||||||
|
msg := fmt.Sprintf(format, args...)
|
||||||
|
fmt.Fprintf(os.Stderr, "%s %s\n", keyword("34", "info", stderrIsTerminal()), msg)
|
||||||
|
}
|
||||||
|
|
||||||
func okf(format string, args ...any) {
|
func okf(format string, args ...any) {
|
||||||
msg := fmt.Sprintf(format, args...)
|
msg := fmt.Sprintf(format, args...)
|
||||||
fmt.Fprintf(os.Stderr, "%s %s\n", keyword("32", "ok", stderrIsTerminal()), msg)
|
fmt.Fprintf(os.Stderr, "%s %s\n", keyword("32", "ok", stderrIsTerminal()), msg)
|
||||||
|
|
|
||||||
129
cmd/mv-db.go
Normal file
129
cmd/mv-db.go
Normal file
|
|
@ -0,0 +1,129 @@
|
||||||
|
/*
|
||||||
|
Copyright © 2025 Lewis Wynne <lew@ily.rs>
|
||||||
|
|
||||||
|
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 cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
)
|
||||||
|
|
||||||
|
// mvStoreCmd represents the move-store command
|
||||||
|
var mvStoreCmd = &cobra.Command{
|
||||||
|
Use: "move-store FROM TO",
|
||||||
|
Short: "Rename a store",
|
||||||
|
Aliases: []string{"mvs"},
|
||||||
|
Args: cobra.ExactArgs(2),
|
||||||
|
RunE: mvStore,
|
||||||
|
SilenceUsage: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
func mvStore(cmd *cobra.Command, args []string) error {
|
||||||
|
store := &Store{}
|
||||||
|
|
||||||
|
fromName, err := store.parseDB(args[0], false)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", args[0], err)
|
||||||
|
}
|
||||||
|
toName, err := store.parseDB(args[1], false)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", args[1], err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if fromName == toName {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': source and destination are the same", fromName)
|
||||||
|
}
|
||||||
|
|
||||||
|
var notFound errNotFound
|
||||||
|
fromPath, err := store.FindStore(fromName)
|
||||||
|
if errors.As(err, ¬Found) {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %w", fromName, err)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
interactive, err := cmd.Flags().GetBool("interactive")
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
safe, err := cmd.Flags().GetBool("safe")
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
yes, err := cmd.Flags().GetBool("yes")
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
promptOverwrite := !yes && (interactive || config.Store.AlwaysPromptOverwrite)
|
||||||
|
|
||||||
|
toPath, err := store.storePath(toName)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(toPath); err == nil {
|
||||||
|
if safe {
|
||||||
|
infof("skipped '@%s': already exists", toName)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if promptOverwrite {
|
||||||
|
promptf("overwrite store '%s'? (y/n)", toName)
|
||||||
|
var confirm string
|
||||||
|
if err := scanln(&confirm); err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
if strings.ToLower(confirm) != "y" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
copy, _ := cmd.Flags().GetBool("copy")
|
||||||
|
if copy {
|
||||||
|
data, err := os.ReadFile(fromPath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("cannot copy store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
if err := os.WriteFile(toPath, data, 0o640); err != nil {
|
||||||
|
return fmt.Errorf("cannot copy store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
okf("copied @%s to @%s", fromName, toName)
|
||||||
|
} else {
|
||||||
|
if err := os.Rename(fromPath, toPath); err != nil {
|
||||||
|
return fmt.Errorf("cannot rename store '%s': %v", fromName, err)
|
||||||
|
}
|
||||||
|
okf("renamed @%s to @%s", fromName, toName)
|
||||||
|
}
|
||||||
|
return autoSync()
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
mvStoreCmd.Flags().Bool("copy", false, "Copy instead of move (keeps source)")
|
||||||
|
mvStoreCmd.Flags().BoolP("interactive", "i", false, "Prompt before overwriting destination")
|
||||||
|
mvStoreCmd.Flags().BoolP("yes", "y", false, "Skip all confirmation prompts")
|
||||||
|
mvStoreCmd.Flags().Bool("safe", false, "Do not overwrite if the destination store already exists")
|
||||||
|
rootCmd.AddCommand(mvStoreCmd)
|
||||||
|
}
|
||||||
24
cmd/mv.go
24
cmd/mv.go
|
|
@ -64,7 +64,15 @@ func mvImpl(cmd *cobra.Command, args []string, keepSource bool) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
promptOverwrite := interactive || config.Key.AlwaysPromptOverwrite
|
safe, err := cmd.Flags().GetBool("safe")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
yes, err := cmd.Flags().GetBool("yes")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
promptOverwrite := !yes && (interactive || config.Key.AlwaysPromptOverwrite)
|
||||||
|
|
||||||
identity, _ := loadIdentity()
|
identity, _ := loadIdentity()
|
||||||
var recipient *age.X25519Recipient
|
var recipient *age.X25519Recipient
|
||||||
|
|
@ -114,6 +122,11 @@ func mvImpl(cmd *cobra.Command, args []string, keepSource bool) error {
|
||||||
|
|
||||||
dstIdx := findEntry(dstEntries, toSpec.Key)
|
dstIdx := findEntry(dstEntries, toSpec.Key)
|
||||||
|
|
||||||
|
if safe && dstIdx >= 0 {
|
||||||
|
infof("skipped '%s': already exists", toSpec.Display())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
if promptOverwrite && dstIdx >= 0 {
|
if promptOverwrite && dstIdx >= 0 {
|
||||||
var confirm string
|
var confirm string
|
||||||
promptf("overwrite '%s'? (y/n)", toSpec.Display())
|
promptf("overwrite '%s'? (y/n)", toSpec.Display())
|
||||||
|
|
@ -169,13 +182,22 @@ func mvImpl(cmd *cobra.Command, args []string, keepSource bool) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if keepSource {
|
||||||
|
okf("copied %s to %s", fromSpec.Display(), toSpec.Display())
|
||||||
|
} else {
|
||||||
|
okf("renamed %s to %s", fromSpec.Display(), toSpec.Display())
|
||||||
|
}
|
||||||
return autoSync()
|
return autoSync()
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
mvCmd.Flags().Bool("copy", false, "Copy instead of move (keeps source)")
|
mvCmd.Flags().Bool("copy", false, "Copy instead of move (keeps source)")
|
||||||
mvCmd.Flags().BoolP("interactive", "i", false, "Prompt before overwriting destination")
|
mvCmd.Flags().BoolP("interactive", "i", false, "Prompt before overwriting destination")
|
||||||
|
mvCmd.Flags().BoolP("yes", "y", false, "Skip all confirmation prompts")
|
||||||
|
mvCmd.Flags().Bool("safe", false, "Do not overwrite if the destination already exists")
|
||||||
rootCmd.AddCommand(mvCmd)
|
rootCmd.AddCommand(mvCmd)
|
||||||
cpCmd.Flags().BoolP("interactive", "i", false, "Prompt before overwriting destination")
|
cpCmd.Flags().BoolP("interactive", "i", false, "Prompt before overwriting destination")
|
||||||
|
cpCmd.Flags().BoolP("yes", "y", false, "Skip all confirmation prompts")
|
||||||
|
cpCmd.Flags().Bool("safe", false, "Do not overwrite if the destination already exists")
|
||||||
rootCmd.AddCommand(cpCmd)
|
rootCmd.AddCommand(cpCmd)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -65,6 +65,7 @@ func init() {
|
||||||
|
|
||||||
listStoresCmd.GroupID = "stores"
|
listStoresCmd.GroupID = "stores"
|
||||||
delStoreCmd.GroupID = "stores"
|
delStoreCmd.GroupID = "stores"
|
||||||
|
mvStoreCmd.GroupID = "stores"
|
||||||
exportCmd.GroupID = "stores"
|
exportCmd.GroupID = "stores"
|
||||||
restoreCmd.GroupID = "stores"
|
restoreCmd.GroupID = "stores"
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -135,6 +135,7 @@ func set(cmd *cobra.Command, args []string) error {
|
||||||
idx := findEntry(entries, spec.Key)
|
idx := findEntry(entries, spec.Key)
|
||||||
|
|
||||||
if safe && idx >= 0 {
|
if safe && idx >= 0 {
|
||||||
|
infof("skipped '%s': already exists", spec.Display())
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
1
testdata/cp-cross-store.ct
vendored
1
testdata/cp-cross-store.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Cross-store copy
|
# Cross-store copy
|
||||||
$ pda set key@src value
|
$ pda set key@src value
|
||||||
$ pda cp key@src key@dst
|
$ pda cp key@src key@dst
|
||||||
|
ok copied key@src to key@dst
|
||||||
$ pda get key@src
|
$ pda get key@src
|
||||||
value
|
value
|
||||||
$ pda get key@dst
|
$ pda get key@dst
|
||||||
|
|
|
||||||
1
testdata/cp-encrypt.ct
vendored
1
testdata/cp-encrypt.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Copy an encrypted key; both keys should decrypt.
|
# Copy an encrypted key; both keys should decrypt.
|
||||||
$ pda set --encrypt secret-key@cpe hidden-value
|
$ pda set --encrypt secret-key@cpe hidden-value
|
||||||
$ pda cp secret-key@cpe copied-key@cpe
|
$ pda cp secret-key@cpe copied-key@cpe
|
||||||
|
ok copied secret-key@cpe to copied-key@cpe
|
||||||
$ pda get secret-key@cpe
|
$ pda get secret-key@cpe
|
||||||
hidden-value
|
hidden-value
|
||||||
$ pda get copied-key@cpe
|
$ pda get copied-key@cpe
|
||||||
|
|
|
||||||
6
testdata/cp-safe.ct
vendored
Normal file
6
testdata/cp-safe.ct
vendored
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
$ pda set src@csf hello
|
||||||
|
$ pda set dst@csf existing
|
||||||
|
$ pda cp src@csf dst@csf --safe
|
||||||
|
info skipped 'dst@csf': already exists
|
||||||
|
$ pda get dst@csf
|
||||||
|
existing
|
||||||
1
testdata/cp.ct
vendored
1
testdata/cp.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Basic copy
|
# Basic copy
|
||||||
$ pda set source@cpok value
|
$ pda set source@cpok value
|
||||||
$ pda cp source@cpok dest@cpok
|
$ pda cp source@cpok dest@cpok
|
||||||
|
ok copied source@cpok to dest@cpok
|
||||||
$ pda get source@cpok
|
$ pda get source@cpok
|
||||||
value
|
value
|
||||||
$ pda get dest@cpok
|
$ pda get dest@cpok
|
||||||
|
|
|
||||||
2
testdata/help-remove-store.ct
vendored
2
testdata/help-remove-store.ct
vendored
|
|
@ -11,6 +11,7 @@ Aliases:
|
||||||
Flags:
|
Flags:
|
||||||
-h, --help help for remove-store
|
-h, --help help for remove-store
|
||||||
-i, --interactive Prompt yes/no for each deletion
|
-i, --interactive Prompt yes/no for each deletion
|
||||||
|
-y, --yes Skip all confirmation prompts
|
||||||
Delete a store
|
Delete a store
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
@ -22,3 +23,4 @@ Aliases:
|
||||||
Flags:
|
Flags:
|
||||||
-h, --help help for remove-store
|
-h, --help help for remove-store
|
||||||
-i, --interactive Prompt yes/no for each deletion
|
-i, --interactive Prompt yes/no for each deletion
|
||||||
|
-y, --yes Skip all confirmation prompts
|
||||||
|
|
|
||||||
2
testdata/help.ct
vendored
2
testdata/help.ct
vendored
|
|
@ -26,6 +26,7 @@ Store commands:
|
||||||
export Export store as NDJSON (alias for list --format ndjson)
|
export Export store as NDJSON (alias for list --format ndjson)
|
||||||
import Restore key/value pairs from an NDJSON dump
|
import Restore key/value pairs from an NDJSON dump
|
||||||
list-stores List all stores
|
list-stores List all stores
|
||||||
|
move-store Rename a store
|
||||||
remove-store Delete a store
|
remove-store Delete a store
|
||||||
|
|
||||||
Git commands:
|
Git commands:
|
||||||
|
|
@ -68,6 +69,7 @@ Store commands:
|
||||||
export Export store as NDJSON (alias for list --format ndjson)
|
export Export store as NDJSON (alias for list --format ndjson)
|
||||||
import Restore key/value pairs from an NDJSON dump
|
import Restore key/value pairs from an NDJSON dump
|
||||||
list-stores List all stores
|
list-stores List all stores
|
||||||
|
move-store Rename a store
|
||||||
remove-store Delete a store
|
remove-store Delete a store
|
||||||
|
|
||||||
Git commands:
|
Git commands:
|
||||||
|
|
|
||||||
1
testdata/mv-cross-store.ct
vendored
1
testdata/mv-cross-store.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Cross-store move
|
# Cross-store move
|
||||||
$ pda set key@src value
|
$ pda set key@src value
|
||||||
$ pda mv key@src key@dst
|
$ pda mv key@src key@dst
|
||||||
|
ok renamed key@src to key@dst
|
||||||
$ pda get key@dst
|
$ pda get key@dst
|
||||||
value
|
value
|
||||||
$ pda get key@src --> FAIL
|
$ pda get key@src --> FAIL
|
||||||
|
|
|
||||||
1
testdata/mv-encrypt.ct
vendored
1
testdata/mv-encrypt.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Move an encrypted key; the new key should still decrypt.
|
# Move an encrypted key; the new key should still decrypt.
|
||||||
$ pda set --encrypt secret-key@mve hidden-value
|
$ pda set --encrypt secret-key@mve hidden-value
|
||||||
$ pda mv secret-key@mve moved-key@mve
|
$ pda mv secret-key@mve moved-key@mve
|
||||||
|
ok renamed secret-key@mve to moved-key@mve
|
||||||
$ pda get moved-key@mve
|
$ pda get moved-key@mve
|
||||||
hidden-value
|
hidden-value
|
||||||
$ pda get secret-key@mve --> FAIL
|
$ pda get secret-key@mve --> FAIL
|
||||||
|
|
|
||||||
8
testdata/mv-safe.ct
vendored
Normal file
8
testdata/mv-safe.ct
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
$ pda set src@msf hello
|
||||||
|
$ pda set dst@msf existing
|
||||||
|
$ pda mv src@msf dst@msf --safe
|
||||||
|
info skipped 'dst@msf': already exists
|
||||||
|
$ pda get src@msf
|
||||||
|
hello
|
||||||
|
$ pda get dst@msf
|
||||||
|
existing
|
||||||
7
testdata/mv-store-copy.ct
vendored
Normal file
7
testdata/mv-store-copy.ct
vendored
Normal file
|
|
@ -0,0 +1,7 @@
|
||||||
|
$ pda set key@msc1 value
|
||||||
|
$ pda move-store msc1 msc2 --copy
|
||||||
|
ok copied @msc1 to @msc2
|
||||||
|
$ pda get key@msc1
|
||||||
|
value
|
||||||
|
$ pda get key@msc2
|
||||||
|
value
|
||||||
2
testdata/mv-store-missing-err.ct
vendored
Normal file
2
testdata/mv-store-missing-err.ct
vendored
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
$ pda move-store nonexistent dest --> FAIL
|
||||||
|
FAIL cannot rename store 'nonexistent': no such store
|
||||||
8
testdata/mv-store-safe.ct
vendored
Normal file
8
testdata/mv-store-safe.ct
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
$ pda set a@mssf1 1
|
||||||
|
$ pda set b@mssf2 2
|
||||||
|
$ pda move-store mssf1 mssf2 --safe
|
||||||
|
info skipped '@mssf2': already exists
|
||||||
|
$ pda get a@mssf1
|
||||||
|
1
|
||||||
|
$ pda get b@mssf2
|
||||||
|
2
|
||||||
3
testdata/mv-store-same-err.ct
vendored
Normal file
3
testdata/mv-store-same-err.ct
vendored
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
$ pda set a@mss same
|
||||||
|
$ pda move-store mss mss --> FAIL
|
||||||
|
FAIL cannot rename store 'mss': source and destination are the same
|
||||||
5
testdata/mv-store.ct
vendored
Normal file
5
testdata/mv-store.ct
vendored
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
$ pda set key@mvs1 value
|
||||||
|
$ pda move-store mvs1 mvs2
|
||||||
|
ok renamed @mvs1 to @mvs2
|
||||||
|
$ pda get key@mvs2
|
||||||
|
value
|
||||||
1
testdata/mv.ct
vendored
1
testdata/mv.ct
vendored
|
|
@ -1,6 +1,7 @@
|
||||||
# Basic move
|
# Basic move
|
||||||
$ pda set source@mvok value
|
$ pda set source@mvok value
|
||||||
$ pda mv source@mvok dest@mvok
|
$ pda mv source@mvok dest@mvok
|
||||||
|
ok renamed source@mvok to dest@mvok
|
||||||
$ pda get dest@mvok
|
$ pda get dest@mvok
|
||||||
value
|
value
|
||||||
$ pda get source@mvok --> FAIL
|
$ pda get source@mvok --> FAIL
|
||||||
|
|
|
||||||
1
testdata/root.ct
vendored
1
testdata/root.ct
vendored
|
|
@ -25,6 +25,7 @@ Store commands:
|
||||||
export Export store as NDJSON (alias for list --format ndjson)
|
export Export store as NDJSON (alias for list --format ndjson)
|
||||||
import Restore key/value pairs from an NDJSON dump
|
import Restore key/value pairs from an NDJSON dump
|
||||||
list-stores List all stores
|
list-stores List all stores
|
||||||
|
move-store Rename a store
|
||||||
remove-store Delete a store
|
remove-store Delete a store
|
||||||
|
|
||||||
Git commands:
|
Git commands:
|
||||||
|
|
|
||||||
1
testdata/set-safe.ct
vendored
1
testdata/set-safe.ct
vendored
|
|
@ -2,6 +2,7 @@ $ pda set key@ss "original" --safe
|
||||||
$ pda get key@ss
|
$ pda get key@ss
|
||||||
"original"
|
"original"
|
||||||
$ pda set key@ss "overwritten" --safe
|
$ pda set key@ss "overwritten" --safe
|
||||||
|
info skipped 'key@ss': already exists
|
||||||
$ pda get key@ss
|
$ pda get key@ss
|
||||||
"original"
|
"original"
|
||||||
$ pda set newkey@ss "fresh" --safe
|
$ pda set newkey@ss "fresh" --safe
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue