api_quota.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. // Copyright (C) 2019 Nicola Murino
  2. //
  3. // This program is free software: you can redistribute it and/or modify
  4. // it under the terms of the GNU Affero General Public License as published
  5. // by the Free Software Foundation, version 3.
  6. //
  7. // This program is distributed in the hope that it will be useful,
  8. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. // GNU Affero General Public License for more details.
  11. //
  12. // You should have received a copy of the GNU Affero General Public License
  13. // along with this program. If not, see <https://www.gnu.org/licenses/>.
  14. package httpd
  15. import (
  16. "errors"
  17. "fmt"
  18. "net/http"
  19. "github.com/go-chi/render"
  20. "github.com/drakkan/sftpgo/v2/internal/common"
  21. "github.com/drakkan/sftpgo/v2/internal/dataprovider"
  22. "github.com/drakkan/sftpgo/v2/internal/logger"
  23. "github.com/drakkan/sftpgo/v2/internal/vfs"
  24. )
  25. const (
  26. quotaUpdateModeAdd = "add"
  27. quotaUpdateModeReset = "reset"
  28. )
  29. type quotaUsage struct {
  30. UsedQuotaSize int64 `json:"used_quota_size"`
  31. UsedQuotaFiles int `json:"used_quota_files"`
  32. }
  33. type transferQuotaUsage struct {
  34. UsedUploadDataTransfer int64 `json:"used_upload_data_transfer"`
  35. UsedDownloadDataTransfer int64 `json:"used_download_data_transfer"`
  36. }
  37. func getUsersQuotaScans(w http.ResponseWriter, r *http.Request) {
  38. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  39. claims, err := getTokenClaims(r)
  40. if err != nil || claims.Username == "" {
  41. sendAPIResponse(w, r, err, "Invalid token claims", http.StatusBadRequest)
  42. return
  43. }
  44. render.JSON(w, r, common.QuotaScans.GetUsersQuotaScans(claims.Role))
  45. }
  46. func getFoldersQuotaScans(w http.ResponseWriter, r *http.Request) {
  47. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  48. render.JSON(w, r, common.QuotaScans.GetVFoldersQuotaScans())
  49. }
  50. func updateUserQuotaUsage(w http.ResponseWriter, r *http.Request) {
  51. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  52. var usage quotaUsage
  53. err := render.DecodeJSON(r.Body, &usage)
  54. if err != nil {
  55. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  56. return
  57. }
  58. doUpdateUserQuotaUsage(w, r, getURLParam(r, "username"), usage)
  59. }
  60. func updateFolderQuotaUsage(w http.ResponseWriter, r *http.Request) {
  61. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  62. var usage quotaUsage
  63. err := render.DecodeJSON(r.Body, &usage)
  64. if err != nil {
  65. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  66. return
  67. }
  68. doUpdateFolderQuotaUsage(w, r, getURLParam(r, "name"), usage)
  69. }
  70. func startUserQuotaScan(w http.ResponseWriter, r *http.Request) {
  71. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  72. doStartUserQuotaScan(w, r, getURLParam(r, "username"))
  73. }
  74. func startFolderQuotaScan(w http.ResponseWriter, r *http.Request) {
  75. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  76. doStartFolderQuotaScan(w, r, getURLParam(r, "name"))
  77. }
  78. func updateUserTransferQuotaUsage(w http.ResponseWriter, r *http.Request) {
  79. r.Body = http.MaxBytesReader(w, r.Body, maxRequestSize)
  80. claims, err := getTokenClaims(r)
  81. if err != nil || claims.Username == "" {
  82. sendAPIResponse(w, r, err, "Invalid token claims", http.StatusBadRequest)
  83. return
  84. }
  85. var usage transferQuotaUsage
  86. err = render.DecodeJSON(r.Body, &usage)
  87. if err != nil {
  88. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  89. return
  90. }
  91. if usage.UsedUploadDataTransfer < 0 || usage.UsedDownloadDataTransfer < 0 {
  92. sendAPIResponse(w, r, errors.New("invalid used transfer quota parameters, negative values are not allowed"),
  93. "", http.StatusBadRequest)
  94. return
  95. }
  96. mode, err := getQuotaUpdateMode(r)
  97. if err != nil {
  98. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  99. return
  100. }
  101. user, err := dataprovider.GetUserWithGroupSettings(getURLParam(r, "username"), claims.Role)
  102. if err != nil {
  103. sendAPIResponse(w, r, err, "", getRespStatus(err))
  104. return
  105. }
  106. if mode == quotaUpdateModeAdd && !user.HasTransferQuotaRestrictions() && dataprovider.GetQuotaTracking() == 2 {
  107. sendAPIResponse(w, r, errors.New("this user has no transfer quota restrictions, only reset mode is supported"),
  108. "", http.StatusBadRequest)
  109. return
  110. }
  111. err = dataprovider.UpdateUserTransferQuota(&user, usage.UsedUploadDataTransfer, usage.UsedDownloadDataTransfer,
  112. mode == quotaUpdateModeReset)
  113. if err != nil {
  114. sendAPIResponse(w, r, err, "", getRespStatus(err))
  115. return
  116. }
  117. sendAPIResponse(w, r, err, "Quota updated", http.StatusOK)
  118. }
  119. func doUpdateUserQuotaUsage(w http.ResponseWriter, r *http.Request, username string, usage quotaUsage) {
  120. claims, err := getTokenClaims(r)
  121. if err != nil || claims.Username == "" {
  122. sendAPIResponse(w, r, err, "Invalid token claims", http.StatusBadRequest)
  123. return
  124. }
  125. if usage.UsedQuotaFiles < 0 || usage.UsedQuotaSize < 0 {
  126. sendAPIResponse(w, r, errors.New("invalid used quota parameters, negative values are not allowed"),
  127. "", http.StatusBadRequest)
  128. return
  129. }
  130. mode, err := getQuotaUpdateMode(r)
  131. if err != nil {
  132. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  133. return
  134. }
  135. user, err := dataprovider.GetUserWithGroupSettings(username, claims.Role)
  136. if err != nil {
  137. sendAPIResponse(w, r, err, "", getRespStatus(err))
  138. return
  139. }
  140. if mode == quotaUpdateModeAdd && !user.HasQuotaRestrictions() && dataprovider.GetQuotaTracking() == 2 {
  141. sendAPIResponse(w, r, errors.New("this user has no quota restrictions, only reset mode is supported"),
  142. "", http.StatusBadRequest)
  143. return
  144. }
  145. if !common.QuotaScans.AddUserQuotaScan(user.Username, user.Role) {
  146. sendAPIResponse(w, r, err, "A quota scan is in progress for this user", http.StatusConflict)
  147. return
  148. }
  149. defer common.QuotaScans.RemoveUserQuotaScan(user.Username)
  150. err = dataprovider.UpdateUserQuota(&user, usage.UsedQuotaFiles, usage.UsedQuotaSize, mode == quotaUpdateModeReset)
  151. if err != nil {
  152. sendAPIResponse(w, r, err, "", getRespStatus(err))
  153. return
  154. }
  155. sendAPIResponse(w, r, err, "Quota updated", http.StatusOK)
  156. }
  157. func doUpdateFolderQuotaUsage(w http.ResponseWriter, r *http.Request, name string, usage quotaUsage) {
  158. if usage.UsedQuotaFiles < 0 || usage.UsedQuotaSize < 0 {
  159. sendAPIResponse(w, r, errors.New("invalid used quota parameters, negative values are not allowed"),
  160. "", http.StatusBadRequest)
  161. return
  162. }
  163. mode, err := getQuotaUpdateMode(r)
  164. if err != nil {
  165. sendAPIResponse(w, r, err, "", http.StatusBadRequest)
  166. return
  167. }
  168. folder, err := dataprovider.GetFolderByName(name)
  169. if err != nil {
  170. sendAPIResponse(w, r, err, "", getRespStatus(err))
  171. return
  172. }
  173. if !common.QuotaScans.AddVFolderQuotaScan(folder.Name) {
  174. sendAPIResponse(w, r, err, "A quota scan is in progress for this folder", http.StatusConflict)
  175. return
  176. }
  177. defer common.QuotaScans.RemoveVFolderQuotaScan(folder.Name)
  178. err = dataprovider.UpdateVirtualFolderQuota(&folder, usage.UsedQuotaFiles, usage.UsedQuotaSize, mode == quotaUpdateModeReset)
  179. if err != nil {
  180. sendAPIResponse(w, r, err, "", getRespStatus(err))
  181. } else {
  182. sendAPIResponse(w, r, err, "Quota updated", http.StatusOK)
  183. }
  184. }
  185. func doStartUserQuotaScan(w http.ResponseWriter, r *http.Request, username string) {
  186. if dataprovider.GetQuotaTracking() == 0 {
  187. sendAPIResponse(w, r, nil, "Quota tracking is disabled!", http.StatusForbidden)
  188. return
  189. }
  190. claims, err := getTokenClaims(r)
  191. if err != nil || claims.Username == "" {
  192. sendAPIResponse(w, r, err, "Invalid token claims", http.StatusBadRequest)
  193. return
  194. }
  195. user, err := dataprovider.GetUserWithGroupSettings(username, claims.Role)
  196. if err != nil {
  197. sendAPIResponse(w, r, err, "", getRespStatus(err))
  198. return
  199. }
  200. if !common.QuotaScans.AddUserQuotaScan(user.Username, user.Role) {
  201. sendAPIResponse(w, r, nil, fmt.Sprintf("Another scan is already in progress for user %q", username),
  202. http.StatusConflict)
  203. return
  204. }
  205. go doUserQuotaScan(user) //nolint:errcheck
  206. sendAPIResponse(w, r, err, "Scan started", http.StatusAccepted)
  207. }
  208. func doStartFolderQuotaScan(w http.ResponseWriter, r *http.Request, name string) {
  209. if dataprovider.GetQuotaTracking() == 0 {
  210. sendAPIResponse(w, r, nil, "Quota tracking is disabled!", http.StatusForbidden)
  211. return
  212. }
  213. folder, err := dataprovider.GetFolderByName(name)
  214. if err != nil {
  215. sendAPIResponse(w, r, err, "", getRespStatus(err))
  216. return
  217. }
  218. if !common.QuotaScans.AddVFolderQuotaScan(folder.Name) {
  219. sendAPIResponse(w, r, err, fmt.Sprintf("Another scan is already in progress for folder %q", name),
  220. http.StatusConflict)
  221. return
  222. }
  223. go doFolderQuotaScan(folder) //nolint:errcheck
  224. sendAPIResponse(w, r, err, "Scan started", http.StatusAccepted)
  225. }
  226. func doUserQuotaScan(user dataprovider.User) error {
  227. defer common.QuotaScans.RemoveUserQuotaScan(user.Username)
  228. numFiles, size, err := user.ScanQuota()
  229. if err != nil {
  230. logger.Warn(logSender, "", "error scanning user quota %q: %v", user.Username, err)
  231. return err
  232. }
  233. err = dataprovider.UpdateUserQuota(&user, numFiles, size, true)
  234. logger.Debug(logSender, "", "user quota scanned, user: %q, error: %v", user.Username, err)
  235. return err
  236. }
  237. func doFolderQuotaScan(folder vfs.BaseVirtualFolder) error {
  238. defer common.QuotaScans.RemoveVFolderQuotaScan(folder.Name)
  239. f := vfs.VirtualFolder{
  240. BaseVirtualFolder: folder,
  241. VirtualPath: "/",
  242. }
  243. numFiles, size, err := f.ScanQuota()
  244. if err != nil {
  245. logger.Warn(logSender, "", "error scanning folder %q: %v", folder.Name, err)
  246. return err
  247. }
  248. err = dataprovider.UpdateVirtualFolderQuota(&folder, numFiles, size, true)
  249. logger.Debug(logSender, "", "virtual folder %q scanned, error: %v", folder.Name, err)
  250. return err
  251. }
  252. func getQuotaUpdateMode(r *http.Request) (string, error) {
  253. mode := quotaUpdateModeReset
  254. if _, ok := r.URL.Query()["mode"]; ok {
  255. mode = r.URL.Query().Get("mode")
  256. if mode != quotaUpdateModeReset && mode != quotaUpdateModeAdd {
  257. return "", errors.New("invalid mode")
  258. }
  259. }
  260. return mode, nil
  261. }