/* 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 ( "errors" "fmt" "os" "path/filepath" "strings" "camlistore.org/pkg/blobref" "camlistore.org/pkg/jsonconfig" "camlistore.org/pkg/jsonsign" ) // various parameters derived from the high-level user config // and needed to set up the low-level config. type configPrefixesParams struct { secretRing string keyId string indexerPath string blobPath string searchOwner *blobref.BlobRef } func addPublishedConfig(prefixes jsonconfig.Obj, published jsonconfig.Obj) ([]interface{}, error) { 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" rootPermanode, template, style := "", "", "" 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 case "template": template = val case "style": style = val default: return nil, fmt.Errorf("Unexpected key %q in config for %s", pk, k) } } if rootPermanode == "" || template == "" { return nil, fmt.Errorf("Missing key in configuration for %s, need \"rootPermanode\" and \"template\"", k) } ob := map[string]interface{}{} ob["handler"] = "publish" handlerArgs := map[string]interface{}{ "rootName": rootName, "blobRoot": "/bs-and-maybe-also-index/", "searchRoot": "/my-search/", "cache": "/cache/", "rootPermanode": []interface{}{"/sighelper/", rootPermanode}, } switch template { case "gallery": if style == "" { style = "pics.css" } handlerArgs["css"] = []interface{}{style} handlerArgs["js"] = []interface{}{"camli.js", "pics.js"} handlerArgs["scaledImage"] = "lrucache" case "blog": if style != "" { handlerArgs["css"] = []interface{}{style} } } ob["handlerArgs"] = handlerArgs prefixes[k] = ob pubPrefixes = append(pubPrefixes, k) } return pubPrefixes, nil } func addUIConfig(prefixes jsonconfig.Obj, uiPrefix string, published []interface{}) { ob := map[string]interface{}{} ob["handler"] = "ui" handlerArgs := map[string]interface{}{ "jsonSignRoot": "/sighelper/", "cache": "/cache/", "scaledImage": "lrucache", } if len(published) > 0 { handlerArgs["publishRoots"] = published } ob["handlerArgs"] = handlerArgs prefixes[uiPrefix] = ob } func addMongoConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) { 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]) } ob := map[string]interface{}{} ob["enabled"] = true ob["handler"] = "storage-mongodbindexer" ob["handlerArgs"] = map[string]interface{}{ "host": host, "user": fields[0], "password": fields[1], "database": dbname, "blobSource": "/bs/", } prefixes["/index-mongo/"] = ob } func addSQLConfig(rdbms string, prefixes jsonconfig.Obj, dbname string, dbinfo string) { fields := strings.Split(dbinfo, "@") if len(fields) != 2 { exitFailure("Malformed " + rdbms + " config string. Want: \"user@host:password\"") } user := fields[0] fields = strings.Split(fields[1], ":") if len(fields) != 2 { exitFailure("Malformed " + rdbms + " config string. Want: \"user@host:password\"") } ob := map[string]interface{}{} ob["enabled"] = true ob["handler"] = "storage-" + rdbms + "indexer" ob["handlerArgs"] = map[string]interface{}{ "host": fields[0], "user": user, "password": fields[1], "database": dbname, "blobSource": "/bs/", } prefixes["/index-"+rdbms+"/"] = ob } func addPostgresConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) { addSQLConfig("postgres", prefixes, dbname, dbinfo) } func addMySQLConfig(prefixes jsonconfig.Obj, dbname string, dbinfo string) { addSQLConfig("mysql", prefixes, dbname, dbinfo) } func addMemindexConfig(prefixes jsonconfig.Obj) { ob := map[string]interface{}{} ob["handler"] = "storage-memory-only-dev-indexer" ob["handlerArgs"] = map[string]interface{}{ "blobSource": "/bs/", } prefixes["/index-mem/"] = ob } func addS3Config(prefixes jsonconfig.Obj, s3 string) error { f := strings.SplitN(s3, ":", 3) if len(f) != 3 { 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] isPrimary := false if _, ok := prefixes["/bs/"]; !ok { isPrimary = true } s3Prefix := "" if isPrimary { s3Prefix = "/bs/" } else { s3Prefix = "/sto-s3/" } prefixes[s3Prefix] = map[string]interface{}{ "handler": "storage-s3", "handlerArgs": map[string]interface{}{ "aws_access_key": accessKey, "aws_secret_access_key": secret, "bucket": bucket, }, } 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{}{ "path": filepath.Join(os.TempDir(), "camli-cache"), }, } } else { prefixes["/sync-to-s3/"] = map[string]interface{}{ "handler": "sync", "handlerArgs": map[string]interface{}{ "from": "/bs/", "to": s3Prefix, }, } } return nil } func genLowLevelPrefixes(params *configPrefixesParams) (m jsonconfig.Obj) { m = make(jsonconfig.Obj) m["/"] = map[string]interface{}{ "handler": "root", "handlerArgs": map[string]interface{}{ "stealth": false, "blobRoot": "/bs-and-maybe-also-index/", "searchRoot": "/my-search/", }, } m["/setup/"] = map[string]interface{}{ "handler": "setup", } m["/sync/"] = map[string]interface{}{ "handler": "sync", "handlerArgs": map[string]interface{}{ "from": "/bs/", "to": params.indexerPath, }, } m["/sighelper/"] = map[string]interface{}{ "handler": "jsonsign", "handlerArgs": map[string]interface{}{ "secretRing": params.secretRing, "keyId": params.keyId, "publicKeyDest": "/bs-and-index/", }, } 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/", }, } if params.blobPath != "" { m["/bs/"] = map[string]interface{}{ "handler": "storage-filesystem", "handlerArgs": map[string]interface{}{ "path": params.blobPath, }, } m["/cache/"] = map[string]interface{}{ "handler": "storage-filesystem", "handlerArgs": map[string]interface{}{ "path": filepath.Join(params.blobPath, "/cache"), }, } } m["/my-search/"] = map[string]interface{}{ "handler": "search", "handlerArgs": map[string]interface{}{ "index": params.indexerPath, "owner": params.searchOwner.String(), }, } return } // genLowLevelConfig returns a low-level config from a high-level config. func genLowLevelConfig(conf *Config) (lowLevelConf *Config, err error) { var ( baseURL = conf.OptionalString("baseURL", "") listen = conf.OptionalString("listen", "") auth = conf.RequiredString("auth") keyId = conf.RequiredString("identity") secretRing = conf.RequiredString("identitySecretRing") blobPath = conf.OptionalString("blobPath", "") tlsOn = conf.OptionalBool("https", false) tlsCert = conf.OptionalString("HTTPSCertFile", "") tlsKey = conf.OptionalString("HTTPSKeyFile", "") dbname = conf.OptionalString("dbname", "") mysql = conf.OptionalString("mysql", "") postgres = conf.OptionalString("postgres", "") mongo = conf.OptionalString("mongo", "") _ = conf.OptionalList("replicateTo") s3 = conf.OptionalString("s3", "") publish = conf.OptionalObject("publish") ) if err := conf.Validate(); err != nil { return nil, err } obj := jsonconfig.Obj{} if tlsOn { 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 { obj["TLSCertFile"] = "config/selfgen_cert.pem" obj["TLSKeyFile"] = "config/selfgen_key.pem" } } if baseURL != "" { if strings.HasSuffix(baseURL, "/") { baseURL = baseURL[:len(baseURL)-1] } obj["baseURL"] = baseURL } if listen != "" { obj["listen"] = listen } obj["https"] = tlsOn obj["auth"] = auth if dbname == "" { username := os.Getenv("USER") if username == "" { return nil, fmt.Errorf("USER env var not set; needed to define dbname") } dbname = "camli" + username } var indexerPath string switch { case mongo != "" && mysql != "" || mongo != "" && postgres != "" || mysql != "" && postgres != "": return nil, errors.New("You can only pick one of the db engines (mongo, mysql, postgres).") case mysql != "": indexerPath = "/index-mysql/" case postgres != "": indexerPath = "/index-postgres/" case mongo != "": indexerPath = "/index-mongo/" default: indexerPath = "/index-mem/" } entity, err := jsonsign.EntityFromSecring(keyId, secretRing) if err != nil { return nil, err } armoredPublicKey, err := jsonsign.ArmoredPublicKey(entity) if err != nil { return nil, err } nolocaldisk := blobPath == "" if nolocaldisk && s3 == "" { return nil, errors.New("You need at least one of blobPath (for localdisk) or s3 configured for a blobserver.") } prefixesParams := &configPrefixesParams{ secretRing: secretRing, keyId: keyId, indexerPath: indexerPath, blobPath: blobPath, searchOwner: blobref.SHA1FromString(armoredPublicKey), } prefixes := genLowLevelPrefixes(prefixesParams) 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 cacheDir = filepath.Join(os.TempDir(), "camli-cache") } else { cacheDir = filepath.Join(blobPath, "/cache") } if err := os.MkdirAll(cacheDir, 0700); err != nil { return nil, fmt.Errorf("Could not create blobs cache dir %s: %v", cacheDir, err) } published := []interface{}{} if publish != nil { published, err = addPublishedConfig(prefixes, publish) if err != nil { return nil, fmt.Errorf("Could not generate config for published: %v", err) } } addUIConfig(prefixes, "/ui/", published) if mysql != "" { addMySQLConfig(prefixes, dbname, mysql) } if postgres != "" { addPostgresConfig(prefixes, dbname, postgres) } if mongo != "" { addMongoConfig(prefixes, dbname, mongo) } if s3 != "" { if err := addS3Config(prefixes, s3); err != nil { return nil, err } } if indexerPath == "/index-mem/" { addMemindexConfig(prefixes) } obj["prefixes"] = (map[string]interface{})(prefixes) lowLevelConf = &Config{ Obj: obj, configPath: conf.configPath, } return lowLevelConf, nil }