mirror of
https://github.com/vbatts/imgsrv.git
synced 2024-11-23 16:45:39 +00:00
adding initial commit
This commit is contained in:
commit
1d83869269
1 changed files with 580 additions and 0 deletions
580
imgsrv.go
Normal file
580
imgsrv.go
Normal file
|
@ -0,0 +1,580 @@
|
|||
package main
|
||||
|
||||
/*
|
||||
Fun Times. This is to serve as a single utility that
|
||||
* can be an image server, that stores into a mongo backend,
|
||||
OR
|
||||
* the client side tool that pushes/pulls images to the running server.
|
||||
*/
|
||||
|
||||
import (
|
||||
"log"
|
||||
"labix.org/v2/mgo"
|
||||
"labix.org/v2/mgo/bson"
|
||||
goconf "git.corp.redhat.com/srv/git/dev/towers/engineering/go/conf.git"
|
||||
"net/http"
|
||||
"flag"
|
||||
"fmt"
|
||||
"path/filepath"
|
||||
"os"
|
||||
"strings"
|
||||
"time"
|
||||
"errors"
|
||||
"mime"
|
||||
|
||||
"crypto/md5"
|
||||
"hash/adler32"
|
||||
"io"
|
||||
)
|
||||
|
||||
var (
|
||||
ConfigFile = fmt.Sprintf("%s/.imgsrv.conf", os.Getenv("HOME"))
|
||||
|
||||
DefaultRunAsServer = false
|
||||
RunAsServer = DefaultRunAsServer
|
||||
DefaultServerIP = "0.0.0.0"
|
||||
ServerIP = DefaultServerIP
|
||||
DefaultServerPort = "7777"
|
||||
ServerPort = DefaultServerPort
|
||||
DefaultMongoHost = "localhost"
|
||||
MongoHost = DefaultMongoHost
|
||||
DefaultMongoDB = "images"
|
||||
MongoDB = DefaultMongoDB
|
||||
MongoCollectionData = "data"
|
||||
MongoCollectionImages = "data"
|
||||
|
||||
MSession *mgo.Session
|
||||
MongoImagesDb *mgo.Database
|
||||
Mgfs *mgo.GridFS
|
||||
McImages *mgo.Collection
|
||||
|
||||
DefaultRemoteHost = ""
|
||||
RemoteHost = DefaultRemoteHost
|
||||
GetFile = ""
|
||||
PutFile = ""
|
||||
FileKeywords = ""
|
||||
)
|
||||
|
||||
type Image struct {
|
||||
Filename string // name for the given data
|
||||
Keywords []string // tags
|
||||
MimeType string // maybe preserve the type provided?
|
||||
Ip string // who uploaded it
|
||||
Date time.Time // when?
|
||||
//Name string // not sure that this is different than Filename
|
||||
}
|
||||
|
||||
/* Check whether this Image filename is on Mongo */
|
||||
func hasImageByFilename(filename string) (exists bool, err error) {
|
||||
query := McImages.Find(bson.M{"filename": filename})
|
||||
c, err := query.Count()
|
||||
if (err != nil) {
|
||||
return false, err
|
||||
}
|
||||
exists = (c > 0)
|
||||
return exists, nil
|
||||
}
|
||||
func hasImage(i Image) (exists bool, err error) {
|
||||
exists, err = hasImageByFilename(i.Filename)
|
||||
return
|
||||
}
|
||||
|
||||
func saveImage(i Image) (err error) {
|
||||
exists, err := hasImage(i)
|
||||
if (err != nil) {
|
||||
return
|
||||
}
|
||||
if (exists) {
|
||||
return errors.New("Image Filename Exists")
|
||||
}
|
||||
err = McImages.Insert(i)
|
||||
return
|
||||
}
|
||||
|
||||
func getImage(filename string) (i Image, err error) {
|
||||
err = McImages.Find(bson.M{"filename":filename}).One(&i)
|
||||
if (err != nil) {
|
||||
return i, err
|
||||
}
|
||||
return i, nil
|
||||
}
|
||||
|
||||
|
||||
/* return a <a href/> for a given filename
|
||||
and root is the relavtive base of the explicit link.
|
||||
*/
|
||||
func linkToImage(root string, filename string) (html string) {
|
||||
return fmt.Sprintf("<a href='%s/i/%s'>%s</a>",
|
||||
root,
|
||||
filename,
|
||||
filename)
|
||||
}
|
||||
|
||||
/* return the sections of the URI Path.
|
||||
This will disregard the leading '/'
|
||||
*/
|
||||
func chunkURI(uri string) (chunks []string) {
|
||||
var str string
|
||||
if (uri[0] == '/') {
|
||||
str = uri[1:]
|
||||
} else {
|
||||
str = uri
|
||||
}
|
||||
return strings.Split(str, "/")
|
||||
}
|
||||
|
||||
/* given an url.URL.RawQuery, get a dictionary in return */
|
||||
func parseRawQuery(qry string) (params map[string]string) {
|
||||
qryChunks := strings.Split(qry, "&")
|
||||
params = make(map[string]string, len(qryChunks))
|
||||
for _, chunk := range qryChunks {
|
||||
p := strings.SplitN(chunk, "=", 2)
|
||||
if (len(p) == 2) {
|
||||
params[p[0]] = p[1]
|
||||
}
|
||||
}
|
||||
return params
|
||||
}
|
||||
|
||||
/* kindof a common log type output */
|
||||
func LogRequest(r *http.Request, statusCode int) {
|
||||
var addr string
|
||||
var user_agent string
|
||||
|
||||
user_agent = ""
|
||||
addr = r.RemoteAddr
|
||||
|
||||
for k, v := range r.Header {
|
||||
if k == "User-Agent" {
|
||||
user_agent = strings.Join(v, " ")
|
||||
}
|
||||
if k == "X-Forwarded-For" {
|
||||
addr = strings.Join(v," ")
|
||||
}
|
||||
}
|
||||
|
||||
fmt.Printf("%s - - [%s] \"%s %s\" \"%s\" %d %d\n",
|
||||
addr,
|
||||
time.Now(),
|
||||
r.Method,
|
||||
r.URL.Path,
|
||||
user_agent,
|
||||
statusCode,
|
||||
r.ContentLength )
|
||||
}
|
||||
|
||||
/* Convinience method for getting md5 sum of a string */
|
||||
func getMd5FromString(blob string) (sum []byte) {
|
||||
h := md5.New()
|
||||
defer h.Reset()
|
||||
io.WriteString(h,blob)
|
||||
return h.Sum(nil)
|
||||
}
|
||||
|
||||
/* Convinience method for getting md5 sum of some bytes */
|
||||
func getMd5FromBytes(blob []byte) (sum []byte) {
|
||||
h := md5.New()
|
||||
defer h.Reset()
|
||||
h.Write(blob)
|
||||
return h.Sum(nil)
|
||||
}
|
||||
|
||||
/* get a small, decently unique hash */
|
||||
func getSmallHash() (small_hash string) {
|
||||
h := adler32.New()
|
||||
io.WriteString(h, fmt.Sprintf("%d", time.Now().Unix()))
|
||||
return fmt.Sprintf("%X", h.Sum(nil))
|
||||
}
|
||||
|
||||
// Show a page of most recent images, and tags, and uploaders ...
|
||||
func routeImagesGET(w http.ResponseWriter, r *http.Request) {
|
||||
uriChunks := chunkURI(r.URL.Path)
|
||||
if ( len(uriChunks) > 2 ) {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
|
||||
// preliminary checks, if they've passed an image name
|
||||
if (len(uriChunks) == 2 && len(uriChunks[1]) > 0) {
|
||||
exists, err := hasImageByFilename(uriChunks[1])
|
||||
if (err != nil) {
|
||||
LogRequest(r,503)
|
||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
||||
http.Error(w, "Service Unavailable", 503)
|
||||
return
|
||||
}
|
||||
if (!exists) {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
if (len(uriChunks) == 2 && len(uriChunks[1]) > 0) {
|
||||
ext := filepath.Ext(uriChunks[1])
|
||||
w.Header().Set("Content-Type", mime.TypeByExtension(ext))
|
||||
w.Header().Set("Cache-Control", "max-age=315360000")
|
||||
w.WriteHeader(http.StatusOK)
|
||||
|
||||
file, err := Mgfs.Open(uriChunks[1])
|
||||
if (err != nil) {
|
||||
log.Println(err)
|
||||
LogRequest(r,503)
|
||||
fmt.Fprintf(w,"Error fetching image: %s", uriChunks[1])
|
||||
http.Error(w, "Service Unavailable", 503)
|
||||
return
|
||||
}
|
||||
|
||||
io.Copy(w,file) // send the contents of the file in the body
|
||||
|
||||
} else {
|
||||
// TODO show a list of recent uploads? ...
|
||||
}
|
||||
LogRequest(r,200)
|
||||
}
|
||||
|
||||
// Create the file by the name in the path and/or parameter?
|
||||
// add keywords from the parameters
|
||||
// look for an image in the r.Body
|
||||
func routeImagesPOST(w http.ResponseWriter, r *http.Request) {
|
||||
uriChunks := chunkURI(r.URL.Path)
|
||||
if (len(uriChunks) > 2 &&
|
||||
((len(uriChunks) == 2 && len(uriChunks[1]) == 0) &&
|
||||
len(r.URL.RawQuery) == 0 )) {
|
||||
LogRequest(r,403)
|
||||
http.Error(w, "Not Acceptable", 403)
|
||||
return
|
||||
}
|
||||
|
||||
var i Image
|
||||
|
||||
i.Date = bson.Now()
|
||||
i.Ip = r.RemoteAddr
|
||||
|
||||
if (len(uriChunks) == 2 && len(uriChunks[1]) != 0) {
|
||||
i.Filename = uriChunks[1]
|
||||
}
|
||||
params := parseRawQuery(r.URL.RawQuery)
|
||||
var p_ext string
|
||||
for k,v := range params {
|
||||
switch {
|
||||
case (k == "filename"):
|
||||
i.Filename = v
|
||||
case (k == "ext"):
|
||||
if (v[0] != '.') {
|
||||
p_ext = fmt.Sprintf(".%s", v)
|
||||
} else {
|
||||
p_ext = v
|
||||
}
|
||||
case (k == "k" || k == "key" || k == "keyword"):
|
||||
i.Keywords = append(i.Keywords[:], v)
|
||||
case (k == "keys" || k == "keywords"):
|
||||
for _, key := range strings.Split(v, ",") {
|
||||
i.Keywords = append(i.Keywords[:], key)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (len(i.Filename) == 0) {
|
||||
str := getSmallHash()
|
||||
if (len(p_ext) == 0) {
|
||||
i.Filename = fmt.Sprintf("%s.jpg", str)
|
||||
} else {
|
||||
i.Filename = fmt.Sprintf("%s%s", str, p_ext)
|
||||
}
|
||||
}
|
||||
err := saveImage(i)
|
||||
if (err != nil && err.Error() != "Image Filename Exists") {
|
||||
file, err := Mgfs.Create(i.Filename)
|
||||
if (err != nil) {
|
||||
LogRequest(r,503)
|
||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
||||
http.Error(w, "Service Unavailable", 503)
|
||||
return
|
||||
}
|
||||
n, err := io.Copy(file, r.Body)
|
||||
if (err != nil) {
|
||||
LogRequest(r,503)
|
||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
||||
http.Error(w, "Service Unavailable", 503)
|
||||
return
|
||||
}
|
||||
if (n != r.ContentLength) {
|
||||
log.Printf("WARNING: [%s] content-length (%d), content written (%d)",
|
||||
i.Filename,
|
||||
r.ContentLength,
|
||||
n)
|
||||
}
|
||||
file.Close()
|
||||
} else {
|
||||
log.Printf("%s: %s", err, i.Filename)
|
||||
}
|
||||
io.WriteString(w,
|
||||
fmt.Sprintf("%s%s/i/%s\n", r.URL.Scheme, r.URL.Host, i.Filename))
|
||||
|
||||
LogRequest(r,200)
|
||||
}
|
||||
|
||||
func routeImagesPUT(w http.ResponseWriter, r *http.Request) {
|
||||
// update the file by the name in the path and/or parameter?
|
||||
// update/add keywords from the parameters
|
||||
// look for an image in the r.Body
|
||||
}
|
||||
|
||||
func routeImagesDELETE(w http.ResponseWriter, r *http.Request) {
|
||||
// delete the name in the path and/or parameter?
|
||||
}
|
||||
|
||||
func routeImages(w http.ResponseWriter, r *http.Request) {
|
||||
switch {
|
||||
case r.Method == "GET":
|
||||
routeImagesGET(w,r)
|
||||
case r.Method == "PUT":
|
||||
routeImagesPUT(w,r)
|
||||
case r.Method == "POST":
|
||||
routeImagesPOST(w,r)
|
||||
case r.Method == "DELETE":
|
||||
routeImagesDELETE(w,r)
|
||||
default:
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func routeRoot(w http.ResponseWriter, r *http.Request) {
|
||||
if (r.Method != "GET") {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
// Show a page of most recent images, and tags, and uploaders ...
|
||||
|
||||
w.Header().Set("Content-Type", "text/html")
|
||||
var i Image
|
||||
iter := McImages.Find(bson.M{"date": bson.M{"$gt": time.Now().Add(-time.Hour)}}).Limit(10).Iter()
|
||||
fmt.Fprintf(w, "<li>\n")
|
||||
for iter.Next(&i) {
|
||||
log.Println(i.Filename)
|
||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
||||
}
|
||||
fmt.Fprintf(w, "</li>\n")
|
||||
}
|
||||
|
||||
func routeAll(w http.ResponseWriter, r *http.Request) {
|
||||
if (r.Method != "GET") {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
// Show a page of all the images
|
||||
|
||||
var i Image
|
||||
iter := McImages.Find(bson.M{"date": bson.M{"$gt": time.Now().Add(-time.Hour)}}).Limit(10).Iter()
|
||||
fmt.Fprintf(w, "<li>\n")
|
||||
for iter.Next(&i) {
|
||||
log.Println(i.Filename)
|
||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
||||
}
|
||||
fmt.Fprintf(w, "</li>\n")
|
||||
}
|
||||
|
||||
/*
|
||||
GET /k/
|
||||
GET /k/:name
|
||||
GET /k/:name/r
|
||||
|
||||
Show a page of all the keyword tags, and then the images
|
||||
If /k/:name/r then show a random image by keyword name
|
||||
Otherwise 404
|
||||
*/
|
||||
func routeKeywords(w http.ResponseWriter, r *http.Request) {
|
||||
uriChunks := chunkURI(r.URL.Path)
|
||||
if (r.Method != "GET" ||
|
||||
len(uriChunks) > 3 ||
|
||||
(len(uriChunks) == 3 && uriChunks[2] != "r")) {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
|
||||
log.Printf("K: %s (%d)", uriChunks, len(uriChunks))
|
||||
params := parseRawQuery(r.URL.RawQuery)
|
||||
log.Printf("K: params: %s", params)
|
||||
|
||||
var i Image
|
||||
if (len(uriChunks) == 1) {
|
||||
// show a sorted list of tag name links
|
||||
iter := McImages.Find(bson.M{"keywords": uriChunks[1]}).Sort("$natural").Limit(100).Iter()
|
||||
fmt.Fprintf(w, "<li>\n")
|
||||
for iter.Next(&i) {
|
||||
log.Println(i.Filename)
|
||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
||||
}
|
||||
fmt.Fprintf(w, "</li>\n")
|
||||
} else if (len(uriChunks) == 2) {
|
||||
iter := McImages.Find(bson.M{"keywords": uriChunks[1]}).Limit(10).Iter()
|
||||
|
||||
fmt.Fprintf(w, "<li>\n")
|
||||
for iter.Next(&i) {
|
||||
log.Println(i.Filename)
|
||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
||||
}
|
||||
fmt.Fprintf(w, "</li>\n")
|
||||
} else if (uriChunks[2] == "r") {
|
||||
// TODO determine how to show a random image by keyword ...
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
// Show a page of all the uploader's IPs, and the images
|
||||
func routeIPs(w http.ResponseWriter, r *http.Request) {
|
||||
if (r.Method != "GET") {
|
||||
LogRequest(r,404)
|
||||
http.NotFound(w,r)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func initMongo() {
|
||||
MSession, err := mgo.Dial(MongoHost)
|
||||
if err != nil {
|
||||
log.Panic(err)
|
||||
}
|
||||
|
||||
MongoImagesDb = MSession.DB(MongoDB)
|
||||
Mgfs = MongoImagesDb.GridFS("fs")
|
||||
McImages = MongoImagesDb.C(MongoCollectionImages)
|
||||
}
|
||||
|
||||
/* Run as the image server */
|
||||
func runServer(ip, port string) {
|
||||
var addr = fmt.Sprintf("%s:%s", ip, port)
|
||||
|
||||
initMongo()
|
||||
defer MSession.Close()
|
||||
|
||||
http.HandleFunc("/", routeRoot)
|
||||
http.HandleFunc("/all", routeAll)
|
||||
http.HandleFunc("/i/", routeImages)
|
||||
http.HandleFunc("/k/", routeKeywords)
|
||||
http.HandleFunc("/ip/", routeIPs)
|
||||
|
||||
log.Printf("Serving on %s ...", addr)
|
||||
log.Fatal(http.ListenAndServe(addr, nil))
|
||||
}
|
||||
|
||||
/* http://golang.org/doc/effective_go.html#init */
|
||||
func init() {
|
||||
flag.StringVar(&ConfigFile,
|
||||
"config",
|
||||
ConfigFile,
|
||||
"Provide alternate configuration file")
|
||||
|
||||
/* Server-side */
|
||||
flag.BoolVar(&RunAsServer,
|
||||
"server",
|
||||
RunAsServer,
|
||||
"Run as an image server (defaults to client-side)")
|
||||
flag.StringVar(&ServerIP,
|
||||
"ip",
|
||||
ServerIP,
|
||||
"IP to bind to (if running as a server)('ip' in the config)")
|
||||
flag.StringVar(&ServerPort,
|
||||
"port",
|
||||
ServerPort,
|
||||
"Port to listen on (if running as a server)('port' in the config)")
|
||||
|
||||
/* MongoDB settings */
|
||||
flag.StringVar(&MongoHost,
|
||||
"mongo-host",
|
||||
MongoHost,
|
||||
"Mongo Host to connect to ('mongohost' in the config)")
|
||||
flag.StringVar(&MongoDB,
|
||||
"mongo-db",
|
||||
MongoDB,
|
||||
"Mongo db to connect to ('mongodb' in the config)")
|
||||
|
||||
/* Client-side */
|
||||
flag.StringVar(&RemoteHost,
|
||||
"remotehost",
|
||||
RemoteHost,
|
||||
"Remote host to get/put files on ('remotehost' in the config)")
|
||||
flag.StringVar(&GetFile,
|
||||
"get",
|
||||
GetFile,
|
||||
"Fetch file on remote server (needs -remotehost)")
|
||||
flag.StringVar(&PutFile,
|
||||
"put",
|
||||
PutFile,
|
||||
"Put file on remote server (needs -remotehost)")
|
||||
flag.StringVar(&FileKeywords,
|
||||
"keywords",
|
||||
FileKeywords,
|
||||
"Keywords to associate with file. (comma delimited) (needs -put)")
|
||||
|
||||
}
|
||||
|
||||
func loadConfiguration(filename string) (c *goconf.ConfigFile) {
|
||||
//log.Printf("Attempting to load config file: %s", filename)
|
||||
c, err := goconf.ReadConfigFile(filename)
|
||||
if (err != nil) {
|
||||
//log.Println(err)
|
||||
return goconf.NewConfigFile()
|
||||
}
|
||||
|
||||
cRunAsServer, _ := c.GetBool("", "server")
|
||||
cServerIp, _ := c.GetString("", "ip")
|
||||
cServerPort, _ := c.GetString("", "port")
|
||||
cMongoHost, _ := c.GetString("", "mongohost")
|
||||
cMongoDB, _ := c.GetString("", "mongodb")
|
||||
cRemoteHost, _ := c.GetString("", "remotehost")
|
||||
|
||||
// Only set variables from config file,
|
||||
// if they weren't passed as flags
|
||||
if (DefaultRunAsServer == RunAsServer && cRunAsServer) {
|
||||
RunAsServer = cRunAsServer
|
||||
}
|
||||
if (DefaultServerIP == ServerIP && len(cServerIp) > 0) {
|
||||
ServerIP = cServerIp
|
||||
}
|
||||
if (DefaultServerPort == ServerPort && len(cServerPort) > 0) {
|
||||
ServerPort = cServerPort
|
||||
}
|
||||
if (DefaultMongoHost == MongoHost && len(cMongoHost) > 0) {
|
||||
MongoHost = cMongoHost
|
||||
}
|
||||
if (DefaultMongoDB == MongoDB && len(cMongoDB) > 0) {
|
||||
MongoDB = cMongoDB
|
||||
}
|
||||
if (DefaultRemoteHost == RemoteHost && len(cRemoteHost) > 0) {
|
||||
RemoteHost = cRemoteHost
|
||||
}
|
||||
|
||||
return c
|
||||
}
|
||||
|
||||
func main() {
|
||||
flag.Parse()
|
||||
for _, arg := range flag.Args() {
|
||||
// What to do with these floating args ...
|
||||
log.Printf("%s", arg)
|
||||
}
|
||||
|
||||
// loads either default or flag specified config
|
||||
// to override variables
|
||||
loadConfiguration(ConfigFile)
|
||||
|
||||
if (RunAsServer) {
|
||||
runServer(ServerIP,ServerPort)
|
||||
}
|
||||
|
||||
log.Printf("%s", ServerIP)
|
||||
log.Printf("%d", len(GetFile))
|
||||
|
||||
//log.Printf("Good Morning!")
|
||||
//log.Printf("%x", getMd5FromString("Good Morning!"))
|
||||
}
|
||||
|
Loading…
Reference in a new issue