mirror of
https://github.com/vbatts/imgsrv.git
synced 2024-11-27 10:35:41 +00:00
refactored for Mongo::GridFS. It almost works
This commit is contained in:
parent
9fdcddcfb6
commit
a2f39ef4d1
1 changed files with 87 additions and 93 deletions
180
imgsrv.go
180
imgsrv.go
|
@ -19,7 +19,7 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
"errors"
|
//"errors"
|
||||||
"mime"
|
"mime"
|
||||||
|
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
|
@ -43,10 +43,9 @@ var (
|
||||||
MongoCollectionData = "data"
|
MongoCollectionData = "data"
|
||||||
MongoCollectionImages = "data"
|
MongoCollectionImages = "data"
|
||||||
|
|
||||||
MSession *mgo.Session
|
mongo_session *mgo.Session
|
||||||
MongoImagesDb *mgo.Database
|
images_db *mgo.Database
|
||||||
Mgfs *mgo.GridFS
|
gfs *mgo.GridFS
|
||||||
McImages *mgo.Collection
|
|
||||||
|
|
||||||
DefaultRemoteHost = ""
|
DefaultRemoteHost = ""
|
||||||
RemoteHost = DefaultRemoteHost
|
RemoteHost = DefaultRemoteHost
|
||||||
|
@ -57,44 +56,35 @@ var (
|
||||||
|
|
||||||
type Info struct {
|
type Info struct {
|
||||||
Keywords []string // tags
|
Keywords []string // tags
|
||||||
MimeType string // maybe preserve the type provided?
|
|
||||||
Ip string // who uploaded it
|
Ip string // who uploaded it
|
||||||
Date time.Time // when?
|
}
|
||||||
|
|
||||||
|
type ImgFile struct {
|
||||||
|
Metadata Info ",omitempty"
|
||||||
|
Md5 string
|
||||||
|
ChunkSize int
|
||||||
|
UploadDate time.Time
|
||||||
|
Length int64
|
||||||
|
Filename string ",omitempty"
|
||||||
|
ContentType string "contentType,omitempty"
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether this Image filename is on Mongo */
|
/* Check whether this Image filename is on Mongo */
|
||||||
func hasImageByFilename(filename string) (exists bool, err error) {
|
func hasImageByFilename(filename string) (exists bool, err error) {
|
||||||
query := McImages.Find(bson.M{"filename": filename})
|
c, err := gfs.Find(bson.M{"filename": filename}).Count()
|
||||||
c, err := query.Count()
|
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
exists = (c > 0)
|
exists = (c > 0)
|
||||||
return exists, nil
|
return exists, nil
|
||||||
}
|
}
|
||||||
func hasImage(i Image) (exists bool, err error) {
|
|
||||||
exists, err = hasImageByFilename(i.Filename)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func saveImage(i Image) (err error) {
|
func getImage(filename string) (img ImgFile, err error) {
|
||||||
exists, err := hasImage(i)
|
err = gfs.Find(bson.M{"filename":filename}).One(&img)
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
return
|
return img, err
|
||||||
}
|
}
|
||||||
if (exists) {
|
return img, nil
|
||||||
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
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -184,6 +174,13 @@ func getSmallHash() (small_hash string) {
|
||||||
return fmt.Sprintf("%X", h.Sum(nil))
|
return fmt.Sprintf("%X", h.Sum(nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func returnErr(w http.ResponseWriter, r *http.Request, e error) {
|
||||||
|
LogRequest(r,503)
|
||||||
|
fmt.Fprintf(w,"Error fetching image: %s", e)
|
||||||
|
http.Error(w, "Service Unavailable", 503)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
GET /i/
|
GET /i/
|
||||||
GET /i/:name
|
GET /i/:name
|
||||||
|
@ -198,16 +195,17 @@ func routeImagesGET(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len(uriChunks) == 2 && len(uriChunks[1]) > 0) {
|
if (len(uriChunks) == 2 && len(uriChunks[1]) > 0) {
|
||||||
query, err := Mgfs.Find(bson.M{"metadata": uriChunks[1] })
|
log.Printf("Searching for [%s] ...", uriChunks[1])
|
||||||
|
query := gfs.Find(bson.M{"metadata": bson.M{"filename": uriChunks[1] } })
|
||||||
|
|
||||||
|
c, err := query.Count()
|
||||||
// preliminary checks, if they've passed an image name
|
// preliminary checks, if they've passed an image name
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
LogRequest(r,503)
|
returnErr(w,r,err)
|
||||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
|
||||||
http.Error(w, "Service Unavailable", 503)
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (query.Count() == 0) {
|
log.Printf("Results for [%s] = %d", uriChunks[1], c)
|
||||||
|
if (c == 0) {
|
||||||
LogRequest(r,404)
|
LogRequest(r,404)
|
||||||
http.NotFound(w,r)
|
http.NotFound(w,r)
|
||||||
return
|
return
|
||||||
|
@ -218,12 +216,9 @@ func routeImagesGET(w http.ResponseWriter, r *http.Request) {
|
||||||
w.Header().Set("Cache-Control", "max-age=315360000")
|
w.Header().Set("Cache-Control", "max-age=315360000")
|
||||||
w.WriteHeader(http.StatusOK)
|
w.WriteHeader(http.StatusOK)
|
||||||
|
|
||||||
file, err := Mgfs.Open(uriChunks[1])
|
file, err := gfs.Open(uriChunks[1])
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
log.Println(err)
|
returnErr(w,r,err)
|
||||||
LogRequest(r,503)
|
|
||||||
fmt.Fprintf(w,"Error fetching image: %s", uriChunks[1])
|
|
||||||
http.Error(w, "Service Unavailable", 503)
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -251,20 +246,20 @@ func routeImagesPOST(w http.ResponseWriter, r *http.Request) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
var i Info
|
var filename string
|
||||||
|
info := Info{
|
||||||
i.Date = bson.Now()
|
Ip: r.RemoteAddr,
|
||||||
i.Ip = r.RemoteAddr
|
}
|
||||||
|
|
||||||
if (len(uriChunks) == 2 && len(uriChunks[1]) != 0) {
|
if (len(uriChunks) == 2 && len(uriChunks[1]) != 0) {
|
||||||
i.Filename = uriChunks[1]
|
filename = uriChunks[1]
|
||||||
}
|
}
|
||||||
params := parseRawQuery(r.URL.RawQuery)
|
params := parseRawQuery(r.URL.RawQuery)
|
||||||
var p_ext string
|
var p_ext string
|
||||||
for k,v := range params {
|
for k,v := range params {
|
||||||
switch {
|
switch {
|
||||||
case (k == "filename"):
|
case (k == "filename"):
|
||||||
i.Filename = v
|
filename = v
|
||||||
case (k == "ext"):
|
case (k == "ext"):
|
||||||
if (v[0] != '.') {
|
if (v[0] != '.') {
|
||||||
p_ext = fmt.Sprintf(".%s", v)
|
p_ext = fmt.Sprintf(".%s", v)
|
||||||
|
@ -272,52 +267,51 @@ func routeImagesPOST(w http.ResponseWriter, r *http.Request) {
|
||||||
p_ext = v
|
p_ext = v
|
||||||
}
|
}
|
||||||
case (k == "k" || k == "key" || k == "keyword"):
|
case (k == "k" || k == "key" || k == "keyword"):
|
||||||
i.Keywords = append(i.Keywords[:], v)
|
info.Keywords = append(info.Keywords[:], v)
|
||||||
case (k == "keys" || k == "keywords"):
|
case (k == "keys" || k == "keywords"):
|
||||||
for _, key := range strings.Split(v, ",") {
|
for _, key := range strings.Split(v, ",") {
|
||||||
i.Keywords = append(i.Keywords[:], key)
|
info.Keywords = append(info.Keywords[:], key)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len(i.Filename) == 0) {
|
if (len(filename) == 0) {
|
||||||
str := getSmallHash()
|
str := getSmallHash()
|
||||||
if (len(p_ext) == 0) {
|
if (len(p_ext) == 0) {
|
||||||
i.Filename = fmt.Sprintf("%s.jpg", str)
|
filename = fmt.Sprintf("%s.jpg", str)
|
||||||
} else {
|
} else {
|
||||||
i.Filename = fmt.Sprintf("%s%s", str, p_ext)
|
filename = fmt.Sprintf("%s%s", str, p_ext)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
err := saveImage(i)
|
exists, err := hasImageByFilename(filename)
|
||||||
if (err != nil && err.Error() != "Image Filename Exists") {
|
if (err == nil && !exists) {
|
||||||
file, err := Mgfs.Create(i.Filename)
|
file, err := gfs.Create(filename)
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
LogRequest(r,503)
|
returnErr(w,r,err)
|
||||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
|
||||||
http.Error(w, "Service Unavailable", 503)
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
n, err := io.Copy(file, r.Body)
|
n, err := io.Copy(file, r.Body)
|
||||||
if (err != nil) {
|
if (err != nil) {
|
||||||
LogRequest(r,503)
|
returnErr(w,r,err)
|
||||||
fmt.Fprintf(w,"Error fetching image: %s", err)
|
|
||||||
http.Error(w, "Service Unavailable", 503)
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (n != r.ContentLength) {
|
if (n != r.ContentLength) {
|
||||||
log.Printf("WARNING: [%s] content-length (%d), content written (%d)",
|
log.Printf("WARNING: [%s] content-length (%d), content written (%d)",
|
||||||
i.Filename,
|
filename,
|
||||||
r.ContentLength,
|
r.ContentLength,
|
||||||
n)
|
n)
|
||||||
}
|
}
|
||||||
file.Close()
|
file.Close()
|
||||||
|
} else if (exists) {
|
||||||
|
log.Printf("[%s] already exists", filename)
|
||||||
} else {
|
} else {
|
||||||
log.Printf("%s: %s", err, i.Filename)
|
returnErr(w,r,err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
io.WriteString(w,
|
io.WriteString(w,
|
||||||
fmt.Sprintf("%s%s/i/%s\n", r.URL.Scheme, r.URL.Host, i.Filename))
|
fmt.Sprintf("%s%s/i/%s\n", r.URL.Scheme, r.URL.Host, filename))
|
||||||
|
|
||||||
LogRequest(r,200)
|
LogRequest(r,200)
|
||||||
}
|
}
|
||||||
|
@ -358,14 +352,14 @@ func routeRoot(w http.ResponseWriter, r *http.Request) {
|
||||||
// Show a page of most recent images, and tags, and uploaders ...
|
// Show a page of most recent images, and tags, and uploaders ...
|
||||||
|
|
||||||
w.Header().Set("Content-Type", "text/html")
|
w.Header().Set("Content-Type", "text/html")
|
||||||
var i Image
|
var img ImgFile
|
||||||
iter := McImages.Find(bson.M{"date": bson.M{"$gt": time.Now().Add(-time.Hour)}}).Limit(10).Iter()
|
iter := gfs.Find(bson.M{"uploadDate": bson.M{"$gt": time.Now().Add(-time.Hour)}}).Limit(10).Iter()
|
||||||
fmt.Fprintf(w, "<li>\n")
|
fmt.Fprintf(w, "<ul>\n")
|
||||||
for iter.Next(&i) {
|
for iter.Next(&img) {
|
||||||
log.Println(i.Filename)
|
log.Println(img.Filename)
|
||||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
fmt.Fprintf(w, "<li>%s</li>\n", linkToImage("", img.Filename))
|
||||||
}
|
}
|
||||||
fmt.Fprintf(w, "</li>\n")
|
fmt.Fprintf(w, "</ul>\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
func routeAll(w http.ResponseWriter, r *http.Request) {
|
func routeAll(w http.ResponseWriter, r *http.Request) {
|
||||||
|
@ -374,16 +368,18 @@ func routeAll(w http.ResponseWriter, r *http.Request) {
|
||||||
http.NotFound(w,r)
|
http.NotFound(w,r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// Show a page of all the images
|
|
||||||
|
|
||||||
var i Image
|
w.Header().Set("Content-Type", "text/html")
|
||||||
iter := McImages.Find(bson.M{"date": bson.M{"$gt": time.Now().Add(-time.Hour)}}).Limit(10).Iter()
|
var img ImgFile
|
||||||
fmt.Fprintf(w, "<li>\n")
|
|
||||||
for iter.Next(&i) {
|
// Show a page of all the images
|
||||||
log.Println(i.Filename)
|
iter := gfs.Find(nil).Iter()
|
||||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
fmt.Fprintf(w, "<ul>\n")
|
||||||
|
for iter.Next(&img) {
|
||||||
|
log.Println(img.Filename)
|
||||||
|
fmt.Fprintf(w, "<li>%s</li>\n", linkToImage("", img.Filename))
|
||||||
}
|
}
|
||||||
fmt.Fprintf(w, "</li>\n")
|
fmt.Fprintf(w, "</ul>\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -397,7 +393,7 @@ func routeAll(w http.ResponseWriter, r *http.Request) {
|
||||||
*/
|
*/
|
||||||
func routeKeywords(w http.ResponseWriter, r *http.Request) {
|
func routeKeywords(w http.ResponseWriter, r *http.Request) {
|
||||||
uriChunks := chunkURI(r.URL.Path)
|
uriChunks := chunkURI(r.URL.Path)
|
||||||
if (r.Method != "GET" ||
|
if (r.Method != "GET" ||
|
||||||
len(uriChunks) > 3 ||
|
len(uriChunks) > 3 ||
|
||||||
(len(uriChunks) == 3 && uriChunks[2] != "r")) {
|
(len(uriChunks) == 3 && uriChunks[2] != "r")) {
|
||||||
LogRequest(r,404)
|
LogRequest(r,404)
|
||||||
|
@ -409,23 +405,23 @@ func routeKeywords(w http.ResponseWriter, r *http.Request) {
|
||||||
params := parseRawQuery(r.URL.RawQuery)
|
params := parseRawQuery(r.URL.RawQuery)
|
||||||
log.Printf("K: params: %s", params)
|
log.Printf("K: params: %s", params)
|
||||||
|
|
||||||
var i Image
|
var img ImgFile
|
||||||
if (len(uriChunks) == 1) {
|
if (len(uriChunks) == 1) {
|
||||||
// show a sorted list of tag name links
|
// show a sorted list of tag name links
|
||||||
iter := McImages.Find(bson.M{"keywords": uriChunks[1]}).Sort("$natural").Limit(100).Iter()
|
iter := gfs.Find(bson.M{"metadata": bson.M{"keywords": uriChunks[1] } }).Sort("$natural").Limit(100).Iter()
|
||||||
fmt.Fprintf(w, "<li>\n")
|
fmt.Fprintf(w, "<li>\n")
|
||||||
for iter.Next(&i) {
|
for iter.Next(&img) {
|
||||||
log.Println(i.Filename)
|
log.Println(img.Filename)
|
||||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", img.Filename))
|
||||||
}
|
}
|
||||||
fmt.Fprintf(w, "</li>\n")
|
fmt.Fprintf(w, "</li>\n")
|
||||||
} else if (len(uriChunks) == 2) {
|
} else if (len(uriChunks) == 2) {
|
||||||
iter := McImages.Find(bson.M{"keywords": uriChunks[1]}).Limit(10).Iter()
|
iter := gfs.Find(bson.M{"metadata": bson.M{"keywords": uriChunks[1] } }).Limit(10).Iter()
|
||||||
|
|
||||||
fmt.Fprintf(w, "<li>\n")
|
fmt.Fprintf(w, "<li>\n")
|
||||||
for iter.Next(&i) {
|
for iter.Next(&img) {
|
||||||
log.Println(i.Filename)
|
log.Println(img.Filename)
|
||||||
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", i.Filename))
|
fmt.Fprintf(w, "<ul>%s</ul>\n", linkToImage("", img.Filename))
|
||||||
}
|
}
|
||||||
fmt.Fprintf(w, "</li>\n")
|
fmt.Fprintf(w, "</li>\n")
|
||||||
} else if (uriChunks[2] == "r") {
|
} else if (uriChunks[2] == "r") {
|
||||||
|
@ -445,14 +441,12 @@ func routeIPs(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func initMongo() {
|
func initMongo() {
|
||||||
MSession, err := mgo.Dial(MongoHost)
|
mongo_session, err := mgo.Dial(MongoHost)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Panic(err)
|
log.Panic(err)
|
||||||
}
|
}
|
||||||
|
images_db = mongo_session.DB(MongoDB)
|
||||||
MongoImagesDb = MSession.DB(MongoDB)
|
gfs = images_db.GridFS("fs")
|
||||||
Mgfs = MongoImagesDb.GridFS("fs")
|
|
||||||
McImages = MongoImagesDb.C(MongoCollectionImages)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Run as the image server */
|
/* Run as the image server */
|
||||||
|
@ -460,7 +454,7 @@ func runServer(ip, port string) {
|
||||||
var addr = fmt.Sprintf("%s:%s", ip, port)
|
var addr = fmt.Sprintf("%s:%s", ip, port)
|
||||||
|
|
||||||
initMongo()
|
initMongo()
|
||||||
defer MSession.Close()
|
defer mongo_session.Close()
|
||||||
|
|
||||||
http.HandleFunc("/", routeRoot)
|
http.HandleFunc("/", routeRoot)
|
||||||
http.HandleFunc("/all", routeAll)
|
http.HandleFunc("/all", routeAll)
|
||||||
|
|
Loading…
Reference in a new issue