2011-04-16 22:44:22 +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-10-18 18:12:01 +00:00
|
|
|
package server
|
2011-04-16 22:44:22 +00:00
|
|
|
|
|
|
|
import (
|
2012-11-07 17:57:43 +00:00
|
|
|
"errors"
|
2011-04-16 22:44:22 +00:00
|
|
|
"fmt"
|
2011-05-26 04:56:48 +00:00
|
|
|
"log"
|
Update from r60 to [almost] Go 1.
A lot is still broken, but most stuff at least compiles now.
The directory tree has been rearranged now too. Go libraries are now
under "pkg". Fully qualified, they are e.g. "camlistore.org/pkg/jsonsign".
The go tool cannot yet fetch from arbitrary domains, but discussion is
happening now on which mechanism to use to allow that.
For now, put the camlistore root under $GOPATH/src. Typically $GOPATH
is $HOME, so Camlistore should be at $HOME/src/camlistore.org.
Then you can:
$ go build ./server/camlistored
... etc
The build.pl script is currently disabled. It'll be resurrected at
some point, but with a very different role (helping create a fake
GOPATH and running the go build command, if things are installed at
the wrong place, and/or running fileembed generators).
Many things are certainly broken.
Many things are disabled. (MySQL, all indexing, etc).
Many things need to be moved into
camlistore.org/third_party/{code.google.com,github.com} and updated
from their r60 to Go 1 versions, where applicable.
The GoMySQL stuff should be updated to use database/sql and the ziutek
library implementing database/sql/driver.
Help wanted.
Change-Id: If71217dc5c8f0e70dbe46e9504ca5131c6eeacde
2012-02-19 05:53:06 +00:00
|
|
|
"net/http"
|
2012-12-23 20:39:15 +00:00
|
|
|
"os"
|
2012-05-13 20:02:20 +00:00
|
|
|
"path"
|
2012-12-03 22:46:08 +00:00
|
|
|
"path/filepath"
|
2011-04-16 22:44:22 +00:00
|
|
|
"regexp"
|
2011-06-15 08:42:18 +00:00
|
|
|
"strconv"
|
2011-05-31 17:20:28 +00:00
|
|
|
"strings"
|
2012-12-12 02:29:58 +00:00
|
|
|
"time"
|
2011-04-16 22:44:22 +00:00
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
"camlistore.org/pkg/blob"
|
Update from r60 to [almost] Go 1.
A lot is still broken, but most stuff at least compiles now.
The directory tree has been rearranged now too. Go libraries are now
under "pkg". Fully qualified, they are e.g. "camlistore.org/pkg/jsonsign".
The go tool cannot yet fetch from arbitrary domains, but discussion is
happening now on which mechanism to use to allow that.
For now, put the camlistore root under $GOPATH/src. Typically $GOPATH
is $HOME, so Camlistore should be at $HOME/src/camlistore.org.
Then you can:
$ go build ./server/camlistored
... etc
The build.pl script is currently disabled. It'll be resurrected at
some point, but with a very different role (helping create a fake
GOPATH and running the go build command, if things are installed at
the wrong place, and/or running fileembed generators).
Many things are certainly broken.
Many things are disabled. (MySQL, all indexing, etc).
Many things need to be moved into
camlistore.org/third_party/{code.google.com,github.com} and updated
from their r60 to Go 1 versions, where applicable.
The GoMySQL stuff should be updated to use database/sql and the ziutek
library implementing database/sql/driver.
Help wanted.
Change-Id: If71217dc5c8f0e70dbe46e9504ca5131c6eeacde
2012-02-19 05:53:06 +00:00
|
|
|
"camlistore.org/pkg/blobserver"
|
2013-06-20 21:58:12 +00:00
|
|
|
"camlistore.org/pkg/fileembed"
|
Update from r60 to [almost] Go 1.
A lot is still broken, but most stuff at least compiles now.
The directory tree has been rearranged now too. Go libraries are now
under "pkg". Fully qualified, they are e.g. "camlistore.org/pkg/jsonsign".
The go tool cannot yet fetch from arbitrary domains, but discussion is
happening now on which mechanism to use to allow that.
For now, put the camlistore root under $GOPATH/src. Typically $GOPATH
is $HOME, so Camlistore should be at $HOME/src/camlistore.org.
Then you can:
$ go build ./server/camlistored
... etc
The build.pl script is currently disabled. It'll be resurrected at
some point, but with a very different role (helping create a fake
GOPATH and running the go build command, if things are installed at
the wrong place, and/or running fileembed generators).
Many things are certainly broken.
Many things are disabled. (MySQL, all indexing, etc).
Many things need to be moved into
camlistore.org/third_party/{code.google.com,github.com} and updated
from their r60 to Go 1 versions, where applicable.
The GoMySQL stuff should be updated to use database/sql and the ziutek
library implementing database/sql/driver.
Help wanted.
Change-Id: If71217dc5c8f0e70dbe46e9504ca5131c6eeacde
2012-02-19 05:53:06 +00:00
|
|
|
"camlistore.org/pkg/httputil"
|
|
|
|
"camlistore.org/pkg/jsonconfig"
|
2012-12-29 14:51:42 +00:00
|
|
|
"camlistore.org/pkg/jsonsign/signhandler"
|
2013-06-13 00:21:29 +00:00
|
|
|
"camlistore.org/pkg/misc/closure"
|
2013-07-29 08:02:40 +00:00
|
|
|
"camlistore.org/pkg/search"
|
2013-12-14 17:37:56 +00:00
|
|
|
"camlistore.org/pkg/sorted"
|
2014-01-06 01:49:59 +00:00
|
|
|
"camlistore.org/pkg/syncutil"
|
2013-06-12 15:49:35 +00:00
|
|
|
uistatic "camlistore.org/server/camlistored/ui"
|
2013-06-19 06:14:36 +00:00
|
|
|
closurestatic "camlistore.org/server/camlistored/ui/closure"
|
2013-12-31 23:25:19 +00:00
|
|
|
glitchstatic "camlistore.org/third_party/glitch"
|
2013-12-29 05:27:28 +00:00
|
|
|
reactstatic "camlistore.org/third_party/react"
|
2011-04-16 22:44:22 +00:00
|
|
|
)
|
|
|
|
|
2012-05-13 19:14:31 +00:00
|
|
|
var (
|
2013-09-29 23:51:09 +00:00
|
|
|
staticFilePattern = regexp.MustCompile(`^([a-zA-Z0-9\-\_]+\.(html|js|css|png|jpg|gif|svg))$`)
|
2013-06-11 09:50:56 +00:00
|
|
|
identOrDotPattern = regexp.MustCompile(`^[a-zA-Z\_]+(\.[a-zA-Z\_]+)*$`)
|
2012-05-13 19:14:31 +00:00
|
|
|
|
|
|
|
// Download URL suffix:
|
|
|
|
// $1: blobref (checked in download handler)
|
|
|
|
// $2: optional "/filename" to be sent as recommended download name,
|
|
|
|
// if sane looking
|
2013-06-12 17:10:24 +00:00
|
|
|
downloadPattern = regexp.MustCompile(`^download/([^/]+)(/.*)?$`)
|
|
|
|
|
2013-06-11 09:50:56 +00:00
|
|
|
thumbnailPattern = regexp.MustCompile(`^thumbnail/([^/]+)(/.*)?$`)
|
|
|
|
treePattern = regexp.MustCompile(`^tree/([^/]+)(/.*)?$`)
|
|
|
|
closurePattern = regexp.MustCompile(`^closure/(([^/]+)(/.*)?)$`)
|
2013-12-29 05:27:28 +00:00
|
|
|
reactPattern = regexp.MustCompile(`^react/(.+)$`)
|
2013-12-31 23:25:19 +00:00
|
|
|
glitchPattern = regexp.MustCompile(`^glitch/(.+)$`)
|
2013-12-17 02:11:52 +00:00
|
|
|
|
|
|
|
disableThumbCache, _ = strconv.ParseBool(os.Getenv("CAMLI_DISABLE_THUMB_CACHE"))
|
2012-05-13 19:14:31 +00:00
|
|
|
)
|
2011-04-16 22:44:22 +00:00
|
|
|
|
|
|
|
// UIHandler handles serving the UI and discovery JSON.
|
|
|
|
type UIHandler struct {
|
2012-11-07 17:57:43 +00:00
|
|
|
// JSONSignRoot is the optional path or full URL to the JSON
|
|
|
|
// Signing helper. Only used by the UI and thus necessary if
|
|
|
|
// UI is true.
|
|
|
|
// TODO(bradfitz): also move this up to the root handler,
|
|
|
|
// if we start having clients (like phones) that we want to upload
|
|
|
|
// but don't trust to have private signing keys?
|
2011-04-17 23:01:41 +00:00
|
|
|
JSONSignRoot string
|
2011-04-16 22:44:22 +00:00
|
|
|
|
2013-12-14 16:41:26 +00:00
|
|
|
publishRoots map[string]*PublishHandler
|
2011-06-19 20:09:43 +00:00
|
|
|
|
2012-11-07 17:57:43 +00:00
|
|
|
prefix string // of the UI handler itself
|
|
|
|
root *RootHandler
|
2012-12-29 14:51:42 +00:00
|
|
|
sigh *signhandler.Handler // or nil
|
2012-11-07 17:57:43 +00:00
|
|
|
|
2013-12-14 14:16:21 +00:00
|
|
|
// Cache optionally specifies a cache blob server, used for
|
|
|
|
// caching image thumbnails and other emphemeral data.
|
2012-11-07 17:57:43 +00:00
|
|
|
Cache blobserver.Storage // or nil
|
2013-12-14 14:16:21 +00:00
|
|
|
|
2014-01-06 01:49:59 +00:00
|
|
|
// Limit peak RAM used by concurrent image thumbnail calls.
|
|
|
|
resizeSem *syncutil.Sem
|
2013-12-14 17:37:56 +00:00
|
|
|
thumbMeta *thumbMeta // optional thumbnail key->blob.Ref cache
|
2011-07-05 19:10:35 +00:00
|
|
|
|
2013-06-19 06:14:36 +00:00
|
|
|
// sourceRoot optionally specifies the path to root of Camlistore's
|
2013-06-12 17:10:24 +00:00
|
|
|
// source. If empty, the UI files must be compiled in to the
|
2013-06-19 06:14:36 +00:00
|
|
|
// binary (with go run make.go). This comes from the "sourceRoot"
|
2013-06-12 17:10:24 +00:00
|
|
|
// ui handler config option.
|
2013-06-19 06:14:36 +00:00
|
|
|
sourceRoot string
|
2013-06-12 17:10:24 +00:00
|
|
|
|
2013-06-20 21:58:12 +00:00
|
|
|
uiDir string // if sourceRoot != "", this is sourceRoot+"/server/camlistored/ui"
|
|
|
|
|
2013-12-31 23:25:19 +00:00
|
|
|
closureHandler http.Handler
|
|
|
|
fileReactHandler http.Handler
|
|
|
|
fileGlitchHandler http.Handler
|
2011-04-16 22:44:22 +00:00
|
|
|
}
|
|
|
|
|
2011-05-26 14:34:39 +00:00
|
|
|
func init() {
|
2013-06-12 15:49:35 +00:00
|
|
|
blobserver.RegisterHandlerConstructor("ui", uiFromConfig)
|
2011-05-26 14:34:39 +00:00
|
|
|
}
|
|
|
|
|
2013-12-14 17:37:56 +00:00
|
|
|
// newKVOrNil wraps sorted.NewKeyValue and adds the ability
|
|
|
|
// to pass a nil conf to get a (nil, nil) response.
|
|
|
|
func newKVOrNil(conf jsonconfig.Obj) (sorted.KeyValue, error) {
|
|
|
|
if len(conf) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return sorted.NewKeyValue(conf)
|
|
|
|
}
|
|
|
|
|
2013-06-12 15:49:35 +00:00
|
|
|
func uiFromConfig(ld blobserver.Loader, conf jsonconfig.Obj) (h http.Handler, err error) {
|
2012-05-13 19:20:18 +00:00
|
|
|
ui := &UIHandler{
|
2012-05-13 20:02:20 +00:00
|
|
|
prefix: ld.MyPrefix(),
|
2012-05-13 19:20:18 +00:00
|
|
|
JSONSignRoot: conf.OptionalString("jsonSignRoot", ""),
|
2013-06-19 06:14:36 +00:00
|
|
|
sourceRoot: conf.OptionalString("sourceRoot", ""),
|
2014-01-06 01:49:59 +00:00
|
|
|
resizeSem: syncutil.NewSem(int64(conf.OptionalInt("maxResizeBytes",
|
|
|
|
defaultMaxResizeBytes))),
|
2012-05-13 19:20:18 +00:00
|
|
|
}
|
2011-06-19 20:09:43 +00:00
|
|
|
pubRoots := conf.OptionalList("publishRoots")
|
2011-06-04 16:58:50 +00:00
|
|
|
cachePrefix := conf.OptionalString("cache", "")
|
2013-12-14 17:37:56 +00:00
|
|
|
scaledImageConf := conf.OptionalObject("scaledImage")
|
2011-04-16 22:44:22 +00:00
|
|
|
if err = conf.Validate(); err != nil {
|
|
|
|
return
|
|
|
|
}
|
2011-05-09 21:20:19 +00:00
|
|
|
|
2012-12-24 00:58:26 +00:00
|
|
|
if ui.JSONSignRoot != "" {
|
|
|
|
h, _ := ld.GetHandler(ui.JSONSignRoot)
|
2012-12-29 14:51:42 +00:00
|
|
|
if sigh, ok := h.(*signhandler.Handler); ok {
|
2012-12-24 00:58:26 +00:00
|
|
|
ui.sigh = sigh
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-14 16:41:26 +00:00
|
|
|
ui.publishRoots = make(map[string]*PublishHandler)
|
2011-06-19 20:09:43 +00:00
|
|
|
for _, pubRoot := range pubRoots {
|
|
|
|
h, err := ld.GetHandler(pubRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("UI handler's publishRoots references invalid %q", pubRoot)
|
|
|
|
}
|
|
|
|
pubh, ok := h.(*PublishHandler)
|
|
|
|
if !ok {
|
Update from r60 to [almost] Go 1.
A lot is still broken, but most stuff at least compiles now.
The directory tree has been rearranged now too. Go libraries are now
under "pkg". Fully qualified, they are e.g. "camlistore.org/pkg/jsonsign".
The go tool cannot yet fetch from arbitrary domains, but discussion is
happening now on which mechanism to use to allow that.
For now, put the camlistore root under $GOPATH/src. Typically $GOPATH
is $HOME, so Camlistore should be at $HOME/src/camlistore.org.
Then you can:
$ go build ./server/camlistored
... etc
The build.pl script is currently disabled. It'll be resurrected at
some point, but with a very different role (helping create a fake
GOPATH and running the go build command, if things are installed at
the wrong place, and/or running fileembed generators).
Many things are certainly broken.
Many things are disabled. (MySQL, all indexing, etc).
Many things need to be moved into
camlistore.org/third_party/{code.google.com,github.com} and updated
from their r60 to Go 1 versions, where applicable.
The GoMySQL stuff should be updated to use database/sql and the ziutek
library implementing database/sql/driver.
Help wanted.
Change-Id: If71217dc5c8f0e70dbe46e9504ca5131c6eeacde
2012-02-19 05:53:06 +00:00
|
|
|
return nil, fmt.Errorf("UI handler's publishRoots references invalid %q; not a PublishHandler", pubRoot)
|
2011-06-19 20:09:43 +00:00
|
|
|
}
|
2013-12-14 16:41:26 +00:00
|
|
|
ui.publishRoots[pubRoot] = pubh
|
2011-06-19 20:09:43 +00:00
|
|
|
}
|
|
|
|
|
2011-05-09 21:20:19 +00:00
|
|
|
checkType := func(key string, htype string) {
|
|
|
|
v := conf.OptionalString(key, "")
|
|
|
|
if v == "" {
|
|
|
|
return
|
|
|
|
}
|
2011-05-26 14:34:39 +00:00
|
|
|
ct := ld.GetHandlerType(v)
|
2011-05-09 21:20:19 +00:00
|
|
|
if ct == "" {
|
|
|
|
err = fmt.Errorf("UI handler's %q references non-existant %q", key, v)
|
|
|
|
} else if ct != htype {
|
2013-01-11 03:29:10 +00:00
|
|
|
err = fmt.Errorf("UI handler's %q references %q of type %q; expected type %q", key, v, ct, htype)
|
2011-05-09 21:20:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
checkType("searchRoot", "search")
|
|
|
|
checkType("jsonSignRoot", "jsonsign")
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-12-14 17:37:56 +00:00
|
|
|
scaledImageKV, err := newKVOrNil(scaledImageConf)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("in UI handler's scaledImage: %v", err)
|
|
|
|
}
|
|
|
|
if scaledImageKV != nil && cachePrefix == "" {
|
|
|
|
return nil, fmt.Errorf("in UI handler, can't specify scaledImage without cache")
|
|
|
|
}
|
2011-06-04 16:58:50 +00:00
|
|
|
if cachePrefix != "" {
|
|
|
|
bs, err := ld.GetStorage(cachePrefix)
|
|
|
|
if err != nil {
|
2011-06-17 03:45:47 +00:00
|
|
|
return nil, fmt.Errorf("UI handler's cache of %q error: %v", cachePrefix, err)
|
2011-06-04 16:58:50 +00:00
|
|
|
}
|
|
|
|
ui.Cache = bs
|
2013-12-14 17:37:56 +00:00
|
|
|
ui.thumbMeta = newThumbMeta(scaledImageKV)
|
2011-06-04 16:58:50 +00:00
|
|
|
}
|
|
|
|
|
2013-06-20 21:58:12 +00:00
|
|
|
if ui.sourceRoot == "" {
|
|
|
|
ui.sourceRoot = os.Getenv("CAMLI_DEV_CAMLI_ROOT")
|
2013-08-16 16:25:14 +00:00
|
|
|
if uistatic.IsAppEngine {
|
|
|
|
if _, err = os.Stat(filepath.Join(uistatic.GaeSourceRoot,
|
|
|
|
filepath.FromSlash("server/camlistored/ui/index.html"))); err != nil {
|
|
|
|
hint := fmt.Sprintf("\"sourceRoot\" was not specified in the config,"+
|
|
|
|
" and the default sourceRoot dir %v does not exist or does not contain"+
|
2013-08-19 17:33:53 +00:00
|
|
|
" \"server/camlistored/ui/index.html\". devcam appengine can do that for you.",
|
2013-08-16 16:25:14 +00:00
|
|
|
uistatic.GaeSourceRoot)
|
|
|
|
log.Print(hint)
|
|
|
|
return nil, errors.New("No sourceRoot found; UI not available.")
|
|
|
|
}
|
|
|
|
log.Printf("Using the default \"%v\" as the sourceRoot for AppEngine", uistatic.GaeSourceRoot)
|
|
|
|
ui.sourceRoot = uistatic.GaeSourceRoot
|
|
|
|
}
|
2013-06-20 21:58:12 +00:00
|
|
|
}
|
|
|
|
if ui.sourceRoot != "" {
|
|
|
|
ui.uiDir = filepath.Join(ui.sourceRoot, filepath.FromSlash("server/camlistored/ui"))
|
|
|
|
// Ignore any fileembed files:
|
|
|
|
Files = &fileembed.Files{
|
|
|
|
DirFallback: filepath.Join(ui.sourceRoot, filepath.FromSlash("pkg/server")),
|
|
|
|
}
|
|
|
|
uistatic.Files = &fileembed.Files{
|
|
|
|
DirFallback: ui.uiDir,
|
2013-07-04 23:07:50 +00:00
|
|
|
Listable: true,
|
2013-08-16 16:25:14 +00:00
|
|
|
// In dev_appserver, allow edit-and-reload without
|
|
|
|
// restarting. In production, though, it's faster to just
|
|
|
|
// slurp it in.
|
|
|
|
SlurpToMemory: uistatic.IsProdAppEngine,
|
2013-06-20 21:58:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-19 06:14:36 +00:00
|
|
|
ui.closureHandler, err = ui.makeClosureHandler(ui.sourceRoot)
|
2013-06-12 17:10:24 +00:00
|
|
|
if err != nil {
|
2013-06-19 06:14:36 +00:00
|
|
|
return nil, fmt.Errorf(`Invalid "sourceRoot" value of %q: %v"`, ui.sourceRoot, err)
|
2012-12-12 17:04:05 +00:00
|
|
|
}
|
2011-07-05 19:10:35 +00:00
|
|
|
|
2013-12-29 05:27:28 +00:00
|
|
|
if ui.sourceRoot != "" {
|
2013-12-31 23:25:19 +00:00
|
|
|
ui.fileReactHandler, err = makeFileServer(ui.sourceRoot, filepath.Join("third_party", "react"), "react.js")
|
2013-12-29 05:27:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Could not make react handler: %s", err)
|
|
|
|
}
|
2013-12-31 23:25:19 +00:00
|
|
|
ui.fileGlitchHandler, err = makeFileServer(ui.sourceRoot, filepath.Join("third_party", "glitch"), "npc_piggy__x1_walk_png_1354829432.png")
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Could not make glitch handler: %s", err)
|
|
|
|
}
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 17:57:43 +00:00
|
|
|
rootPrefix, _, err := ld.FindHandlerByType("root")
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("No root handler configured, which is necessary for the ui handler")
|
|
|
|
}
|
|
|
|
if h, err := ld.GetHandler(rootPrefix); err == nil {
|
|
|
|
ui.root = h.(*RootHandler)
|
|
|
|
ui.root.registerUIHandler(ui)
|
|
|
|
} else {
|
|
|
|
return nil, errors.New("failed to find the 'root' handler")
|
|
|
|
}
|
|
|
|
|
2011-04-16 22:44:22 +00:00
|
|
|
return ui, nil
|
|
|
|
}
|
|
|
|
|
2013-06-28 15:15:48 +00:00
|
|
|
func (ui *UIHandler) makeClosureHandler(root string) (http.Handler, error) {
|
|
|
|
return makeClosureHandler(root, "ui")
|
|
|
|
}
|
|
|
|
|
2013-06-12 17:10:24 +00:00
|
|
|
// makeClosureHandler returns a handler to serve Closure files.
|
|
|
|
// root is either:
|
2013-06-19 06:14:36 +00:00
|
|
|
// 1) empty: use the Closure files compiled in to the binary (if
|
|
|
|
// available), else redirect to the Internet.
|
|
|
|
// 2) a URL prefix: base of Camlistore to get Closure to redirect to
|
|
|
|
// 3) a path on disk to the root of camlistore's source (which
|
|
|
|
// contains the necessary subset of Closure files)
|
2013-06-28 15:15:48 +00:00
|
|
|
func makeClosureHandler(root, handlerName string) (http.Handler, error) {
|
2013-08-16 16:25:14 +00:00
|
|
|
// devcam server environment variable takes precedence:
|
2013-06-19 06:14:36 +00:00
|
|
|
if d := os.Getenv("CAMLI_DEV_CLOSURE_DIR"); d != "" {
|
2013-07-29 13:59:53 +00:00
|
|
|
log.Printf("%v: serving Closure from devcam server's $CAMLI_DEV_CLOSURE_DIR: %v", handlerName, d)
|
2013-06-19 06:14:36 +00:00
|
|
|
return http.FileServer(http.Dir(d)), nil
|
|
|
|
}
|
2013-06-12 17:10:24 +00:00
|
|
|
if root == "" {
|
2013-06-19 06:14:36 +00:00
|
|
|
fs, err := closurestatic.FileSystem()
|
|
|
|
if err == os.ErrNotExist {
|
2013-06-28 15:15:48 +00:00
|
|
|
log.Printf("%v: no configured setting or embedded resources; serving Closure via %v", handlerName, closureBaseURL)
|
2013-06-19 06:14:36 +00:00
|
|
|
return closureBaseURL, nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error loading embedded Closure zip file: %v", err)
|
|
|
|
}
|
2013-06-28 15:15:48 +00:00
|
|
|
log.Printf("%v: serving Closure from embedded resources", handlerName)
|
2013-06-19 06:14:36 +00:00
|
|
|
return http.FileServer(fs), nil
|
2013-06-12 17:10:24 +00:00
|
|
|
}
|
|
|
|
if strings.HasPrefix(root, "http") {
|
2013-06-28 15:15:48 +00:00
|
|
|
log.Printf("%v: serving Closure using redirects to %v", handlerName, root)
|
2013-06-12 17:10:24 +00:00
|
|
|
return closureRedirector(root), nil
|
|
|
|
}
|
2013-12-29 05:27:28 +00:00
|
|
|
|
|
|
|
path := filepath.Join("third_party", "closure", "lib", "closure")
|
2013-12-31 23:25:19 +00:00
|
|
|
return makeFileServer(root, path, filepath.Join("goog", "base.js"))
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func makeFileServer(sourceRoot string, pathToServe string, expectedContentPath string) (http.Handler, error) {
|
|
|
|
fi, err := os.Stat(sourceRoot)
|
2013-06-12 17:10:24 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if !fi.IsDir() {
|
|
|
|
return nil, errors.New("not a directory")
|
|
|
|
}
|
2013-12-29 05:27:28 +00:00
|
|
|
dirToServe := filepath.Join(sourceRoot, pathToServe)
|
|
|
|
_, err = os.Stat(filepath.Join(dirToServe, expectedContentPath))
|
2013-06-12 17:10:24 +00:00
|
|
|
if err != nil {
|
2013-12-29 05:27:28 +00:00
|
|
|
return nil, fmt.Errorf("directory doesn't contain %s; wrong directory?", expectedContentPath)
|
2013-06-12 17:10:24 +00:00
|
|
|
}
|
2013-12-29 05:27:28 +00:00
|
|
|
return http.FileServer(http.Dir(dirToServe)), nil
|
2013-06-12 17:10:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const closureBaseURL closureRedirector = "https://closure-library.googlecode.com/git"
|
2013-06-11 09:50:56 +00:00
|
|
|
|
2013-06-12 09:05:01 +00:00
|
|
|
// closureRedirector is a hack to redirect requests for Closure's million *.js files
|
|
|
|
// to https://closure-library.googlecode.com/git.
|
|
|
|
// TODO: this doesn't work when offline. We need to run genjsdeps over all of the Camlistore
|
|
|
|
// UI to figure out which Closure *.js files to fileembed and generate zembed. Then this
|
|
|
|
// type can be deleted.
|
2013-06-12 17:10:24 +00:00
|
|
|
type closureRedirector string
|
2013-06-11 09:50:56 +00:00
|
|
|
|
2013-06-12 17:10:24 +00:00
|
|
|
func (base closureRedirector) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
newURL := string(base) + "/" + path.Clean(httputil.PathSuffix(req))
|
2013-06-11 09:50:56 +00:00
|
|
|
http.Redirect(rw, req, newURL, http.StatusTemporaryRedirect)
|
|
|
|
}
|
|
|
|
|
2011-05-26 04:56:48 +00:00
|
|
|
func camliMode(req *http.Request) string {
|
2012-05-13 19:13:58 +00:00
|
|
|
return req.URL.Query().Get("camli.mode")
|
2011-05-26 04:56:48 +00:00
|
|
|
}
|
|
|
|
|
2011-04-16 23:18:31 +00:00
|
|
|
func wantsDiscovery(req *http.Request) bool {
|
2013-09-15 19:12:26 +00:00
|
|
|
return httputil.IsGet(req) &&
|
2011-04-16 23:18:31 +00:00
|
|
|
(req.Header.Get("Accept") == "text/x-camli-configuration" ||
|
2011-05-29 17:50:17 +00:00
|
|
|
camliMode(req) == "config")
|
2011-05-26 04:56:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func wantsUploadHelper(req *http.Request) bool {
|
|
|
|
return req.Method == "POST" && camliMode(req) == "uploadhelper"
|
2011-04-16 23:18:31 +00:00
|
|
|
}
|
|
|
|
|
2011-05-30 04:39:23 +00:00
|
|
|
func wantsPermanode(req *http.Request) bool {
|
2014-01-02 05:56:03 +00:00
|
|
|
if httputil.IsGet(req) && blob.ValidRefString(req.FormValue("p")) {
|
|
|
|
// The new UI is handled by index.html.
|
|
|
|
if req.FormValue("newui") != "1" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
2011-05-30 04:39:23 +00:00
|
|
|
}
|
|
|
|
|
2011-05-30 23:41:56 +00:00
|
|
|
func wantsBlobInfo(req *http.Request) bool {
|
2013-09-15 19:12:26 +00:00
|
|
|
return httputil.IsGet(req) && blob.ValidRefString(req.FormValue("b"))
|
2011-05-30 23:41:56 +00:00
|
|
|
}
|
|
|
|
|
2011-07-17 15:50:55 +00:00
|
|
|
func wantsFileTreePage(req *http.Request) bool {
|
2013-09-15 19:12:26 +00:00
|
|
|
return httputil.IsGet(req) && blob.ValidRefString(req.FormValue("d"))
|
2011-07-17 15:50:55 +00:00
|
|
|
}
|
|
|
|
|
2013-12-31 23:25:19 +00:00
|
|
|
func getSuffixMatches(req *http.Request, pattern *regexp.Regexp) bool {
|
2013-09-15 19:12:26 +00:00
|
|
|
if httputil.IsGet(req) {
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2013-12-31 23:25:19 +00:00
|
|
|
return pattern.MatchString(suffix)
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2011-04-16 22:44:22 +00:00
|
|
|
func (ui *UIHandler) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2011-05-30 02:05:21 +00:00
|
|
|
|
2011-04-16 23:18:31 +00:00
|
|
|
rw.Header().Set("Vary", "Accept")
|
|
|
|
switch {
|
|
|
|
case wantsDiscovery(req):
|
2012-11-07 17:57:43 +00:00
|
|
|
ui.root.serveDiscovery(rw, req)
|
2011-05-26 04:56:48 +00:00
|
|
|
case wantsUploadHelper(req):
|
|
|
|
ui.serveUploadHelper(rw, req)
|
2011-05-31 17:20:28 +00:00
|
|
|
case strings.HasPrefix(suffix, "download/"):
|
|
|
|
ui.serveDownload(rw, req)
|
2011-06-15 08:42:18 +00:00
|
|
|
case strings.HasPrefix(suffix, "thumbnail/"):
|
|
|
|
ui.serveThumbnail(rw, req)
|
2011-07-17 15:50:55 +00:00
|
|
|
case strings.HasPrefix(suffix, "tree/"):
|
|
|
|
ui.serveFileTree(rw, req)
|
2013-12-31 23:25:19 +00:00
|
|
|
case getSuffixMatches(req, closurePattern):
|
2013-06-12 09:05:01 +00:00
|
|
|
ui.serveClosure(rw, req)
|
2013-12-31 23:25:19 +00:00
|
|
|
case getSuffixMatches(req, reactPattern):
|
|
|
|
ui.serveFromDiskOrStatic(rw, req, reactPattern, ui.fileReactHandler, reactstatic.Files)
|
|
|
|
case getSuffixMatches(req, glitchPattern):
|
|
|
|
ui.serveFromDiskOrStatic(rw, req, glitchPattern, ui.fileGlitchHandler, glitchstatic.Files)
|
2011-04-16 23:18:31 +00:00
|
|
|
default:
|
2011-04-17 23:01:41 +00:00
|
|
|
file := ""
|
2011-05-30 02:05:21 +00:00
|
|
|
if m := staticFilePattern.FindStringSubmatch(suffix); m != nil {
|
2011-04-17 23:01:41 +00:00
|
|
|
file = m[1]
|
2011-05-30 02:05:21 +00:00
|
|
|
} else {
|
2011-05-30 23:41:56 +00:00
|
|
|
switch {
|
|
|
|
case wantsPermanode(req):
|
|
|
|
file = "permanode.html"
|
|
|
|
case wantsBlobInfo(req):
|
|
|
|
file = "blobinfo.html"
|
2011-07-17 15:50:55 +00:00
|
|
|
case wantsFileTreePage(req):
|
|
|
|
file = "filetree.html"
|
2013-06-12 09:17:30 +00:00
|
|
|
case req.URL.Path == httputil.PathBase(req):
|
2012-12-12 02:29:58 +00:00
|
|
|
file = "index.html"
|
2011-05-30 23:41:56 +00:00
|
|
|
default:
|
2013-06-11 09:50:56 +00:00
|
|
|
http.Error(rw, "Illegal URL.", http.StatusNotFound)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2013-07-04 23:07:50 +00:00
|
|
|
if file == "deps.js" {
|
2013-06-20 21:58:12 +00:00
|
|
|
serveDepsJS(rw, req, ui.uiDir)
|
|
|
|
return
|
2011-04-16 23:18:31 +00:00
|
|
|
}
|
2013-06-20 21:58:12 +00:00
|
|
|
serveStaticFile(rw, req, uistatic.Files, file)
|
2011-04-16 22:44:22 +00:00
|
|
|
}
|
2011-04-16 23:18:31 +00:00
|
|
|
}
|
|
|
|
|
2012-12-12 02:29:58 +00:00
|
|
|
func serveStaticFile(rw http.ResponseWriter, req *http.Request, root http.FileSystem, file string) {
|
|
|
|
f, err := root.Open("/" + file)
|
|
|
|
if err != nil {
|
|
|
|
http.NotFound(rw, req)
|
2013-12-29 05:27:28 +00:00
|
|
|
log.Printf("Failed to open file %q from embedded resources: %v", file, err)
|
2012-12-12 02:29:58 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
var modTime time.Time
|
|
|
|
if fi, err := f.Stat(); err == nil {
|
|
|
|
modTime = fi.ModTime()
|
|
|
|
}
|
2013-12-28 03:12:37 +00:00
|
|
|
// TODO(wathiede): should pkg/magic be leveraged here somehow? It has a
|
|
|
|
// slightly different purpose.
|
|
|
|
if strings.HasSuffix(file, ".svg") {
|
|
|
|
rw.Header().Set("Content-Type", "image/svg+xml")
|
|
|
|
}
|
2012-12-12 02:29:58 +00:00
|
|
|
http.ServeContent(rw, req, file, modTime, f)
|
|
|
|
}
|
|
|
|
|
2012-11-07 17:57:43 +00:00
|
|
|
func (ui *UIHandler) populateDiscoveryMap(m map[string]interface{}) {
|
2011-06-19 21:12:56 +00:00
|
|
|
pubRoots := map[string]interface{}{}
|
2013-12-14 16:41:26 +00:00
|
|
|
for key, pubh := range ui.publishRoots {
|
2011-07-01 21:31:41 +00:00
|
|
|
m := map[string]interface{}{
|
2011-07-02 16:09:50 +00:00
|
|
|
"name": pubh.RootName,
|
2011-06-24 17:34:10 +00:00
|
|
|
"prefix": []string{key},
|
2011-06-19 21:12:56 +00:00
|
|
|
// TODO: include gpg key id
|
|
|
|
}
|
2013-01-11 18:35:19 +00:00
|
|
|
if sh, ok := ui.root.SearchHandler(); ok {
|
|
|
|
pn, err := sh.Index().PermanodeOfSignerAttrValue(sh.Owner(), "camliRoot", pubh.RootName)
|
2011-07-01 21:31:41 +00:00
|
|
|
if err == nil {
|
|
|
|
m["currentPermanode"] = pn.String()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pubRoots[pubh.RootName] = m
|
2011-06-19 21:12:56 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 17:57:43 +00:00
|
|
|
uiDisco := map[string]interface{}{
|
2011-08-25 15:14:47 +00:00
|
|
|
"jsonSignRoot": ui.JSONSignRoot,
|
2012-05-13 20:02:20 +00:00
|
|
|
"uploadHelper": ui.prefix + "?camli.mode=uploadhelper", // hack; remove with better javascript
|
|
|
|
"downloadHelper": path.Join(ui.prefix, "download") + "/",
|
|
|
|
"directoryHelper": path.Join(ui.prefix, "tree") + "/",
|
2011-08-25 15:14:47 +00:00
|
|
|
"publishRoots": pubRoots,
|
2012-11-07 17:57:43 +00:00
|
|
|
}
|
2012-12-24 00:58:26 +00:00
|
|
|
if ui.sigh != nil {
|
2012-12-29 14:51:42 +00:00
|
|
|
uiDisco["signing"] = ui.sigh.DiscoveryMap(ui.JSONSignRoot)
|
2012-12-24 00:58:26 +00:00
|
|
|
}
|
2012-11-07 17:57:43 +00:00
|
|
|
for k, v := range uiDisco {
|
|
|
|
if _, ok := m[k]; ok {
|
|
|
|
log.Fatalf("Duplicate discovery key %q", k)
|
|
|
|
}
|
|
|
|
m[k] = v
|
|
|
|
}
|
2011-04-16 22:44:22 +00:00
|
|
|
}
|
2011-05-26 04:56:48 +00:00
|
|
|
|
2011-05-31 17:20:28 +00:00
|
|
|
func (ui *UIHandler) serveDownload(rw http.ResponseWriter, req *http.Request) {
|
2012-11-07 17:57:43 +00:00
|
|
|
if ui.root.Storage == nil {
|
2011-05-31 17:20:28 +00:00
|
|
|
http.Error(rw, "No BlobRoot configured", 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2011-05-31 17:20:28 +00:00
|
|
|
m := downloadPattern.FindStringSubmatch(suffix)
|
|
|
|
if m == nil {
|
|
|
|
httputil.ErrorRouting(rw, req)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
fbr, ok := blob.Parse(m[1])
|
|
|
|
if !ok {
|
2011-05-31 17:20:28 +00:00
|
|
|
http.Error(rw, "Invalid blobref", 400)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2011-07-03 17:53:02 +00:00
|
|
|
dh := &DownloadHandler{
|
2012-11-07 17:57:43 +00:00
|
|
|
Fetcher: ui.root.Storage,
|
2011-07-03 17:53:02 +00:00
|
|
|
Cache: ui.Cache,
|
2011-06-06 16:07:07 +00:00
|
|
|
}
|
2011-07-03 17:53:02 +00:00
|
|
|
dh.ServeHTTP(rw, req, fbr)
|
2011-05-31 17:20:28 +00:00
|
|
|
}
|
2011-06-15 08:42:18 +00:00
|
|
|
|
|
|
|
func (ui *UIHandler) serveThumbnail(rw http.ResponseWriter, req *http.Request) {
|
2012-11-07 17:57:43 +00:00
|
|
|
if ui.root.Storage == nil {
|
2011-06-15 08:42:18 +00:00
|
|
|
http.Error(rw, "No BlobRoot configured", 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2011-06-15 08:42:18 +00:00
|
|
|
m := thumbnailPattern.FindStringSubmatch(suffix)
|
|
|
|
if m == nil {
|
|
|
|
httputil.ErrorRouting(rw, req)
|
|
|
|
return
|
|
|
|
}
|
2011-06-19 20:09:43 +00:00
|
|
|
|
2011-06-15 08:42:18 +00:00
|
|
|
query := req.URL.Query()
|
2013-07-29 08:02:40 +00:00
|
|
|
width, _ := strconv.Atoi(query.Get("mw"))
|
|
|
|
height, _ := strconv.Atoi(query.Get("mh"))
|
2013-08-04 02:54:30 +00:00
|
|
|
blobref, ok := blob.Parse(m[1])
|
|
|
|
if !ok {
|
2011-06-15 08:42:18 +00:00
|
|
|
http.Error(rw, "Invalid blobref", 400)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-07-29 08:02:40 +00:00
|
|
|
if width == 0 {
|
|
|
|
width = search.MaxImageSize
|
|
|
|
}
|
|
|
|
if height == 0 {
|
|
|
|
height = search.MaxImageSize
|
|
|
|
}
|
|
|
|
|
2011-07-03 20:33:41 +00:00
|
|
|
th := &ImageHandler{
|
2012-11-07 17:57:43 +00:00
|
|
|
Fetcher: ui.root.Storage,
|
2011-07-03 20:33:41 +00:00
|
|
|
Cache: ui.Cache,
|
|
|
|
MaxWidth: width,
|
|
|
|
MaxHeight: height,
|
2013-12-14 17:37:56 +00:00
|
|
|
thumbMeta: ui.thumbMeta,
|
2014-01-06 01:49:59 +00:00
|
|
|
resizeSem: ui.resizeSem,
|
2011-06-15 08:42:18 +00:00
|
|
|
}
|
2011-07-03 20:33:41 +00:00
|
|
|
th.ServeHTTP(rw, req, blobref)
|
2011-06-15 08:42:18 +00:00
|
|
|
}
|
2011-07-17 15:50:55 +00:00
|
|
|
|
|
|
|
func (ui *UIHandler) serveFileTree(rw http.ResponseWriter, req *http.Request) {
|
2012-11-07 17:57:43 +00:00
|
|
|
if ui.root.Storage == nil {
|
2011-07-17 15:50:55 +00:00
|
|
|
http.Error(rw, "No BlobRoot configured", 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2011-07-17 15:50:55 +00:00
|
|
|
m := treePattern.FindStringSubmatch(suffix)
|
|
|
|
if m == nil {
|
|
|
|
httputil.ErrorRouting(rw, req)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-08-04 02:54:30 +00:00
|
|
|
blobref, ok := blob.Parse(m[1])
|
|
|
|
if !ok {
|
2011-07-17 15:50:55 +00:00
|
|
|
http.Error(rw, "Invalid blobref", 400)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fth := &FileTreeHandler{
|
2012-11-07 17:57:43 +00:00
|
|
|
Fetcher: ui.root.Storage,
|
2011-08-25 15:14:47 +00:00
|
|
|
file: blobref,
|
2011-07-17 15:50:55 +00:00
|
|
|
}
|
|
|
|
fth.ServeHTTP(rw, req)
|
|
|
|
}
|
2012-12-13 18:20:45 +00:00
|
|
|
|
2013-06-12 09:05:01 +00:00
|
|
|
func (ui *UIHandler) serveClosure(rw http.ResponseWriter, req *http.Request) {
|
2013-06-12 09:17:30 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2012-12-23 19:01:19 +00:00
|
|
|
if ui.closureHandler == nil {
|
2013-06-11 09:50:56 +00:00
|
|
|
log.Printf("%v not served: closure handler is nil", suffix)
|
2012-12-13 18:20:45 +00:00
|
|
|
http.NotFound(rw, req)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
m := closurePattern.FindStringSubmatch(suffix)
|
2013-06-11 09:50:56 +00:00
|
|
|
if m == nil {
|
|
|
|
httputil.ErrorRouting(rw, req)
|
2012-12-23 20:39:15 +00:00
|
|
|
return
|
|
|
|
}
|
2013-06-11 09:50:56 +00:00
|
|
|
req.URL.Path = "/" + m[1]
|
|
|
|
ui.closureHandler.ServeHTTP(rw, req)
|
2012-12-13 18:20:45 +00:00
|
|
|
}
|
2012-12-23 20:39:15 +00:00
|
|
|
|
2013-12-31 23:25:19 +00:00
|
|
|
// serveFromDiskOrStatic matches rx against req's path and serves the match either from disk (if non-nil) or from static (embedded in the binary).
|
|
|
|
func (ui *UIHandler) serveFromDiskOrStatic(rw http.ResponseWriter, req *http.Request, rx *regexp.Regexp, disk http.Handler, static *fileembed.Files) {
|
2013-12-29 05:27:28 +00:00
|
|
|
suffix := httputil.PathSuffix(req)
|
2013-12-31 23:25:19 +00:00
|
|
|
m := rx.FindStringSubmatch(suffix)
|
2013-12-29 05:27:28 +00:00
|
|
|
if m == nil {
|
2013-12-31 23:25:19 +00:00
|
|
|
panic("Caller should verify that rx matches")
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
|
|
|
file := m[1]
|
2013-12-31 23:25:19 +00:00
|
|
|
if disk != nil {
|
2013-12-29 05:27:28 +00:00
|
|
|
req.URL.Path = "/" + file
|
2013-12-31 23:25:19 +00:00
|
|
|
disk.ServeHTTP(rw, req)
|
2013-12-29 05:27:28 +00:00
|
|
|
} else {
|
2013-12-31 23:25:19 +00:00
|
|
|
serveStaticFile(rw, req, static, file)
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
2013-12-31 23:25:19 +00:00
|
|
|
|
2013-12-29 05:27:28 +00:00
|
|
|
}
|
|
|
|
|
2012-12-23 20:39:15 +00:00
|
|
|
// serveDepsJS serves an auto-generated Closure deps.js file.
|
2013-06-20 21:58:12 +00:00
|
|
|
func serveDepsJS(rw http.ResponseWriter, req *http.Request, dir string) {
|
2013-07-04 23:07:50 +00:00
|
|
|
var root http.FileSystem
|
|
|
|
if dir == "" {
|
|
|
|
root = uistatic.Files
|
|
|
|
} else {
|
|
|
|
root = http.Dir(dir)
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := closure.GenDeps(root)
|
2012-12-23 20:39:15 +00:00
|
|
|
if err != nil {
|
2013-06-13 00:21:29 +00:00
|
|
|
log.Print(err)
|
2012-12-23 20:39:15 +00:00
|
|
|
http.Error(rw, "Server error", 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
rw.Header().Set("Content-Type", "text/javascript; charset=utf-8")
|
2013-06-20 21:58:12 +00:00
|
|
|
rw.Write([]byte("// auto-generated from camlistored\n"))
|
2013-06-13 00:21:29 +00:00
|
|
|
rw.Write(b)
|
2012-12-23 20:39:15 +00:00
|
|
|
}
|