You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

236 lines
6.7 KiB

9 years ago
7 years ago
9 years ago
7 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
  1. package weed_server
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "io/ioutil"
  6. "net/http"
  7. "net/url"
  8. "strconv"
  9. "strings"
  10. "time"
  11. "github.com/chrislusf/seaweedfs/weed/filer2"
  12. "github.com/chrislusf/seaweedfs/weed/glog"
  13. "github.com/chrislusf/seaweedfs/weed/operation"
  14. "github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
  15. "github.com/chrislusf/seaweedfs/weed/util"
  16. "os"
  17. )
  18. var (
  19. OS_UID = uint32(os.Getuid())
  20. OS_GID = uint32(os.Getgid())
  21. )
  22. type FilerPostResult struct {
  23. Name string `json:"name,omitempty"`
  24. Size uint32 `json:"size,omitempty"`
  25. Error string `json:"error,omitempty"`
  26. Fid string `json:"fid,omitempty"`
  27. Url string `json:"url,omitempty"`
  28. }
  29. func (fs *FilerServer) queryFileInfoByPath(w http.ResponseWriter, r *http.Request, path string) (fileId, urlLocation string, err error) {
  30. var entry *filer2.Entry
  31. entry, err = fs.filer.FindEntry(filer2.FullPath(path))
  32. if err == filer2.ErrNotFound {
  33. return "", "", nil
  34. }
  35. if err != nil {
  36. glog.V(0).Infoln("failing to find path in filer store", path, err.Error())
  37. writeJsonError(w, r, http.StatusInternalServerError, err)
  38. return
  39. }
  40. if len(entry.Chunks) == 0 {
  41. glog.V(1).Infof("empty entry: %s", path)
  42. w.WriteHeader(http.StatusNoContent)
  43. } else {
  44. fileId = entry.Chunks[0].FileId
  45. urlLocation, err = operation.LookupFileId(fs.filer.GetMaster(), fileId)
  46. if err != nil {
  47. glog.V(1).Infof("operation LookupFileId %s failed, err is %s", fileId, err.Error())
  48. w.WriteHeader(http.StatusNotFound)
  49. }
  50. }
  51. return
  52. }
  53. func (fs *FilerServer) assignNewFileInfo(w http.ResponseWriter, r *http.Request, replication, collection string, dataCenter string) (fileId, urlLocation string, err error) {
  54. ar := &operation.VolumeAssignRequest{
  55. Count: 1,
  56. Replication: replication,
  57. Collection: collection,
  58. Ttl: r.URL.Query().Get("ttl"),
  59. DataCenter: dataCenter,
  60. }
  61. var altRequest *operation.VolumeAssignRequest
  62. if dataCenter != "" {
  63. altRequest = &operation.VolumeAssignRequest{
  64. Count: 1,
  65. Replication: replication,
  66. Collection: collection,
  67. Ttl: r.URL.Query().Get("ttl"),
  68. DataCenter: "",
  69. }
  70. }
  71. assignResult, ae := operation.Assign(fs.filer.GetMaster(), ar, altRequest)
  72. if ae != nil {
  73. glog.Errorf("failing to assign a file id: %v", ae)
  74. writeJsonError(w, r, http.StatusInternalServerError, ae)
  75. err = ae
  76. return
  77. }
  78. fileId = assignResult.Fid
  79. urlLocation = "http://" + assignResult.Url + "/" + assignResult.Fid
  80. return
  81. }
  82. func (fs *FilerServer) PostHandler(w http.ResponseWriter, r *http.Request) {
  83. query := r.URL.Query()
  84. replication := query.Get("replication")
  85. if replication == "" {
  86. replication = fs.option.DefaultReplication
  87. }
  88. collection := query.Get("collection")
  89. if collection == "" {
  90. collection = fs.option.Collection
  91. }
  92. dataCenter := query.Get("dataCenter")
  93. if dataCenter == "" {
  94. dataCenter = fs.option.DataCenter
  95. }
  96. if autoChunked := fs.autoChunk(w, r, replication, collection, dataCenter); autoChunked {
  97. return
  98. }
  99. fileId, urlLocation, err := fs.queryFileInfoByPath(w, r, r.URL.Path)
  100. if err == nil && fileId == "" {
  101. fileId, urlLocation, err = fs.assignNewFileInfo(w, r, replication, collection, dataCenter)
  102. }
  103. if err != nil || fileId == "" || urlLocation == "" {
  104. return
  105. }
  106. glog.V(0).Infof("request header %+v, urlLocation: %v", r.Header, urlLocation)
  107. u, _ := url.Parse(urlLocation)
  108. // This allows a client to generate a chunk manifest and submit it to the filer -- it is a little off
  109. // because they need to provide FIDs instead of file paths...
  110. cm, _ := strconv.ParseBool(query.Get("cm"))
  111. if cm {
  112. q := u.Query()
  113. q.Set("cm", "true")
  114. u.RawQuery = q.Encode()
  115. }
  116. glog.V(4).Infoln("post to", u)
  117. // send request to volume server
  118. request := &http.Request{
  119. Method: r.Method,
  120. URL: u,
  121. Proto: r.Proto,
  122. ProtoMajor: r.ProtoMajor,
  123. ProtoMinor: r.ProtoMinor,
  124. Header: r.Header,
  125. Body: r.Body,
  126. Host: r.Host,
  127. ContentLength: r.ContentLength,
  128. }
  129. resp, do_err := util.Do(request)
  130. if do_err != nil {
  131. glog.Errorf("failing to connect to volume server %s: %v, %+v", r.RequestURI, do_err, r.Method)
  132. writeJsonError(w, r, http.StatusInternalServerError, do_err)
  133. return
  134. }
  135. defer resp.Body.Close()
  136. resp_body, ra_err := ioutil.ReadAll(resp.Body)
  137. if ra_err != nil {
  138. glog.V(0).Infoln("failing to upload to volume server", r.RequestURI, ra_err.Error())
  139. writeJsonError(w, r, http.StatusInternalServerError, ra_err)
  140. return
  141. }
  142. glog.V(4).Infoln("post result", string(resp_body))
  143. var ret operation.UploadResult
  144. unmarshal_err := json.Unmarshal(resp_body, &ret)
  145. if unmarshal_err != nil {
  146. glog.V(0).Infoln("failing to read upload resonse", r.RequestURI, string(resp_body))
  147. writeJsonError(w, r, http.StatusInternalServerError, unmarshal_err)
  148. return
  149. }
  150. if ret.Error != "" {
  151. glog.V(0).Infoln("failing to post to volume server", r.RequestURI, ret.Error)
  152. writeJsonError(w, r, http.StatusInternalServerError, errors.New(ret.Error))
  153. return
  154. }
  155. // find correct final path
  156. path := r.URL.Path
  157. if strings.HasSuffix(path, "/") {
  158. if ret.Name != "" {
  159. path += ret.Name
  160. } else {
  161. operation.DeleteFile(fs.filer.GetMaster(), fileId, fs.jwt(fileId)) //clean up
  162. glog.V(0).Infoln("Can not to write to folder", path, "without a file name!")
  163. writeJsonError(w, r, http.StatusInternalServerError,
  164. errors.New("Can not to write to folder "+path+" without a file name"))
  165. return
  166. }
  167. }
  168. // update metadata in filer store
  169. glog.V(4).Infoln("saving", path, "=>", fileId)
  170. entry := &filer2.Entry{
  171. FullPath: filer2.FullPath(path),
  172. Attr: filer2.Attr{
  173. Mtime: time.Now(),
  174. Crtime: time.Now(),
  175. Mode: 0660,
  176. Uid: OS_UID,
  177. Gid: OS_GID,
  178. Replication: replication,
  179. Collection: collection,
  180. TtlSec: int32(util.ParseInt(r.URL.Query().Get("ttl"), 0)),
  181. },
  182. Chunks: []*filer_pb.FileChunk{{
  183. FileId: fileId,
  184. Size: uint64(ret.Size),
  185. Mtime: time.Now().UnixNano(),
  186. }},
  187. }
  188. if db_err := fs.filer.CreateEntry(entry); db_err != nil {
  189. operation.DeleteFile(fs.filer.GetMaster(), fileId, fs.jwt(fileId)) //clean up
  190. glog.V(0).Infof("failing to write %s to filer server : %v", path, db_err)
  191. writeJsonError(w, r, http.StatusInternalServerError, db_err)
  192. return
  193. }
  194. // send back post result
  195. reply := FilerPostResult{
  196. Name: ret.Name,
  197. Size: ret.Size,
  198. Error: ret.Error,
  199. Fid: fileId,
  200. Url: urlLocation,
  201. }
  202. writeJsonQuiet(w, r, http.StatusCreated, reply)
  203. }
  204. // curl -X DELETE http://localhost:8888/path/to
  205. func (fs *FilerServer) DeleteHandler(w http.ResponseWriter, r *http.Request) {
  206. err := fs.filer.DeleteEntryMetaAndData(filer2.FullPath(r.URL.Path), false, true)
  207. if err != nil {
  208. glog.V(4).Infoln("deleting", r.URL.Path, ":", err.Error())
  209. writeJsonError(w, r, http.StatusInternalServerError, err)
  210. return
  211. }
  212. writeJsonQuiet(w, r, http.StatusAccepted, map[string]string{"error": ""})
  213. }