Start with id generation

This commit is contained in:
Felix Geisendörfer 2013-03-17 16:42:47 +01:00
parent 30de6b93eb
commit 0f4a7bd33f
3 changed files with 86 additions and 52 deletions

61
src/cmd/tusd/http.go Normal file
View File

@ -0,0 +1,61 @@
package main
import (
"fmt"
"net/http"
"log"
)
func serveHttp() error {
http.HandleFunc("/", route)
addr := ":1080"
log.Printf("serving clients at %s", addr)
return http.ListenAndServe(addr, nil)
}
func route(w http.ResponseWriter, r *http.Request) {
log.Printf("request: %s %s", r.Method, r.URL.RequestURI())
w.Header().Set("Server", "tusd")
if r.Method == "POST" && r.URL.Path == "/files" {
createFile(w, r)
} else {
reply(w, http.StatusNotFound, "No matching route")
}
}
func reply(w http.ResponseWriter, code int, message string) {
w.WriteHeader(code)
fmt.Fprintf(w, "%d - %s: %s\n", code, http.StatusText(code), message)
}
func createFile(w http.ResponseWriter, r *http.Request) {
contentRange, err := parseContentRange(r.Header.Get("Content-Range"))
if err != nil {
reply(w, http.StatusBadRequest, err.Error())
return
}
if contentRange.Size == -1 {
reply(w, http.StatusBadRequest, "Content-Range must indicate total file size.")
return
}
if contentRange.End != -1 {
reply(w, http.StatusNotImplemented, "File data in initial request.")
return
}
contentType := r.Header.Get("Content-Type")
if contentType == "" {
contentType = "application/octet-stream"
}
_ = contentType
id := uid()
w.Header().Set("Location", "/files/"+id)
}

View File

@ -1,8 +1,6 @@
package main
import (
"fmt"
"net/http"
"log"
)
@ -10,56 +8,7 @@ func main() {
log.SetFlags(log.Ldate | log.Ltime | log.Lmicroseconds)
log.Printf("tusd started")
http.HandleFunc("/", route)
addr := ":1080"
log.Printf("serving clients at %s", addr)
err := http.ListenAndServe(addr, nil)
if err != nil {
if err := serveHttp(); err != nil {
log.Fatal(err)
}
}
func route(w http.ResponseWriter, r *http.Request) {
log.Printf("request: %s %s", r.Method, r.URL.RequestURI())
w.Header().Set("Server", "tusd")
if r.Method == "POST" && r.URL.Path == "/files" {
createFile(w, r)
} else {
reply(w, http.StatusNotFound, "No matching route")
}
}
func reply(w http.ResponseWriter, code int, message string) {
w.WriteHeader(code)
fmt.Fprintf(w, "%d - %s: %s\n", code, http.StatusText(code), message)
}
func createFile(w http.ResponseWriter, r *http.Request) {
contentRange, err := parseContentRange(r.Header.Get("Content-Range"))
if err != nil {
reply(w, http.StatusBadRequest, err.Error())
return
}
if contentRange.Size == -1 {
reply(w, http.StatusBadRequest, "Content-Range must indicate total file size.")
return
}
if contentRange.End != -1 {
reply(w, http.StatusNotImplemented, "File data in initial request.")
return
}
contentType := r.Header.Get("Content-Type")
if contentType == "" {
contentType = "application/octet-stream"
}
log.Printf("contentType: %s", contentType)
log.Printf("range: %#v", contentRange)
}

24
src/cmd/tusd/uid.go Normal file
View File

@ -0,0 +1,24 @@
package main
import (
"crypto/rand"
"fmt"
"io"
)
// uid returns a unique id. These ids consist of 128 bits from a
// cryptographically strong pseudo-random generator and are like uuids, but
// without the dashes and significant bits.
//
// See: http://en.wikipedia.org/wiki/UUID#Random_UUID_probability_of_duplicates
func uid() string {
id := make([]byte, 16)
_, err := io.ReadFull(rand.Reader, id)
if err != nil {
// This is probably an appropiate way to handle errors from our source
// for random bits.
panic(err)
}
return fmt.Sprintf("%x", id)
}