Initial commit
This commit is contained in:
commit
2dbe318b18
24
.gitignore
vendored
Normal file
24
.gitignore
vendored
Normal file
@ -0,0 +1,24 @@
|
||||
# Compiled Object files, Static and Dynamic libs (Shared Objects)
|
||||
*.o
|
||||
*.a
|
||||
*.so
|
||||
|
||||
# Folders
|
||||
_obj
|
||||
_test
|
||||
|
||||
# Architecture specific extensions/prefixes
|
||||
*.[568vq]
|
||||
[568vq].out
|
||||
|
||||
*.cgo1.go
|
||||
*.cgo2.c
|
||||
_cgo_defun.c
|
||||
_cgo_gotypes.go
|
||||
_cgo_export.*
|
||||
|
||||
_testmain.go
|
||||
|
||||
*.exe
|
||||
*.test
|
||||
*.prof
|
24
README.md
Normal file
24
README.md
Normal file
@ -0,0 +1,24 @@
|
||||
Linx Server
|
||||
======
|
||||
Soon-to-be opensource replacement of Linx
|
||||
|
||||
**Please do not use yet! Consider it in pre-alpha development stages.**
|
||||
|
||||
Build & Run
|
||||
----------------
|
||||
|
||||
1. ```go get -u github.com/andreimarcu/linx-server ```
|
||||
2. ```cd $GOPATH/src/github.com/andreimarcu/linx-server ```
|
||||
3. ```go build && ./linx-server```
|
||||
|
||||
By default linx will bind to ```http://127.0.0.1:8080/```, use the "files/" files directory and the "linx" sitename.
|
||||
Configurable flags can be found in ```server.go```.
|
||||
|
||||
|
||||
TODO
|
||||
--------
|
||||
Please refer to the [main TODO issue](https://github.com/andreimarcu/linx-server/issues/1)
|
||||
|
||||
Author
|
||||
-------
|
||||
Andrei Marcu, http://andreim.net/
|
54
display.go
Normal file
54
display.go
Normal file
@ -0,0 +1,54 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"os"
|
||||
"path"
|
||||
"strings"
|
||||
|
||||
"github.com/flosch/pongo2"
|
||||
"github.com/rakyll/magicmime"
|
||||
"github.com/zenazn/goji/web"
|
||||
)
|
||||
|
||||
func fileDisplayHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
||||
filename := c.URLParams["name"]
|
||||
absPath := path.Join(Config.filesDir, filename)
|
||||
fileInfo, err := os.Stat(absPath)
|
||||
|
||||
if os.IsNotExist(err) {
|
||||
http.Error(w, http.StatusText(404), 404)
|
||||
return
|
||||
}
|
||||
|
||||
if err := magicmime.Open(magicmime.MAGIC_MIME_TYPE |
|
||||
magicmime.MAGIC_SYMLINK |
|
||||
magicmime.MAGIC_ERROR); err != nil {
|
||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||
}
|
||||
defer magicmime.Close()
|
||||
|
||||
mimetype, err := magicmime.TypeByFile(absPath)
|
||||
if err != nil {
|
||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||
}
|
||||
|
||||
var tpl *pongo2.Template
|
||||
|
||||
if strings.HasPrefix(mimetype, "image/") {
|
||||
tpl = pongo2.Must(pongo2.FromCache("templates/display/image.html"))
|
||||
} else {
|
||||
tpl = pongo2.Must(pongo2.FromCache("templates/display/file.html"))
|
||||
}
|
||||
|
||||
err = tpl.ExecuteWriter(pongo2.Context{
|
||||
"mime": mimetype,
|
||||
"sitename": Config.siteName,
|
||||
"filename": filename,
|
||||
"size": fileInfo.Size(),
|
||||
}, w)
|
||||
|
||||
if err != nil {
|
||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||
}
|
||||
}
|
17
pages.go
Normal file
17
pages.go
Normal file
@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"github.com/flosch/pongo2"
|
||||
"net/http"
|
||||
|
||||
"github.com/zenazn/goji/web"
|
||||
)
|
||||
|
||||
func indexHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
||||
indexTpl := pongo2.Must(pongo2.FromCache("templates/index.html"))
|
||||
|
||||
err := indexTpl.ExecuteWriter(pongo2.Context{"sitename": Config.siteName}, w)
|
||||
if err != nil {
|
||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||
}
|
||||
}
|
47
server.go
Normal file
47
server.go
Normal file
@ -0,0 +1,47 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"fmt"
|
||||
"log"
|
||||
"net"
|
||||
"net/http"
|
||||
"regexp"
|
||||
|
||||
"github.com/zenazn/goji"
|
||||
// "github.com/zenazn/goji/web"
|
||||
)
|
||||
|
||||
var Config struct {
|
||||
bind string
|
||||
filesDir string
|
||||
siteName string
|
||||
}
|
||||
|
||||
func main() {
|
||||
flag.StringVar(&Config.bind, "b", "127.0.0.1:8080",
|
||||
"host to bind to (default: 127.0.0.1:8080)")
|
||||
flag.StringVar(&Config.filesDir, "d", "files/",
|
||||
"path to files directory (default: files/)")
|
||||
flag.StringVar(&Config.siteName, "n", "linx",
|
||||
"name of the site")
|
||||
flag.Parse()
|
||||
|
||||
fmt.Printf("About to listen on http://%s\n", Config.bind)
|
||||
|
||||
nameRe := regexp.MustCompile(`^/(?P<name>[a-z0-9-\.]+)$`)
|
||||
selifRe := regexp.MustCompile(`^/selif/(?P<name>[a-z0-9-\.]+)$`)
|
||||
|
||||
goji.Get("/", indexHandler)
|
||||
goji.Post("/upload", uploadPostHandler)
|
||||
goji.Put("/upload", uploadPutHandler)
|
||||
goji.Get(nameRe, fileDisplayHandler)
|
||||
goji.Handle(selifRe, http.StripPrefix("/selif/", http.FileServer(http.Dir(Config.filesDir))))
|
||||
|
||||
listener, err := net.Listen("tcp", Config.bind)
|
||||
if err != nil {
|
||||
log.Fatal("Could not bind: ", err)
|
||||
}
|
||||
|
||||
goji.ServeListener(listener)
|
||||
}
|
13
templates/base.html
Normal file
13
templates/base.html
Normal file
@ -0,0 +1,13 @@
|
||||
<html>
|
||||
<head>
|
||||
<title>{% block title %}{{ sitename }}{% endblock %}</title>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<div>
|
||||
{% block content %}{% endblock %}
|
||||
</div>
|
||||
|
||||
|
||||
</body>
|
||||
</html>
|
6
templates/display/file.html
Normal file
6
templates/display/file.html
Normal file
@ -0,0 +1,6 @@
|
||||
{% extends "../base.html" %}
|
||||
|
||||
{% block content %}
|
||||
<p>Viewing file {{ filename }} of mime: {{ mime }}</p>
|
||||
<p><a href="/selif/{{ filename }}">Click to download</a></p>
|
||||
{% endblock %}
|
6
templates/display/image.html
Normal file
6
templates/display/image.html
Normal file
@ -0,0 +1,6 @@
|
||||
{% extends "../base.html" %}
|
||||
|
||||
{% block content %}
|
||||
<p>Viewing file {{ filename }}</p>
|
||||
<img src="/selif/{{ filename }}" />
|
||||
{% endblock %}
|
8
templates/index.html
Normal file
8
templates/index.html
Normal file
@ -0,0 +1,8 @@
|
||||
{% extends "base.html" %}
|
||||
|
||||
{% block content %}
|
||||
<form enctype="multipart/form-data" action="/upload" method="post">
|
||||
<input type="file" name="file" />
|
||||
<input type="submit" value="upload" />
|
||||
</form>
|
||||
{% endblock %}
|
118
upload.go
Normal file
118
upload.go
Normal file
@ -0,0 +1,118 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"os"
|
||||
"path"
|
||||
"regexp"
|
||||
"strings"
|
||||
|
||||
"code.google.com/p/go-uuid/uuid"
|
||||
"github.com/zenazn/goji/web"
|
||||
)
|
||||
|
||||
type UploadRequest struct {
|
||||
src io.Reader
|
||||
filename string
|
||||
expiry int
|
||||
randomBarename bool
|
||||
}
|
||||
|
||||
type Upload struct {
|
||||
Filename string
|
||||
Size int64
|
||||
Expiry int
|
||||
}
|
||||
|
||||
func uploadPostHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
||||
upReq := UploadRequest{}
|
||||
|
||||
file, headers, err := r.FormFile("file")
|
||||
if err != nil {
|
||||
fmt.Fprintf(w, err.Error())
|
||||
return
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
upReq.src = file
|
||||
upReq.filename = headers.Filename
|
||||
|
||||
upload, err := processUpload(upReq)
|
||||
if err != nil {
|
||||
fmt.Fprintf(w, "Failed to upload: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
fmt.Fprintf(w, "File %s uploaded successfully.", upload.Filename)
|
||||
}
|
||||
|
||||
func uploadPutHandler(c web.C, w http.ResponseWriter, r *http.Request) {
|
||||
upReq := UploadRequest{}
|
||||
|
||||
defer r.Body.Close()
|
||||
upReq.src = r.Body
|
||||
|
||||
upload, err := processUpload(upReq)
|
||||
if err != nil {
|
||||
fmt.Fprintf(w, "Failed to upload")
|
||||
return
|
||||
}
|
||||
|
||||
fmt.Fprintf(w, "File %s uploaded successfully.", upload.Filename)
|
||||
}
|
||||
|
||||
func processUpload(upReq UploadRequest) (upload Upload, err error) {
|
||||
barename, extension := barePlusExt(upReq.filename)
|
||||
|
||||
if upReq.randomBarename || len(barename) == 0 {
|
||||
barename = generateBarename()
|
||||
}
|
||||
|
||||
if len(extension) == 0 {
|
||||
extension = "ext"
|
||||
}
|
||||
|
||||
upload.Filename = strings.Join([]string{barename, extension}, ".")
|
||||
|
||||
dst, ferr := os.Create(path.Join("files/", upload.Filename))
|
||||
defer dst.Close()
|
||||
if ferr != nil {
|
||||
err = ferr
|
||||
return
|
||||
}
|
||||
|
||||
bytes, cerr := io.Copy(dst, upReq.src)
|
||||
if cerr != nil {
|
||||
err = cerr
|
||||
return
|
||||
} else if bytes == 0 {
|
||||
err = errors.New("Empty file")
|
||||
return
|
||||
}
|
||||
|
||||
upload.Size = bytes
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func generateBarename() string {
|
||||
return uuid.New()[:8]
|
||||
}
|
||||
|
||||
func barePlusExt(filename string) (barename, extension string) {
|
||||
re := regexp.MustCompile(`[^A-Za-z0-9\-]`)
|
||||
|
||||
filename = strings.TrimSpace(filename)
|
||||
filename = strings.ToLower(filename)
|
||||
|
||||
extension = path.Ext(filename)
|
||||
barename = filename[:len(filename)-len(extension)]
|
||||
|
||||
extension = re.ReplaceAllString(extension, "")
|
||||
barename = re.ReplaceAllString(barename, "")
|
||||
|
||||
return
|
||||
}
|
Loading…
Reference in New Issue
Block a user