2011-01-28 07:07:18 +00:00
|
|
|
/*
|
|
|
|
Copyright 2011 Google Inc.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2011-01-02 22:36:03 +00:00
|
|
|
package client
|
2010-12-20 22:54:41 +00:00
|
|
|
|
|
|
|
import (
|
2011-01-17 05:16:26 +00:00
|
|
|
"camli/blobref"
|
2010-12-20 22:54:41 +00:00
|
|
|
"flag"
|
2011-01-17 05:16:26 +00:00
|
|
|
"json"
|
2010-12-20 22:54:41 +00:00
|
|
|
"log"
|
2011-01-17 05:16:26 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
2010-12-20 22:54:41 +00:00
|
|
|
"strings"
|
2011-01-17 05:16:26 +00:00
|
|
|
"sync"
|
|
|
|
"syscall"
|
2010-12-20 22:54:41 +00:00
|
|
|
)
|
|
|
|
|
2011-01-17 05:16:26 +00:00
|
|
|
// These override the JSON config file ~/.camli/config "server" and
|
2010-12-20 22:54:41 +00:00
|
|
|
// "password" keys
|
|
|
|
var flagServer *string = flag.String("blobserver", "", "camlistore blob server")
|
|
|
|
var flagPassword *string = flag.String("password", "", "password for blob server")
|
|
|
|
|
2011-01-18 02:28:38 +00:00
|
|
|
func ConfigDir() string {
|
|
|
|
return path.Join(os.Getenv("HOME"), ".camli")
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConfigFilePath() string {
|
2011-01-17 05:16:26 +00:00
|
|
|
return path.Join(os.Getenv("HOME"), ".camli", "config")
|
|
|
|
}
|
|
|
|
|
|
|
|
var configOnce sync.Once
|
|
|
|
var config = make(map[string]interface{})
|
|
|
|
func parseConfig() {
|
2011-01-18 02:28:38 +00:00
|
|
|
f, err := os.Open(ConfigFilePath(), os.O_RDONLY, 0)
|
2011-01-17 05:16:26 +00:00
|
|
|
switch {
|
|
|
|
case err != nil && err.(*os.PathError).Error.(os.Errno) == syscall.ENOENT:
|
|
|
|
// TODO: write empty file?
|
|
|
|
return
|
|
|
|
case err != nil:
|
2011-01-18 02:28:38 +00:00
|
|
|
log.Printf("Error opening config file %q: %v", ConfigFilePath(), err)
|
2011-01-17 05:16:26 +00:00
|
|
|
return
|
|
|
|
default:
|
|
|
|
defer f.Close()
|
|
|
|
dj := json.NewDecoder(f)
|
|
|
|
if err := dj.Decode(&config); err != nil {
|
2011-01-18 02:28:38 +00:00
|
|
|
log.Printf("Error parsing JSON in config file %q: %v", ConfigFilePath(), err)
|
2011-01-17 05:16:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-20 22:54:41 +00:00
|
|
|
func cleanServer(server string) string {
|
|
|
|
// Remove trailing slash if provided.
|
|
|
|
if strings.HasSuffix(server, "/") {
|
|
|
|
server = server[0 : len(server)-1]
|
|
|
|
}
|
|
|
|
// Add "http://" prefix if not present:
|
|
|
|
if !strings.HasPrefix(server, "http") {
|
|
|
|
server = "http://" + server
|
|
|
|
}
|
|
|
|
return server
|
|
|
|
}
|
|
|
|
|
2011-01-02 22:36:03 +00:00
|
|
|
func blobServerOrDie() string {
|
2010-12-20 22:54:41 +00:00
|
|
|
if *flagServer != "" {
|
|
|
|
return cleanServer(*flagServer)
|
|
|
|
}
|
2011-01-17 05:16:26 +00:00
|
|
|
configOnce.Do(parseConfig)
|
2011-01-18 02:28:38 +00:00
|
|
|
value, ok := config["blobServer"]
|
|
|
|
var server string
|
|
|
|
if ok {
|
|
|
|
server = value.(string)
|
|
|
|
}
|
|
|
|
server = cleanServer(server)
|
|
|
|
if !ok || server == "" {
|
2011-02-02 20:27:30 +00:00
|
|
|
log.Fatalf("Missing or invalid \"blobServer\" in %q", ConfigFilePath())
|
|
|
|
}
|
2011-01-18 02:28:38 +00:00
|
|
|
return server
|
2010-12-20 22:54:41 +00:00
|
|
|
}
|
|
|
|
|
2011-01-02 22:36:03 +00:00
|
|
|
func passwordOrDie() string {
|
2010-12-20 22:54:41 +00:00
|
|
|
if *flagPassword != "" {
|
|
|
|
return *flagPassword
|
|
|
|
}
|
2011-01-17 05:16:26 +00:00
|
|
|
configOnce.Do(parseConfig)
|
2011-01-18 02:28:38 +00:00
|
|
|
value, ok := config["blobServerPassword"]
|
|
|
|
var password string
|
|
|
|
if ok {
|
|
|
|
password, ok = value.(string)
|
|
|
|
}
|
|
|
|
if !ok {
|
2011-02-02 20:27:30 +00:00
|
|
|
log.Fatalf("No --password parameter specified, and no \"blobServerPassword\" defined in %q", ConfigFilePath())
|
2011-01-18 02:28:38 +00:00
|
|
|
}
|
|
|
|
if password == "" {
|
|
|
|
// TODO: provide way to override warning?
|
|
|
|
// Or make a way to do deferred errors? A blank password might
|
|
|
|
// be valid, but it might also signal the root cause of an error
|
|
|
|
// in the future.
|
|
|
|
log.Printf("Warning: blank \"blobServerPassword\" defined in %q", ConfigFilePath())
|
|
|
|
}
|
|
|
|
return password
|
2010-12-20 22:54:41 +00:00
|
|
|
}
|
|
|
|
|
2011-01-17 05:16:26 +00:00
|
|
|
// Returns blobref of signer's public key, or nil if unconfigured.
|
|
|
|
func (c *Client) SignerPublicKeyBlobref() *blobref.BlobRef {
|
|
|
|
configOnce.Do(parseConfig)
|
|
|
|
key := "publicKeyBlobref"
|
|
|
|
v, ok := config[key]
|
|
|
|
if !ok {
|
2011-01-26 21:18:28 +00:00
|
|
|
log.Printf("No key %q in JSON configuration file %q; have you run \"camput --init\"?", key, ConfigFilePath())
|
2011-01-17 05:16:26 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
s, ok := v.(string)
|
|
|
|
if !ok {
|
|
|
|
log.Printf("Expected a string value for key %q in JSON file %q",
|
2011-01-18 02:28:38 +00:00
|
|
|
key, ConfigFilePath())
|
2011-01-17 05:16:26 +00:00
|
|
|
}
|
|
|
|
ref := blobref.Parse(s)
|
|
|
|
if ref == nil {
|
|
|
|
log.Printf("Bogus value %#v for key %q in file %q; not a valid blobref",
|
2011-01-18 02:28:38 +00:00
|
|
|
s, key, ConfigFilePath())
|
2011-01-17 05:16:26 +00:00
|
|
|
}
|
|
|
|
return ref
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) GetBlobFetcher() blobref.Fetcher {
|
|
|
|
// TODO: make a NewSeriesAttemptFetcher(...all configured fetch paths...)
|
2011-01-18 18:29:38 +00:00
|
|
|
return blobref.NewSimpleDirectoryFetcher(path.Join(os.Getenv("HOME"), ".camli", "keyblobs"))
|
2011-01-17 05:16:26 +00:00
|
|
|
}
|