2012-03-15 12:31:06 +00:00
|
|
|
/*
|
|
|
|
Copyright 2012 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package serverconfig
|
|
|
|
|
|
|
|
import (
|
2012-08-04 01:12:39 +00:00
|
|
|
"errors"
|
2012-03-15 12:31:06 +00:00
|
|
|
"fmt"
|
2013-08-09 00:50:50 +00:00
|
|
|
"net/url"
|
2012-03-15 12:31:06 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
"camlistore.org/pkg/blob"
|
2012-03-15 12:31:06 +00:00
|
|
|
"camlistore.org/pkg/jsonconfig"
|
2012-04-12 18:39:53 +00:00
|
|
|
"camlistore.org/pkg/jsonsign"
|
2013-09-01 16:50:35 +00:00
|
|
|
"camlistore.org/pkg/osutil"
|
2012-03-15 12:31:06 +00:00
|
|
|
)
|
|
|
|
|
2013-04-08 13:50:50 +00:00
|
|
|
const (
|
|
|
|
DefaultTLSCert = "config/selfgen_pem.crt"
|
|
|
|
DefaultTLSKey = "config/selfgen_pem.key"
|
|
|
|
)
|
|
|
|
|
2012-03-15 12:31:06 +00:00
|
|
|
// various parameters derived from the high-level user config
|
|
|
|
// and needed to set up the low-level config.
|
|
|
|
type configPrefixesParams struct {
|
2013-07-16 15:55:20 +00:00
|
|
|
secretRing string
|
|
|
|
keyId string
|
|
|
|
indexerPath string
|
|
|
|
blobPath string
|
2013-08-04 02:54:30 +00:00
|
|
|
searchOwner blob.Ref
|
2013-07-16 15:55:20 +00:00
|
|
|
shareHandlerPath string
|
2013-11-19 04:53:46 +00:00
|
|
|
flickr string
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2013-08-25 17:25:30 +00:00
|
|
|
var (
|
|
|
|
tempDir = os.TempDir
|
|
|
|
noMkdir bool // for tests to not call os.Mkdir
|
|
|
|
)
|
2013-01-11 18:52:22 +00:00
|
|
|
|
2013-06-28 15:15:48 +00:00
|
|
|
func addPublishedConfig(prefixes jsonconfig.Obj,
|
|
|
|
published jsonconfig.Obj,
|
|
|
|
sourceRoot string) ([]interface{}, error) {
|
2012-04-22 15:33:22 +00:00
|
|
|
pubPrefixes := []interface{}{}
|
|
|
|
for k, v := range published {
|
|
|
|
p, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("Wrong type for %s; was expecting map[string]interface{}, got %T", k, v)
|
|
|
|
}
|
|
|
|
rootName := strings.Replace(k, "/", "", -1) + "Root"
|
2013-10-28 15:59:07 +00:00
|
|
|
rootPermanode, goTemplate, style, js := "", "", "", ""
|
2012-04-22 15:33:22 +00:00
|
|
|
for pk, pv := range p {
|
|
|
|
val, ok := pv.(string)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("Was expecting type string for %s, got %T", pk, pv)
|
|
|
|
}
|
|
|
|
switch pk {
|
|
|
|
case "rootPermanode":
|
|
|
|
rootPermanode = val
|
2013-10-28 15:59:07 +00:00
|
|
|
case "goTemplate":
|
|
|
|
goTemplate = val
|
2012-04-22 15:33:22 +00:00
|
|
|
case "style":
|
|
|
|
style = val
|
2013-10-28 15:59:07 +00:00
|
|
|
case "js":
|
|
|
|
js = val
|
2012-04-22 15:33:22 +00:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("Unexpected key %q in config for %s", pk, k)
|
|
|
|
}
|
|
|
|
}
|
2013-10-28 15:59:07 +00:00
|
|
|
if rootPermanode == "" || goTemplate == "" {
|
|
|
|
return nil, fmt.Errorf("Missing key in configuration for %s, need \"rootPermanode\" and \"goTemplate\"", k)
|
2012-04-22 15:33:22 +00:00
|
|
|
}
|
|
|
|
ob := map[string]interface{}{}
|
|
|
|
ob["handler"] = "publish"
|
|
|
|
handlerArgs := map[string]interface{}{
|
2012-05-01 23:02:07 +00:00
|
|
|
"rootName": rootName,
|
|
|
|
"blobRoot": "/bs-and-maybe-also-index/",
|
|
|
|
"searchRoot": "/my-search/",
|
|
|
|
"cache": "/cache/",
|
|
|
|
"rootPermanode": []interface{}{"/sighelper/", rootPermanode},
|
2012-04-22 15:33:22 +00:00
|
|
|
}
|
2013-06-28 15:15:48 +00:00
|
|
|
if sourceRoot != "" {
|
|
|
|
handlerArgs["sourceRoot"] = sourceRoot
|
|
|
|
}
|
2013-10-28 15:59:07 +00:00
|
|
|
handlerArgs["goTemplate"] = goTemplate
|
|
|
|
handlerArgs["css"] = []interface{}{style}
|
|
|
|
handlerArgs["js"] = []interface{}{js}
|
|
|
|
handlerArgs["scaledImage"] = "lrucache"
|
2012-04-22 15:33:22 +00:00
|
|
|
ob["handlerArgs"] = handlerArgs
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes[k] = ob
|
2012-04-22 15:33:22 +00:00
|
|
|
pubPrefixes = append(pubPrefixes, k)
|
|
|
|
}
|
|
|
|
return pubPrefixes, nil
|
|
|
|
}
|
|
|
|
|
2013-06-20 21:33:00 +00:00
|
|
|
func addUIConfig(prefixes jsonconfig.Obj,
|
|
|
|
uiPrefix string,
|
|
|
|
published []interface{},
|
|
|
|
sourceRoot string) {
|
2012-03-15 12:31:06 +00:00
|
|
|
ob := map[string]interface{}{}
|
|
|
|
ob["handler"] = "ui"
|
|
|
|
handlerArgs := map[string]interface{}{
|
|
|
|
"jsonSignRoot": "/sighelper/",
|
|
|
|
"cache": "/cache/",
|
|
|
|
"scaledImage": "lrucache",
|
|
|
|
}
|
|
|
|
if len(published) > 0 {
|
|
|
|
handlerArgs["publishRoots"] = published
|
|
|
|
}
|
2013-06-20 21:33:00 +00:00
|
|
|
if sourceRoot != "" {
|
|
|
|
handlerArgs["sourceRoot"] = sourceRoot
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
ob["handlerArgs"] = handlerArgs
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes[uiPrefix] = ob
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addMongoConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) {
|
2012-10-23 19:45:46 +00:00
|
|
|
fields := strings.Split(dbinfo, "@")
|
|
|
|
if len(fields) != 2 {
|
|
|
|
exitFailure("Malformed mongo config string. Got \"%v\", want: \"user:password@host\"", dbinfo)
|
|
|
|
}
|
|
|
|
host := fields[1]
|
|
|
|
fields = strings.Split(fields[0], ":")
|
|
|
|
if len(fields) != 2 {
|
|
|
|
exitFailure("Malformed mongo config string. Got \"%v\", want: \"user:password\"", fields[0])
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
ob := map[string]interface{}{}
|
|
|
|
ob["enabled"] = true
|
|
|
|
ob["handler"] = "storage-mongodbindexer"
|
|
|
|
ob["handlerArgs"] = map[string]interface{}{
|
2012-10-23 19:45:46 +00:00
|
|
|
"host": host,
|
|
|
|
"user": fields[0],
|
|
|
|
"password": fields[1],
|
2012-03-15 12:31:06 +00:00
|
|
|
"database": dbname,
|
|
|
|
"blobSource": "/bs/",
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes["/index-mongo/"] = ob
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addSQLConfig(rdbms string, prefixes jsonconfig.Obj, dbname string, dbinfo string) {
|
2012-03-15 12:31:06 +00:00
|
|
|
fields := strings.Split(dbinfo, "@")
|
|
|
|
if len(fields) != 2 {
|
2012-11-03 18:58:50 +00:00
|
|
|
exitFailure("Malformed " + rdbms + " config string. Want: \"user@host:password\"")
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
user := fields[0]
|
|
|
|
fields = strings.Split(fields[1], ":")
|
|
|
|
if len(fields) != 2 {
|
2012-11-03 18:58:50 +00:00
|
|
|
exitFailure("Malformed " + rdbms + " config string. Want: \"user@host:password\"")
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
ob := map[string]interface{}{}
|
|
|
|
ob["enabled"] = true
|
2012-11-03 18:58:50 +00:00
|
|
|
ob["handler"] = "storage-" + rdbms + "indexer"
|
2012-03-15 12:31:06 +00:00
|
|
|
ob["handlerArgs"] = map[string]interface{}{
|
|
|
|
"host": fields[0],
|
|
|
|
"user": user,
|
|
|
|
"password": fields[1],
|
|
|
|
"database": dbname,
|
|
|
|
"blobSource": "/bs/",
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes["/index-"+rdbms+"/"] = ob
|
2012-11-03 18:58:50 +00:00
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addPostgresConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) {
|
2012-11-03 18:58:50 +00:00
|
|
|
addSQLConfig("postgres", prefixes, dbname, dbinfo)
|
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addMySQLConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) {
|
2012-11-03 18:58:50 +00:00
|
|
|
addSQLConfig("mysql", prefixes, dbname, dbinfo)
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addMemindexConfig(prefixes jsonconfig.Obj) {
|
2012-03-15 12:31:06 +00:00
|
|
|
ob := map[string]interface{}{}
|
|
|
|
ob["handler"] = "storage-memory-only-dev-indexer"
|
|
|
|
ob["handlerArgs"] = map[string]interface{}{
|
|
|
|
"blobSource": "/bs/",
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes["/index-mem/"] = ob
|
|
|
|
}
|
|
|
|
|
2013-01-10 23:29:08 +00:00
|
|
|
func addSQLiteConfig(prefixes jsonconfig.Obj, file string) {
|
|
|
|
ob := map[string]interface{}{}
|
2013-01-11 00:16:10 +00:00
|
|
|
ob["handler"] = "storage-sqliteindexer"
|
2013-01-10 23:29:08 +00:00
|
|
|
ob["handlerArgs"] = map[string]interface{}{
|
|
|
|
"blobSource": "/bs/",
|
|
|
|
"file": file,
|
|
|
|
}
|
|
|
|
prefixes["/index-sqlite/"] = ob
|
|
|
|
}
|
|
|
|
|
2013-08-25 17:25:30 +00:00
|
|
|
func addKVConfig(prefixes jsonconfig.Obj, file string) {
|
|
|
|
prefixes["/index-kv/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-kvfileindexer",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"blobSource": "/bs/",
|
|
|
|
"file": file,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-22 00:13:36 +00:00
|
|
|
func addS3Config(prefixes jsonconfig.Obj, s3 string) error {
|
2013-09-01 16:50:35 +00:00
|
|
|
f := strings.SplitN(s3, ":", 4)
|
|
|
|
if len(f) < 3 {
|
2012-12-22 00:13:36 +00:00
|
|
|
return errors.New(`genconfig: expected "s3" field to be of form "access_key_id:secret_access_key:bucket"`)
|
|
|
|
}
|
|
|
|
accessKey, secret, bucket := f[0], f[1], f[2]
|
2013-09-01 16:50:35 +00:00
|
|
|
var hostname string
|
|
|
|
if len(f) == 4 {
|
|
|
|
hostname = f[3]
|
|
|
|
}
|
2013-01-08 18:44:59 +00:00
|
|
|
isPrimary := false
|
|
|
|
if _, ok := prefixes["/bs/"]; !ok {
|
|
|
|
isPrimary = true
|
|
|
|
}
|
|
|
|
s3Prefix := ""
|
|
|
|
if isPrimary {
|
|
|
|
s3Prefix = "/bs/"
|
|
|
|
} else {
|
|
|
|
s3Prefix = "/sto-s3/"
|
|
|
|
}
|
2013-09-01 16:50:35 +00:00
|
|
|
args := map[string]interface{}{
|
|
|
|
"aws_access_key": accessKey,
|
|
|
|
"aws_secret_access_key": secret,
|
|
|
|
"bucket": bucket,
|
|
|
|
}
|
|
|
|
if hostname != "" {
|
|
|
|
args["hostname"] = hostname
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
prefixes[s3Prefix] = map[string]interface{}{
|
2013-09-01 16:50:35 +00:00
|
|
|
"handler": "storage-s3",
|
|
|
|
"handlerArgs": args,
|
2012-12-22 00:13:36 +00:00
|
|
|
}
|
2013-01-08 18:44:59 +00:00
|
|
|
if isPrimary {
|
|
|
|
// TODO(mpl): s3CacheBucket
|
|
|
|
// See http://code.google.com/p/camlistore/issues/detail?id=85
|
|
|
|
prefixes["/cache/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-filesystem",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
2013-01-11 18:52:22 +00:00
|
|
|
"path": filepath.Join(tempDir(), "camli-cache"),
|
2013-01-08 18:44:59 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
prefixes["/sync-to-s3/"] = map[string]interface{}{
|
|
|
|
"handler": "sync",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"from": "/bs/",
|
|
|
|
"to": s3Prefix,
|
|
|
|
},
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
}
|
|
|
|
return nil
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2013-08-11 15:07:18 +00:00
|
|
|
func addGoogleDriveConfig(prefixes jsonconfig.Obj, highCfg string) error {
|
2013-07-06 20:29:17 +00:00
|
|
|
f := strings.SplitN(highCfg, ":", 4)
|
|
|
|
if len(f) != 4 {
|
2013-08-11 15:07:18 +00:00
|
|
|
return errors.New(`genconfig: expected "googledrive" field to be of form "client_id:client_secret:refresh_token:parent_id"`)
|
|
|
|
}
|
|
|
|
clientId, secret, refreshToken, parentId := f[0], f[1], f[2], f[3]
|
|
|
|
|
|
|
|
isPrimary := false
|
|
|
|
if _, ok := prefixes["/bs/"]; !ok {
|
|
|
|
isPrimary = true
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix := ""
|
|
|
|
if isPrimary {
|
|
|
|
prefix = "/bs/"
|
|
|
|
} else {
|
|
|
|
prefix = "/sto-googledrive/"
|
|
|
|
}
|
|
|
|
prefixes[prefix] = map[string]interface{}{
|
|
|
|
"handler": "storage-googledrive",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"parent_id": parentId,
|
|
|
|
"auth": map[string]interface{}{
|
|
|
|
"client_id": clientId,
|
|
|
|
"client_secret": secret,
|
|
|
|
"refresh_token": refreshToken,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if isPrimary {
|
|
|
|
prefixes["/cache/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-filesystem",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"path": filepath.Join(tempDir(), "camli-cache"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
prefixes["/sync-to-googledrive/"] = map[string]interface{}{
|
|
|
|
"handler": "sync",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"from": "/bs/",
|
|
|
|
"to": prefix,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func addGoogleCloudStorageConfig(prefixes jsonconfig.Obj, highCfg string) error {
|
|
|
|
f := strings.SplitN(highCfg, ":", 4)
|
|
|
|
if len(f) != 4 {
|
|
|
|
return errors.New(`genconfig: expected "googlecloudstorage" field to be of form "client_id:client_secret:refresh_token:bucket"`)
|
2013-07-06 20:29:17 +00:00
|
|
|
}
|
|
|
|
clientId, secret, refreshToken, bucket := f[0], f[1], f[2], f[3]
|
|
|
|
|
|
|
|
isPrimary := false
|
|
|
|
if _, ok := prefixes["/bs/"]; !ok {
|
|
|
|
isPrimary = true
|
|
|
|
}
|
|
|
|
|
|
|
|
gsPrefix := ""
|
|
|
|
if isPrimary {
|
|
|
|
gsPrefix = "/bs/"
|
|
|
|
} else {
|
2013-08-11 15:07:18 +00:00
|
|
|
gsPrefix = "/sto-googlecloudstorage/"
|
2013-07-06 20:29:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
prefixes[gsPrefix] = map[string]interface{}{
|
2013-08-11 15:07:18 +00:00
|
|
|
"handler": "storage-googlecloudstorage",
|
2013-07-06 20:29:17 +00:00
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"bucket": bucket,
|
|
|
|
"auth": map[string]interface{}{
|
|
|
|
"client_id": clientId,
|
|
|
|
"client_secret": secret,
|
|
|
|
"refresh_token": refreshToken,
|
|
|
|
// If high-level config is for the common user then fullSyncOnStart = true
|
|
|
|
// Then the default just works.
|
|
|
|
//"fullSyncOnStart": true,
|
|
|
|
//"blockingFullSyncOnStart": false
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if isPrimary {
|
|
|
|
// TODO: cacheBucket like s3CacheBucket?
|
|
|
|
prefixes["/cache/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-filesystem",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"path": filepath.Join(tempDir(), "camli-cache"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
} else {
|
2013-08-11 15:07:18 +00:00
|
|
|
prefixes["/sync-to-googlecloudstorage/"] = map[string]interface{}{
|
2013-07-06 20:29:17 +00:00
|
|
|
"handler": "sync",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"from": "/bs/",
|
|
|
|
"to": gsPrefix,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-06-26 19:55:47 +00:00
|
|
|
func genLowLevelPrefixes(params *configPrefixesParams, ownerName string) (m jsonconfig.Obj) {
|
2012-12-21 23:47:08 +00:00
|
|
|
m = make(jsonconfig.Obj)
|
2012-03-15 12:31:06 +00:00
|
|
|
|
2013-01-10 23:29:08 +00:00
|
|
|
haveIndex := params.indexerPath != ""
|
|
|
|
root := "/bs/"
|
|
|
|
pubKeyDest := root
|
|
|
|
if haveIndex {
|
|
|
|
root = "/bs-and-maybe-also-index/"
|
|
|
|
pubKeyDest = "/bs-and-index/"
|
|
|
|
}
|
|
|
|
|
2013-06-26 19:55:47 +00:00
|
|
|
rootArgs := map[string]interface{}{
|
|
|
|
"stealth": false,
|
|
|
|
"blobRoot": root,
|
|
|
|
"statusRoot": "/status/",
|
|
|
|
}
|
|
|
|
if ownerName != "" {
|
|
|
|
rootArgs["ownerName"] = ownerName
|
|
|
|
}
|
2012-12-21 23:47:08 +00:00
|
|
|
m["/"] = map[string]interface{}{
|
2013-06-26 19:55:47 +00:00
|
|
|
"handler": "root",
|
|
|
|
"handlerArgs": rootArgs,
|
2012-11-08 14:27:17 +00:00
|
|
|
}
|
2013-01-10 23:29:08 +00:00
|
|
|
if haveIndex {
|
|
|
|
setMap(m, "/", "handlerArgs", "searchRoot", "/my-search/")
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
|
2012-12-21 23:47:08 +00:00
|
|
|
m["/setup/"] = map[string]interface{}{
|
|
|
|
"handler": "setup",
|
|
|
|
}
|
2012-04-03 21:39:49 +00:00
|
|
|
|
2013-06-05 17:18:27 +00:00
|
|
|
m["/status/"] = map[string]interface{}{
|
|
|
|
"handler": "status",
|
|
|
|
}
|
|
|
|
|
2013-07-16 15:55:20 +00:00
|
|
|
if params.shareHandlerPath != "" {
|
|
|
|
m[params.shareHandlerPath] = map[string]interface{}{
|
2013-02-28 23:30:16 +00:00
|
|
|
"handler": "share",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"blobRoot": "/bs/",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-21 23:47:08 +00:00
|
|
|
m["/sighelper/"] = map[string]interface{}{
|
|
|
|
"handler": "jsonsign",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"secretRing": params.secretRing,
|
|
|
|
"keyId": params.keyId,
|
2013-01-10 23:29:08 +00:00
|
|
|
"publicKeyDest": pubKeyDest,
|
2012-03-15 12:31:06 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2013-01-08 18:44:59 +00:00
|
|
|
if params.blobPath != "" {
|
|
|
|
m["/bs/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-filesystem",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"path": params.blobPath,
|
|
|
|
},
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
|
2013-01-08 18:44:59 +00:00
|
|
|
m["/cache/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-filesystem",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"path": filepath.Join(params.blobPath, "/cache"),
|
|
|
|
},
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2013-11-19 04:53:46 +00:00
|
|
|
if params.flickr != "" {
|
2013-11-17 21:52:45 +00:00
|
|
|
m["/importer-flickr/"] = map[string]interface{}{
|
2013-11-19 04:53:46 +00:00
|
|
|
"apiKey": params.flickr,
|
2013-11-17 21:52:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-10 23:29:08 +00:00
|
|
|
if haveIndex {
|
2013-08-20 18:11:37 +00:00
|
|
|
syncArgs := map[string]interface{}{
|
|
|
|
"from": "/bs/",
|
|
|
|
"to": params.indexerPath,
|
|
|
|
}
|
|
|
|
// TODO(mpl): Brad says the cond should be dest == /index-*.
|
|
|
|
// But what about when dest is index-mem and we have a local disk;
|
|
|
|
// don't we want to have an active synchandler to do the fullSyncOnStart?
|
|
|
|
// Anyway, that condition works for now.
|
|
|
|
if params.blobPath == "" {
|
|
|
|
// When our primary blob store is remote (s3 or google cloud),
|
|
|
|
// i.e not an efficient replication source, we do not want the
|
|
|
|
// synchandler to mirror to the indexer. But we still want a
|
|
|
|
// synchandler to provide the discovery for e.g tools like
|
|
|
|
// camtool sync. See http://camlistore.org/issue/201
|
|
|
|
syncArgs["idle"] = true
|
|
|
|
}
|
2013-01-10 23:29:08 +00:00
|
|
|
m["/sync/"] = map[string]interface{}{
|
2013-08-20 18:11:37 +00:00
|
|
|
"handler": "sync",
|
|
|
|
"handlerArgs": syncArgs,
|
2013-01-10 23:29:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m["/bs-and-index/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-replica",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"backends": []interface{}{"/bs/", params.indexerPath},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
m["/bs-and-maybe-also-index/"] = map[string]interface{}{
|
|
|
|
"handler": "storage-cond",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"write": map[string]interface{}{
|
|
|
|
"if": "isSchema",
|
|
|
|
"then": "/bs-and-index/",
|
|
|
|
"else": "/bs/",
|
|
|
|
},
|
|
|
|
"read": "/bs/",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
m["/my-search/"] = map[string]interface{}{
|
|
|
|
"handler": "search",
|
|
|
|
"handlerArgs": map[string]interface{}{
|
|
|
|
"index": params.indexerPath,
|
|
|
|
"owner": params.searchOwner.String(),
|
|
|
|
},
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-12-21 23:47:08 +00:00
|
|
|
return
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 16:51:42 +00:00
|
|
|
// genLowLevelConfig returns a low-level config from a high-level config.
|
|
|
|
func genLowLevelConfig(conf *Config) (lowLevelConf *Config, err error) {
|
2012-03-19 20:09:00 +00:00
|
|
|
var (
|
2012-08-04 11:42:10 +00:00
|
|
|
baseURL = conf.OptionalString("baseURL", "")
|
2012-09-16 22:20:49 +00:00
|
|
|
listen = conf.OptionalString("listen", "")
|
2012-03-19 20:09:00 +00:00
|
|
|
auth = conf.RequiredString("auth")
|
2012-03-20 04:31:20 +00:00
|
|
|
keyId = conf.RequiredString("identity")
|
|
|
|
secretRing = conf.RequiredString("identitySecretRing")
|
2012-10-18 09:57:46 +00:00
|
|
|
tlsOn = conf.OptionalBool("https", false)
|
|
|
|
tlsCert = conf.OptionalString("HTTPSCertFile", "")
|
|
|
|
tlsKey = conf.OptionalString("HTTPSKeyFile", "")
|
2013-01-10 23:29:08 +00:00
|
|
|
|
|
|
|
// Blob storage options
|
2013-08-11 15:07:18 +00:00
|
|
|
blobPath = conf.OptionalString("blobPath", "")
|
2013-09-01 16:50:35 +00:00
|
|
|
s3 = conf.OptionalString("s3", "") // "access_key_id:secret_access_key:bucket[:hostname]"
|
2013-08-11 15:07:18 +00:00
|
|
|
googlecloudstorage = conf.OptionalString("googlecloudstorage", "") // "clientId:clientSecret:refreshToken:bucket"
|
|
|
|
googledrive = conf.OptionalString("googledrive", "") // "clientId:clientSecret:refreshToken:parentId"
|
2013-07-16 15:55:20 +00:00
|
|
|
// Enable the share handler. If true, and shareHandlerPath is empty,
|
|
|
|
// then shareHandlerPath defaults to "/share/".
|
|
|
|
shareHandler = conf.OptionalBool("shareHandler", false)
|
|
|
|
// URL prefix for the share handler. If set, overrides shareHandler.
|
|
|
|
shareHandlerPath = conf.OptionalString("shareHandlerPath", "")
|
2013-01-10 23:29:08 +00:00
|
|
|
|
|
|
|
// Index options
|
|
|
|
runIndex = conf.OptionalBool("runIndex", true) // if false: no search, no UI, etc.
|
|
|
|
dbname = conf.OptionalString("dbname", "") // for mysql, postgres, mongo
|
2012-03-19 20:09:00 +00:00
|
|
|
mysql = conf.OptionalString("mysql", "")
|
2012-11-03 18:58:50 +00:00
|
|
|
postgres = conf.OptionalString("postgres", "")
|
2013-01-10 23:29:08 +00:00
|
|
|
memIndex = conf.OptionalBool("memIndex", false)
|
2012-03-19 20:09:00 +00:00
|
|
|
mongo = conf.OptionalString("mongo", "")
|
2013-01-10 23:29:08 +00:00
|
|
|
sqliteFile = conf.OptionalString("sqlite", "")
|
2013-08-25 17:25:30 +00:00
|
|
|
kvFile = conf.OptionalString("kvIndexFile", "")
|
2013-01-10 23:29:08 +00:00
|
|
|
|
2013-11-17 21:52:45 +00:00
|
|
|
// Importer options
|
2013-11-19 04:53:46 +00:00
|
|
|
flickr = conf.OptionalString("flickr", "")
|
2013-11-17 21:52:45 +00:00
|
|
|
|
2013-01-10 23:29:08 +00:00
|
|
|
_ = conf.OptionalList("replicateTo")
|
|
|
|
publish = conf.OptionalObject("publish")
|
2013-06-20 21:33:00 +00:00
|
|
|
// alternative source tree, to override the embedded ui and/or closure resources.
|
|
|
|
// If non empty, the ui files will be expected at
|
|
|
|
// sourceRoot + "/server/camlistored/ui" and the closure library at
|
|
|
|
// sourceRoot + "/third_party/closure/lib"
|
2013-06-28 15:15:48 +00:00
|
|
|
// Also used by the publish handler.
|
2013-06-20 21:33:00 +00:00
|
|
|
sourceRoot = conf.OptionalString("sourceRoot", "")
|
2013-06-26 19:55:47 +00:00
|
|
|
|
|
|
|
ownerName = conf.OptionalString("ownerName", "")
|
2012-03-19 20:09:00 +00:00
|
|
|
)
|
|
|
|
if err := conf.Validate(); err != nil {
|
|
|
|
return nil, err
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
2012-03-19 20:09:00 +00:00
|
|
|
|
|
|
|
obj := jsonconfig.Obj{}
|
|
|
|
if tlsOn {
|
2012-08-04 01:12:39 +00:00
|
|
|
if (tlsCert != "") != (tlsKey != "") {
|
|
|
|
return nil, errors.New("Must set both TLSCertFile and TLSKeyFile (or neither to generate a self-signed cert)")
|
|
|
|
}
|
|
|
|
if tlsCert != "" {
|
|
|
|
obj["TLSCertFile"] = tlsCert
|
|
|
|
obj["TLSKeyFile"] = tlsKey
|
|
|
|
} else {
|
2013-04-08 13:50:50 +00:00
|
|
|
obj["TLSCertFile"] = DefaultTLSCert
|
|
|
|
obj["TLSKeyFile"] = DefaultTLSKey
|
2012-08-04 01:12:39 +00:00
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-09-16 22:20:49 +00:00
|
|
|
if baseURL != "" {
|
2013-08-09 00:50:50 +00:00
|
|
|
u, err := url.Parse(baseURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Error parsing baseURL %q as a URL: %v", baseURL, err)
|
|
|
|
}
|
2013-09-08 05:32:39 +00:00
|
|
|
if u.Path != "" && u.Path != "/" {
|
|
|
|
return nil, fmt.Errorf("baseURL can't have a path, only a scheme, host, and optional port.")
|
2012-09-16 22:20:49 +00:00
|
|
|
}
|
2013-09-08 05:32:39 +00:00
|
|
|
u.Path = ""
|
|
|
|
obj["baseURL"] = u.String()
|
2012-09-16 22:20:49 +00:00
|
|
|
}
|
2012-08-04 11:42:10 +00:00
|
|
|
if listen != "" {
|
|
|
|
obj["listen"] = listen
|
|
|
|
}
|
2012-09-16 22:20:49 +00:00
|
|
|
obj["https"] = tlsOn
|
|
|
|
obj["auth"] = auth
|
2012-08-04 11:42:10 +00:00
|
|
|
|
2013-08-28 13:53:58 +00:00
|
|
|
username := ""
|
2012-03-15 12:31:06 +00:00
|
|
|
if dbname == "" {
|
2013-08-28 13:53:58 +00:00
|
|
|
username = osutil.Username()
|
2012-03-15 12:31:06 +00:00
|
|
|
if username == "" {
|
2013-08-28 13:53:58 +00:00
|
|
|
return nil, fmt.Errorf("USER (USERNAME on windows) env var not set; needed to define dbname")
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
dbname = "camli" + username
|
|
|
|
}
|
|
|
|
|
2012-04-12 02:22:02 +00:00
|
|
|
var indexerPath string
|
2013-08-25 17:25:30 +00:00
|
|
|
numIndexers := numSet(mongo, mysql, postgres, sqliteFile, memIndex, kvFile)
|
2012-04-12 02:22:02 +00:00
|
|
|
switch {
|
2013-01-10 23:29:08 +00:00
|
|
|
case runIndex && numIndexers == 0:
|
2013-08-25 17:25:30 +00:00
|
|
|
return nil, fmt.Errorf("Unless runIndex is set to false, you must specify an index option (kvIndexFile, mongo, mysql, postgres, sqlite, memIndex).")
|
2013-01-10 23:29:08 +00:00
|
|
|
case runIndex && numIndexers != 1:
|
2013-08-25 17:25:30 +00:00
|
|
|
return nil, fmt.Errorf("With runIndex set true, you can only pick exactly one indexer (mongo, mysql, postgres, sqlite, memIndex).")
|
2013-01-10 23:29:08 +00:00
|
|
|
case !runIndex && numIndexers != 0:
|
2013-08-25 17:25:30 +00:00
|
|
|
return nil, fmt.Errorf("With runIndex disabled, you can't specify any of mongo, mysql, postgres, sqlite, memIndex.")
|
2012-04-12 02:22:02 +00:00
|
|
|
case mysql != "":
|
|
|
|
indexerPath = "/index-mysql/"
|
2012-11-03 18:58:50 +00:00
|
|
|
case postgres != "":
|
|
|
|
indexerPath = "/index-postgres/"
|
2012-04-12 02:22:02 +00:00
|
|
|
case mongo != "":
|
|
|
|
indexerPath = "/index-mongo/"
|
2013-01-10 23:29:08 +00:00
|
|
|
case sqliteFile != "":
|
|
|
|
indexerPath = "/index-sqlite/"
|
2013-08-25 17:25:30 +00:00
|
|
|
case kvFile != "":
|
|
|
|
indexerPath = "/index-kv/"
|
2013-01-10 23:29:08 +00:00
|
|
|
case memIndex:
|
2012-04-12 02:22:02 +00:00
|
|
|
indexerPath = "/index-mem/"
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
|
2012-04-12 18:39:53 +00:00
|
|
|
entity, err := jsonsign.EntityFromSecring(keyId, secretRing)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
armoredPublicKey, err := jsonsign.ArmoredPublicKey(entity)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2013-01-08 18:44:59 +00:00
|
|
|
nolocaldisk := blobPath == ""
|
2013-08-20 18:11:37 +00:00
|
|
|
if nolocaldisk {
|
|
|
|
if s3 == "" && googlecloudstorage == "" {
|
|
|
|
return nil, errors.New("You need at least one of blobPath (for localdisk) or s3 or googlecloudstorage configured for a blobserver.")
|
|
|
|
}
|
|
|
|
if s3 != "" && googlecloudstorage != "" {
|
|
|
|
return nil, errors.New("Using S3 as a primary storage and Google Cloud Storage as a mirror is not supported for now.")
|
|
|
|
}
|
2013-01-08 18:44:59 +00:00
|
|
|
}
|
|
|
|
|
2013-07-16 15:55:20 +00:00
|
|
|
if shareHandler && shareHandlerPath == "" {
|
|
|
|
shareHandlerPath = "/share/"
|
|
|
|
}
|
|
|
|
|
2012-03-15 12:31:06 +00:00
|
|
|
prefixesParams := &configPrefixesParams{
|
2013-07-16 15:55:20 +00:00
|
|
|
secretRing: secretRing,
|
|
|
|
keyId: keyId,
|
|
|
|
indexerPath: indexerPath,
|
|
|
|
blobPath: blobPath,
|
2013-08-04 02:54:30 +00:00
|
|
|
searchOwner: blob.SHA1FromString(armoredPublicKey),
|
2013-07-16 15:55:20 +00:00
|
|
|
shareHandlerPath: shareHandlerPath,
|
2013-11-17 21:52:45 +00:00
|
|
|
flickr: flickr,
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2013-06-26 19:55:47 +00:00
|
|
|
prefixes := genLowLevelPrefixes(prefixesParams, ownerName)
|
2013-01-08 18:44:59 +00:00
|
|
|
var cacheDir string
|
|
|
|
if nolocaldisk {
|
|
|
|
// Whether camlistored is run from EC2 or not, we use
|
|
|
|
// a temp dir as the cache when primary storage is S3.
|
|
|
|
// TODO(mpl): s3CacheBucket
|
|
|
|
// See http://code.google.com/p/camlistore/issues/detail?id=85
|
2013-01-11 18:52:22 +00:00
|
|
|
cacheDir = filepath.Join(tempDir(), "camli-cache")
|
2013-01-08 18:44:59 +00:00
|
|
|
} else {
|
2013-08-25 17:25:30 +00:00
|
|
|
cacheDir = filepath.Join(blobPath, "cache")
|
2013-01-08 18:44:59 +00:00
|
|
|
}
|
2013-08-25 17:25:30 +00:00
|
|
|
if !noMkdir {
|
|
|
|
if err := os.MkdirAll(cacheDir, 0700); err != nil {
|
|
|
|
return nil, fmt.Errorf("Could not create blobs cache dir %s: %v", cacheDir, err)
|
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
2012-04-22 15:33:22 +00:00
|
|
|
published := []interface{}{}
|
2013-01-10 23:29:08 +00:00
|
|
|
if len(publish) > 0 {
|
|
|
|
if !runIndex {
|
|
|
|
return nil, fmt.Errorf("publishing requires an index")
|
|
|
|
}
|
2013-06-28 15:15:48 +00:00
|
|
|
published, err = addPublishedConfig(prefixes, publish, sourceRoot)
|
2012-04-22 15:33:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Could not generate config for published: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-10 23:29:08 +00:00
|
|
|
if runIndex {
|
2013-06-20 21:33:00 +00:00
|
|
|
addUIConfig(prefixes, "/ui/", published, sourceRoot)
|
2013-01-10 23:29:08 +00:00
|
|
|
}
|
2012-03-15 12:31:06 +00:00
|
|
|
|
|
|
|
if mysql != "" {
|
2012-12-22 00:13:36 +00:00
|
|
|
addMySQLConfig(prefixes, dbname, mysql)
|
2012-04-12 02:22:02 +00:00
|
|
|
}
|
2012-11-03 18:58:50 +00:00
|
|
|
if postgres != "" {
|
2012-12-22 00:13:36 +00:00
|
|
|
addPostgresConfig(prefixes, dbname, postgres)
|
2012-11-03 18:58:50 +00:00
|
|
|
}
|
2012-04-12 02:22:02 +00:00
|
|
|
if mongo != "" {
|
2012-12-22 00:13:36 +00:00
|
|
|
addMongoConfig(prefixes, dbname, mongo)
|
|
|
|
}
|
2013-01-10 23:29:08 +00:00
|
|
|
if sqliteFile != "" {
|
|
|
|
addSQLiteConfig(prefixes, sqliteFile)
|
|
|
|
}
|
2013-08-25 17:25:30 +00:00
|
|
|
if kvFile != "" {
|
|
|
|
addKVConfig(prefixes, kvFile)
|
|
|
|
}
|
2012-12-22 00:13:36 +00:00
|
|
|
if s3 != "" {
|
|
|
|
if err := addS3Config(prefixes, s3); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2012-04-12 02:22:02 +00:00
|
|
|
}
|
2013-08-11 15:07:18 +00:00
|
|
|
if googledrive != "" {
|
|
|
|
if err := addGoogleDriveConfig(prefixes, googledrive); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if googlecloudstorage != "" {
|
|
|
|
if err := addGoogleCloudStorageConfig(prefixes, googlecloudstorage); err != nil {
|
2013-07-06 20:29:17 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2012-04-12 02:22:02 +00:00
|
|
|
if indexerPath == "/index-mem/" {
|
2012-12-22 00:13:36 +00:00
|
|
|
addMemindexConfig(prefixes)
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
obj["prefixes"] = (map[string]interface{})(prefixes)
|
|
|
|
|
|
|
|
lowLevelConf = &Config{
|
2012-10-19 19:51:01 +00:00
|
|
|
Obj: obj,
|
|
|
|
configPath: conf.configPath,
|
2012-03-15 12:31:06 +00:00
|
|
|
}
|
|
|
|
return lowLevelConf, nil
|
|
|
|
}
|
2013-01-10 23:29:08 +00:00
|
|
|
|
|
|
|
func numSet(vv ...interface{}) (num int) {
|
|
|
|
for _, vi := range vv {
|
|
|
|
switch v := vi.(type) {
|
|
|
|
case string:
|
|
|
|
if v != "" {
|
|
|
|
num++
|
|
|
|
}
|
|
|
|
case bool:
|
|
|
|
if v {
|
|
|
|
num++
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
panic("unknown type")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func setMap(m map[string]interface{}, v ...interface{}) {
|
|
|
|
if len(v) < 2 {
|
|
|
|
panic("too few args")
|
|
|
|
}
|
|
|
|
if len(v) == 2 {
|
|
|
|
m[v[0].(string)] = v[1]
|
|
|
|
return
|
|
|
|
}
|
|
|
|
setMap(m[v[0].(string)].(map[string]interface{}), v[1:]...)
|
|
|
|
}
|