seaweedfs/weed/server/filer_server_handlers_write.go

237 lines
6.7 KiB
Go
Raw Normal View History

2016-06-03 11:05:34 +08:00
package weed_server
import (
"encoding/json"
"errors"
"io/ioutil"
"net/http"
"net/url"
"strconv"
"strings"
2018-05-28 14:59:49 +08:00
"time"
2018-05-28 02:52:26 +08:00
"github.com/chrislusf/seaweedfs/weed/filer2"
2016-06-03 11:05:34 +08:00
"github.com/chrislusf/seaweedfs/weed/glog"
"github.com/chrislusf/seaweedfs/weed/operation"
2018-05-28 02:52:26 +08:00
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
2016-06-03 11:05:34 +08:00
"github.com/chrislusf/seaweedfs/weed/util"
"os"
)
var (
OS_UID = uint32(os.Getuid())
OS_GID = uint32(os.Getgid())
2016-06-03 11:05:34 +08:00
)
type FilerPostResult struct {
Name string `json:"name,omitempty"`
Size uint32 `json:"size,omitempty"`
Error string `json:"error,omitempty"`
Fid string `json:"fid,omitempty"`
Url string `json:"url,omitempty"`
}
2016-06-08 15:46:14 +08:00
func (fs *FilerServer) queryFileInfoByPath(w http.ResponseWriter, r *http.Request, path string) (fileId, urlLocation string, err error) {
2018-05-14 14:56:16 +08:00
var entry *filer2.Entry
entry, err = fs.filer.FindEntry(filer2.FullPath(path))
if err == filer2.ErrNotFound {
return "", "", nil
}
if err != nil {
2016-06-08 15:46:14 +08:00
glog.V(0).Infoln("failing to find path in filer store", path, err.Error())
writeJsonError(w, r, http.StatusInternalServerError, err)
return
}
if len(entry.Chunks) == 0 {
glog.V(1).Infof("empty entry: %s", path)
w.WriteHeader(http.StatusNoContent)
} else {
fileId = entry.Chunks[0].FileId
urlLocation, err = operation.LookupFileId(fs.filer.GetMaster(), fileId)
if err != nil {
glog.V(1).Infof("operation LookupFileId %s failed, err is %s", fileId, err.Error())
w.WriteHeader(http.StatusNotFound)
2016-06-08 15:46:14 +08:00
}
}
return
}
func (fs *FilerServer) assignNewFileInfo(w http.ResponseWriter, r *http.Request, replication, collection string, dataCenter string) (fileId, urlLocation string, err error) {
2016-06-26 10:50:18 +08:00
ar := &operation.VolumeAssignRequest{
Count: 1,
Replication: replication,
Collection: collection,
Ttl: r.URL.Query().Get("ttl"),
DataCenter: dataCenter,
2016-06-26 10:50:18 +08:00
}
2018-07-10 15:20:50 +08:00
var altRequest *operation.VolumeAssignRequest
if dataCenter != "" {
2018-07-10 15:20:50 +08:00
altRequest = &operation.VolumeAssignRequest{
Count: 1,
Replication: replication,
Collection: collection,
Ttl: r.URL.Query().Get("ttl"),
DataCenter: "",
}
}
2018-07-10 15:20:50 +08:00
assignResult, ae := operation.Assign(fs.filer.GetMaster(), ar, altRequest)
2016-06-08 15:46:14 +08:00
if ae != nil {
glog.Errorf("failing to assign a file id: %v", ae)
2016-06-08 15:46:14 +08:00
writeJsonError(w, r, http.StatusInternalServerError, ae)
err = ae
return
}
fileId = assignResult.Fid
urlLocation = "http://" + assignResult.Url + "/" + assignResult.Fid
return
}
2016-06-03 11:05:34 +08:00
func (fs *FilerServer) PostHandler(w http.ResponseWriter, r *http.Request) {
2016-06-03 11:05:34 +08:00
query := r.URL.Query()
replication := query.Get("replication")
if replication == "" {
2018-07-07 17:18:47 +08:00
replication = fs.option.DefaultReplication
2016-06-03 11:05:34 +08:00
}
collection := query.Get("collection")
if collection == "" {
2018-07-07 17:18:47 +08:00
collection = fs.option.Collection
2016-06-03 11:05:34 +08:00
}
dataCenter := query.Get("dataCenter")
if dataCenter == "" {
dataCenter = fs.option.DataCenter
}
2016-06-03 11:05:34 +08:00
if autoChunked := fs.autoChunk(w, r, replication, collection, dataCenter); autoChunked {
return
}
fileId, urlLocation, err := fs.queryFileInfoByPath(w, r, r.URL.Path)
if err == nil && fileId == "" {
fileId, urlLocation, err = fs.assignNewFileInfo(w, r, replication, collection, dataCenter)
}
if err != nil || fileId == "" || urlLocation == "" {
return
}
glog.V(0).Infof("request header %+v, urlLocation: %v", r.Header, urlLocation)
2016-06-03 11:05:34 +08:00
u, _ := url.Parse(urlLocation)
// This allows a client to generate a chunk manifest and submit it to the filer -- it is a little off
// because they need to provide FIDs instead of file paths...
cm, _ := strconv.ParseBool(query.Get("cm"))
if cm {
q := u.Query()
q.Set("cm", "true")
u.RawQuery = q.Encode()
}
2016-06-03 11:05:34 +08:00
glog.V(4).Infoln("post to", u)
2018-07-22 08:54:14 +08:00
// send request to volume server
2016-06-03 11:05:34 +08:00
request := &http.Request{
Method: r.Method,
URL: u,
Proto: r.Proto,
ProtoMajor: r.ProtoMajor,
ProtoMinor: r.ProtoMinor,
Header: r.Header,
Body: r.Body,
Host: r.Host,
ContentLength: r.ContentLength,
}
resp, do_err := util.Do(request)
if do_err != nil {
glog.Errorf("failing to connect to volume server %s: %v, %+v", r.RequestURI, do_err, r.Method)
2016-06-03 11:05:34 +08:00
writeJsonError(w, r, http.StatusInternalServerError, do_err)
return
}
defer resp.Body.Close()
resp_body, ra_err := ioutil.ReadAll(resp.Body)
if ra_err != nil {
glog.V(0).Infoln("failing to upload to volume server", r.RequestURI, ra_err.Error())
writeJsonError(w, r, http.StatusInternalServerError, ra_err)
return
}
glog.V(4).Infoln("post result", string(resp_body))
var ret operation.UploadResult
unmarshal_err := json.Unmarshal(resp_body, &ret)
if unmarshal_err != nil {
glog.V(0).Infoln("failing to read upload resonse", r.RequestURI, string(resp_body))
writeJsonError(w, r, http.StatusInternalServerError, unmarshal_err)
return
}
if ret.Error != "" {
glog.V(0).Infoln("failing to post to volume server", r.RequestURI, ret.Error)
writeJsonError(w, r, http.StatusInternalServerError, errors.New(ret.Error))
return
}
2018-07-22 08:54:14 +08:00
// find correct final path
2016-06-03 11:05:34 +08:00
path := r.URL.Path
if strings.HasSuffix(path, "/") {
if ret.Name != "" {
path += ret.Name
} else {
operation.DeleteFile(fs.filer.GetMaster(), fileId, fs.jwt(fileId)) //clean up
2016-06-03 11:05:34 +08:00
glog.V(0).Infoln("Can not to write to folder", path, "without a file name!")
writeJsonError(w, r, http.StatusInternalServerError,
errors.New("Can not to write to folder "+path+" without a file name"))
return
}
}
2018-07-22 08:54:14 +08:00
// update metadata in filer store
2016-06-03 11:05:34 +08:00
glog.V(4).Infoln("saving", path, "=>", fileId)
2018-05-14 14:56:16 +08:00
entry := &filer2.Entry{
FullPath: filer2.FullPath(path),
Attr: filer2.Attr{
Mtime: time.Now(),
Crtime: time.Now(),
Mode: 0660,
Uid: OS_UID,
Gid: OS_GID,
Replication: replication,
Collection: collection,
TtlSec: int32(util.ParseInt(r.URL.Query().Get("ttl"), 0)),
2018-05-14 14:56:16 +08:00
},
Chunks: []*filer_pb.FileChunk{{
FileId: fileId,
Size: uint64(ret.Size),
2018-05-16 15:54:44 +08:00
Mtime: time.Now().UnixNano(),
2018-05-14 14:56:16 +08:00
}},
}
if db_err := fs.filer.CreateEntry(entry); db_err != nil {
operation.DeleteFile(fs.filer.GetMaster(), fileId, fs.jwt(fileId)) //clean up
2016-06-03 11:05:34 +08:00
glog.V(0).Infof("failing to write %s to filer server : %v", path, db_err)
writeJsonError(w, r, http.StatusInternalServerError, db_err)
return
}
2018-07-22 08:54:14 +08:00
// send back post result
reply := FilerPostResult{
Name: ret.Name,
Size: ret.Size,
Error: ret.Error,
Fid: fileId,
Url: urlLocation,
}
writeJsonQuiet(w, r, http.StatusCreated, reply)
2016-06-03 11:05:34 +08:00
}
// curl -X DELETE http://localhost:8888/path/to
func (fs *FilerServer) DeleteHandler(w http.ResponseWriter, r *http.Request) {
2018-05-14 14:56:16 +08:00
err := fs.filer.DeleteEntryMetaAndData(filer2.FullPath(r.URL.Path), false, true)
2018-05-14 14:56:16 +08:00
if err != nil {
2018-07-22 09:47:23 +08:00
glog.V(1).Infoln("deleting", r.URL.Path, ":", err.Error())
2016-06-03 11:05:34 +08:00
writeJsonError(w, r, http.StatusInternalServerError, err)
2018-05-14 14:56:16 +08:00
return
}
2018-07-22 09:49:47 +08:00
w.WriteHeader(http.StatusNoContent)
2016-06-03 11:05:34 +08:00
}