2019-01-06 06:44:33 +08:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2019-01-06 20:26:48 +08:00
|
|
|
"crypto/rand"
|
2019-01-08 16:57:24 +08:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2019-01-08 04:24:23 +08:00
|
|
|
"log"
|
2019-01-06 06:44:33 +08:00
|
|
|
"os"
|
2019-01-08 16:57:24 +08:00
|
|
|
"path/filepath"
|
2019-01-06 06:44:33 +08:00
|
|
|
|
|
|
|
"github.com/asdine/storm"
|
|
|
|
"github.com/filebrowser/filebrowser/v2/storage"
|
|
|
|
"github.com/filebrowser/filebrowser/v2/storage/bolt"
|
|
|
|
"github.com/spf13/cobra"
|
2019-01-06 13:11:15 +08:00
|
|
|
"github.com/spf13/pflag"
|
2019-01-08 16:57:24 +08:00
|
|
|
yaml "gopkg.in/yaml.v2"
|
2019-01-06 06:44:33 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
func checkErr(err error) {
|
|
|
|
if err != nil {
|
2019-01-07 05:06:56 +08:00
|
|
|
log.Fatal(err)
|
2019-01-06 06:44:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 22:07:55 +08:00
|
|
|
func mustGetString(flags *pflag.FlagSet, flag string) string {
|
|
|
|
s, err := flags.GetString(flag)
|
2019-01-06 06:44:33 +08:00
|
|
|
checkErr(err)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2019-01-08 22:07:55 +08:00
|
|
|
func mustGetBool(flags *pflag.FlagSet, flag string) bool {
|
|
|
|
b, err := flags.GetBool(flag)
|
2019-01-06 06:44:33 +08:00
|
|
|
checkErr(err)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2019-01-08 22:07:55 +08:00
|
|
|
func mustGetUint(flags *pflag.FlagSet, flag string) uint {
|
|
|
|
b, err := flags.GetUint(flag)
|
2019-01-06 06:44:33 +08:00
|
|
|
checkErr(err)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2019-01-06 20:26:48 +08:00
|
|
|
func generateRandomBytes(n int) []byte {
|
|
|
|
b := make([]byte, n)
|
|
|
|
_, err := rand.Read(b)
|
|
|
|
checkErr(err)
|
|
|
|
// Note that err == nil only if we read len(b) bytes.
|
|
|
|
return b
|
|
|
|
}
|
2019-01-08 04:24:23 +08:00
|
|
|
|
|
|
|
type cobraFunc func(cmd *cobra.Command, args []string)
|
|
|
|
type pythonFunc func(cmd *cobra.Command, args []string, data pythonData)
|
|
|
|
|
|
|
|
type pythonConfig struct {
|
2019-01-08 04:34:44 +08:00
|
|
|
noDB bool
|
|
|
|
allowNoDB bool
|
2019-01-08 04:24:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type pythonData struct {
|
|
|
|
hadDB bool
|
|
|
|
store *storage.Storage
|
|
|
|
}
|
|
|
|
|
2019-01-10 05:37:47 +08:00
|
|
|
func dbExists(path string) (bool, error) {
|
|
|
|
stat, err := os.Stat(path)
|
|
|
|
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return false, nil
|
|
|
|
} else if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if stat.Size() == 0 {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2019-01-08 04:24:23 +08:00
|
|
|
func python(fn pythonFunc, cfg pythonConfig) cobraFunc {
|
|
|
|
return func(cmd *cobra.Command, args []string) {
|
|
|
|
data := pythonData{hadDB: true}
|
|
|
|
|
2019-01-09 04:10:27 +08:00
|
|
|
path := getParam(cmd.Flags(), "database")
|
2019-01-10 05:37:47 +08:00
|
|
|
exists, err := dbExists(path)
|
2019-01-08 04:24:23 +08:00
|
|
|
|
2019-01-10 05:37:47 +08:00
|
|
|
if err != nil {
|
2019-01-08 04:24:23 +08:00
|
|
|
panic(err)
|
2019-01-10 05:37:47 +08:00
|
|
|
} else if exists && cfg.noDB {
|
2019-01-08 04:24:23 +08:00
|
|
|
log.Fatal(path + " already exists")
|
2019-01-10 05:37:47 +08:00
|
|
|
} else if !exists && !cfg.noDB && !cfg.allowNoDB {
|
|
|
|
log.Fatal(path + " does not exist. Please run 'filebrowser config init' first.")
|
2019-01-08 04:24:23 +08:00
|
|
|
}
|
|
|
|
|
2019-01-10 05:37:47 +08:00
|
|
|
data.hadDB = exists
|
2019-01-08 04:24:23 +08:00
|
|
|
db, err := storm.Open(path)
|
|
|
|
checkErr(err)
|
|
|
|
defer db.Close()
|
2019-01-10 05:37:47 +08:00
|
|
|
data.store, err = bolt.NewStorage(db)
|
|
|
|
checkErr(err)
|
2019-01-08 04:24:23 +08:00
|
|
|
fn(cmd, args, data)
|
|
|
|
}
|
|
|
|
}
|
2019-01-08 16:57:24 +08:00
|
|
|
|
|
|
|
func marshal(filename string, data interface{}) error {
|
|
|
|
fd, err := os.Create(filename)
|
|
|
|
checkErr(err)
|
|
|
|
defer fd.Close()
|
|
|
|
|
|
|
|
switch ext := filepath.Ext(filename); ext {
|
|
|
|
case ".json":
|
|
|
|
encoder := json.NewEncoder(fd)
|
|
|
|
encoder.SetIndent("", " ")
|
|
|
|
return encoder.Encode(data)
|
|
|
|
case ".yml", ".yaml":
|
|
|
|
encoder := yaml.NewEncoder(fd)
|
|
|
|
return encoder.Encode(data)
|
|
|
|
default:
|
|
|
|
return errors.New("invalid format: " + ext)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func unmarshal(filename string, data interface{}) error {
|
|
|
|
fd, err := os.Open(filename)
|
|
|
|
checkErr(err)
|
|
|
|
defer fd.Close()
|
|
|
|
|
|
|
|
switch ext := filepath.Ext(filename); ext {
|
|
|
|
case ".json":
|
|
|
|
return json.NewDecoder(fd).Decode(data)
|
|
|
|
case ".yml", ".yaml":
|
|
|
|
return yaml.NewDecoder(fd).Decode(data)
|
|
|
|
default:
|
|
|
|
return errors.New("invalid format: " + ext)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func jsonYamlArg(cmd *cobra.Command, args []string) error {
|
|
|
|
if err := cobra.ExactArgs(1)(cmd, args); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ext := filepath.Ext(args[0]); ext {
|
|
|
|
case ".json", ".yml", ".yaml":
|
|
|
|
return nil
|
|
|
|
default:
|
|
|
|
return errors.New("invalid format: " + ext)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func cleanUpInterfaceMap(in map[interface{}]interface{}) map[string]interface{} {
|
|
|
|
result := make(map[string]interface{})
|
|
|
|
for k, v := range in {
|
|
|
|
result[fmt.Sprintf("%v", k)] = cleanUpMapValue(v)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func cleanUpInterfaceArray(in []interface{}) []interface{} {
|
|
|
|
result := make([]interface{}, len(in))
|
|
|
|
for i, v := range in {
|
|
|
|
result[i] = cleanUpMapValue(v)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func cleanUpMapValue(v interface{}) interface{} {
|
|
|
|
switch v := v.(type) {
|
|
|
|
case []interface{}:
|
|
|
|
return cleanUpInterfaceArray(v)
|
|
|
|
case map[interface{}]interface{}:
|
|
|
|
return cleanUpInterfaceMap(v)
|
|
|
|
default:
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
}
|