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

repository.go 9.9 kB

2 years ago
3 years ago
3 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. // Copyright 2019 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package repository
  5. import (
  6. "code.gitea.io/gitea/models"
  7. "code.gitea.io/gitea/modules/git"
  8. "code.gitea.io/gitea/modules/log"
  9. "code.gitea.io/gitea/modules/notification"
  10. repo_module "code.gitea.io/gitea/modules/repository"
  11. "code.gitea.io/gitea/modules/setting"
  12. pull_service "code.gitea.io/gitea/services/pull"
  13. "fmt"
  14. "io/ioutil"
  15. "net/http"
  16. "os"
  17. "strings"
  18. "xorm.io/xorm"
  19. )
  20. const SHELL_FLAG_ON = 1
  21. // CreateRepository creates a repository for the user/organization.
  22. func CreateRepository(doer, owner *models.User, opts models.CreateRepoOptions) (*models.Repository, error) {
  23. repo, err := repo_module.CreateRepository(doer, owner, opts)
  24. if err != nil {
  25. if repo != nil {
  26. if errDelete := models.DeleteRepository(doer, owner.ID, repo.ID); errDelete != nil {
  27. log.Error("Rollback deleteRepository: %v", errDelete)
  28. }
  29. }
  30. return nil, err
  31. }
  32. notification.NotifyCreateRepository(doer, owner, repo)
  33. return repo, nil
  34. }
  35. // ForkRepository forks a repository
  36. func ForkRepository(doer, u *models.User, oldRepo *models.Repository, name, desc, alias string) (*models.Repository, error) {
  37. repo, err := repo_module.ForkRepository(doer, u, oldRepo, name, desc, alias)
  38. if err != nil {
  39. if repo != nil {
  40. if errDelete := models.DeleteRepository(doer, u.ID, repo.ID); errDelete != nil {
  41. log.Error("Rollback deleteRepository: %v", errDelete)
  42. }
  43. }
  44. return nil, err
  45. }
  46. notification.NotifyForkRepository(doer, oldRepo, repo)
  47. return repo, nil
  48. }
  49. // DeleteRepository deletes a repository for a user or organization.
  50. func DeleteRepository(doer *models.User, repo *models.Repository) error {
  51. if err := pull_service.CloseRepoBranchesPulls(doer, repo); err != nil {
  52. log.Error("CloseRepoBranchesPulls failed: %v", err)
  53. }
  54. if err := models.DeleteRepository(doer, repo.OwnerID, repo.ID); err != nil {
  55. return err
  56. }
  57. notification.NotifyDeleteRepository(doer, repo)
  58. return nil
  59. }
  60. // PushCreateRepo creates a repository when a new repository is pushed to an appropriate namespace
  61. func PushCreateRepo(authUser, owner *models.User, repoName string) (*models.Repository, error) {
  62. if !authUser.IsAdmin {
  63. if owner.IsOrganization() {
  64. if ok, err := owner.CanCreateOrgRepo(authUser.ID); err != nil {
  65. return nil, err
  66. } else if !ok {
  67. return nil, fmt.Errorf("cannot push-create repository for org")
  68. }
  69. } else if authUser.ID != owner.ID {
  70. return nil, fmt.Errorf("cannot push-create repository for another user")
  71. }
  72. }
  73. repo, err := CreateRepository(authUser, owner, models.CreateRepoOptions{
  74. Name: repoName,
  75. IsPrivate: true,
  76. })
  77. if err != nil {
  78. return nil, err
  79. }
  80. return repo, nil
  81. }
  82. func GetRecommendCourseKeyWords() ([]string, error) {
  83. url := setting.RecommentRepoAddr + "course_keywords"
  84. result, err := RecommendFromPromote(url)
  85. if err != nil {
  86. return []string{}, err
  87. }
  88. return result, err
  89. }
  90. func GetRecommendRepoFromPromote(repoMap []map[string]string) ([]map[string]interface{}, error) {
  91. resultRepo := make([]map[string]interface{}, 0)
  92. //resultRepo := make([]*models.Repository, 0)
  93. for _, record := range repoMap {
  94. repoName := record["project_url"]
  95. //log.Info("repoName=" + repoName + " tmpIndex1=" + fmt.Sprint(tmpIndex1) + " len(repoName)=" + fmt.Sprint(len(repoName)))
  96. tmpIndex := strings.Index(repoName, "/")
  97. if tmpIndex == -1 {
  98. log.Info("error repo name format.")
  99. } else {
  100. ownerName := strings.Trim(repoName[0:tmpIndex], " ")
  101. repoName := strings.Trim(repoName[tmpIndex+1:], " ")
  102. repo, err := models.GetRepositoryByOwnerAndAlias(ownerName, repoName)
  103. if err == nil {
  104. repoMap := make(map[string]interface{})
  105. repoMap["ID"] = fmt.Sprint(repo.ID)
  106. repoMap["Name"] = repo.Name
  107. repoMap["Alias"] = repo.Alias
  108. repoMap["Label"] = record["class"]
  109. repoMap["Label_en"] = record["class_en"]
  110. repoMap["OwnerName"] = repo.OwnerName
  111. repoMap["NumStars"] = repo.NumStars
  112. repoMap["NumForks"] = repo.NumForks
  113. repoMap["Description"] = repo.Description
  114. repoMap["NumWatchs"] = repo.NumWatches
  115. repoMap["Topics"] = repo.Topics
  116. repoMap["Avatar"] = repo.RelAvatarLink()
  117. resultRepo = append(resultRepo, repoMap)
  118. } else {
  119. log.Info("query repo error," + err.Error())
  120. }
  121. }
  122. }
  123. return resultRepo, nil
  124. }
  125. func RecommendContentFromPromote(url string) (string, error) {
  126. defer func() {
  127. if err := recover(); err != nil {
  128. log.Info("not error.", err)
  129. return
  130. }
  131. }()
  132. resp, err := http.Get(url)
  133. if err != nil || resp.StatusCode != 200 {
  134. log.Info("Get organizations url error=" + err.Error())
  135. return "", err
  136. }
  137. bytes, err := ioutil.ReadAll(resp.Body)
  138. resp.Body.Close()
  139. if err != nil {
  140. log.Info("Get organizations url error=" + err.Error())
  141. return "", err
  142. }
  143. allLineStr := string(bytes)
  144. return allLineStr, nil
  145. }
  146. func RecommendFromPromote(url string) ([]string, error) {
  147. defer func() {
  148. if err := recover(); err != nil {
  149. log.Info("not error.", err)
  150. return
  151. }
  152. }()
  153. resp, err := http.Get(url)
  154. if err != nil || resp.StatusCode != 200 {
  155. log.Info("Get organizations url error=" + err.Error())
  156. return nil, err
  157. }
  158. bytes, err := ioutil.ReadAll(resp.Body)
  159. resp.Body.Close()
  160. if err != nil {
  161. log.Info("Get organizations url error=" + err.Error())
  162. return nil, err
  163. }
  164. allLineStr := string(bytes)
  165. lines := strings.Split(allLineStr, "\n")
  166. result := make([]string, len(lines))
  167. for i, line := range lines {
  168. log.Info("i=" + fmt.Sprint(i) + " line=" + line)
  169. result[i] = strings.Trim(line, " ")
  170. }
  171. return result, nil
  172. }
  173. func CheckPushSizeLimit4Web(repo *models.Repository, fileIds []string) error {
  174. if err := CheckRepoNumOnceLimit(len(fileIds)); err != nil {
  175. return err
  176. }
  177. totalSize, err := CountUploadFileSizeByIds(fileIds)
  178. if err != nil {
  179. return UploadFileInvalidErr{}
  180. }
  181. if err := CheckRepoTotalSizeLimit(repo, totalSize); err != nil {
  182. return err
  183. }
  184. return nil
  185. }
  186. func CheckPushSizeLimit4Http(repo *models.Repository, uploadFileSize int64) error {
  187. if err := CheckRepoOnceTotalSizeLimit(uploadFileSize); err != nil {
  188. return err
  189. }
  190. if err := CheckRepoTotalSizeLimit(repo, uploadFileSize); err != nil {
  191. return err
  192. }
  193. return nil
  194. }
  195. func CheckRepoTotalSizeLimit(repo *models.Repository, uploadFileSize int64) error {
  196. if repo.Size+uploadFileSize > setting.Repository.RepoMaxSize*1024*1024 {
  197. return RepoTooLargeErr{}
  198. }
  199. return nil
  200. }
  201. func CheckRepoOnceTotalSizeLimit(uploadFileSize int64) error {
  202. if uploadFileSize > setting.Repository.Upload.TotalMaxSize*1024*1024 {
  203. return UploadFileTooLargeErr{}
  204. }
  205. return nil
  206. }
  207. func CheckRepoNumOnceLimit(uploadFileNum int) error {
  208. if uploadFileNum > setting.Repository.Upload.MaxFiles {
  209. return UploadFileTooMuchErr{}
  210. }
  211. return nil
  212. }
  213. func CountUploadFileSizeByIds(fileIds []string) (int64, error) {
  214. if len(fileIds) == 0 {
  215. return 0, nil
  216. }
  217. uploads, err := models.GetUploadsByUUIDs(fileIds)
  218. if err != nil {
  219. return 0, fmt.Errorf("CountUploadFileSizeByIds error [uuids: %v]: %v", fileIds, err)
  220. }
  221. var totalSize int64
  222. for _, upload := range uploads {
  223. size, err := GetUploadFileSize(upload)
  224. if err != nil {
  225. return 0, err
  226. }
  227. totalSize += size
  228. }
  229. return totalSize, nil
  230. }
  231. func GetUploadFileSize(upload *models.Upload) (int64, error) {
  232. info, err := os.Lstat(upload.LocalPath())
  233. if err != nil {
  234. return 0, err
  235. }
  236. return info.Size(), nil
  237. }
  238. type RepoTooLargeErr struct {
  239. }
  240. func (RepoTooLargeErr) Error() string {
  241. return fmt.Sprintf("Repository can not exceed %d MB. Please remove some unnecessary files and try again", setting.Repository.RepoMaxSize)
  242. }
  243. func IsRepoTooLargeErr(err error) bool {
  244. _, ok := err.(RepoTooLargeErr)
  245. return ok
  246. }
  247. type UploadFileTooLargeErr struct {
  248. }
  249. func (UploadFileTooLargeErr) Error() string {
  250. return fmt.Sprintf("Upload files can not exceed %d MB at a time", setting.Repository.Upload.TotalMaxSize)
  251. }
  252. func IsUploadFileTooLargeErr(err error) bool {
  253. _, ok := err.(UploadFileTooLargeErr)
  254. return ok
  255. }
  256. type RepoFileTooLargeErr struct {
  257. }
  258. func (RepoFileTooLargeErr) Error() string {
  259. return "repository file is too large"
  260. }
  261. func IsRepoFileTooLargeErr(err error) bool {
  262. _, ok := err.(RepoFileTooLargeErr)
  263. return ok
  264. }
  265. type UploadFileTooMuchErr struct {
  266. }
  267. func (UploadFileTooMuchErr) Error() string {
  268. return "upload files are too lmuch"
  269. }
  270. func IsUploadFileTooMuchErr(err error) bool {
  271. _, ok := err.(UploadFileTooMuchErr)
  272. return ok
  273. }
  274. type UploadFileInvalidErr struct {
  275. }
  276. func (UploadFileInvalidErr) Error() string {
  277. return "upload files are invalid"
  278. }
  279. func IsUploadFileInvalidErr(err error) bool {
  280. _, ok := err.(UploadFileInvalidErr)
  281. return ok
  282. }
  283. func IncreaseRepoDatasetNum(datasetID int64, engines ...*xorm.Engine) error {
  284. dataset, err := models.GetDatasetByID(datasetID)
  285. if err != nil {
  286. return err
  287. }
  288. return models.OperateRepoDatasetNum(dataset.RepoID, 1, engines...)
  289. }
  290. func IncreaseRepoModelNum(repoId int64, engines ...*xorm.Engine) error {
  291. return models.OperateRepoModelNum(repoId, 1, engines...)
  292. }
  293. func ResetRepoModelNum(repoId int64) error {
  294. return models.ResetRepoModelNum(repoId)
  295. }
  296. func DecreaseRepoDatasetNum(datasetID int64, engines ...*xorm.Engine) error {
  297. dataset, err := models.GetDatasetByID(datasetID)
  298. if err != nil {
  299. return err
  300. }
  301. return models.OperateRepoDatasetNum(dataset.RepoID, -1, engines...)
  302. }
  303. func DecreaseRepoModelNum(repoId int64, engines ...*xorm.Engine) error {
  304. return models.OperateRepoModelNum(repoId, -1, engines...)
  305. }
  306. func GetDefaultBranchName(repo *models.Repository) string {
  307. gitRepo, err := git.OpenRepository(repo.RepoPath())
  308. if err != nil {
  309. return ""
  310. }
  311. defer gitRepo.Close()
  312. if len(repo.DefaultBranch) > 0 && gitRepo.IsBranchExist(repo.DefaultBranch) {
  313. return repo.DefaultBranch
  314. }
  315. brs, _, err := gitRepo.GetBranches(0, 0)
  316. if len(brs) > 0 {
  317. return brs[0]
  318. }
  319. return ""
  320. }