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.

repo.go 8.0 kB

11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. // Copyright 2014 The Gogs 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 models
  5. import (
  6. "errors"
  7. "fmt"
  8. "io/ioutil"
  9. "os"
  10. "path/filepath"
  11. "strings"
  12. "time"
  13. "unicode/utf8"
  14. "github.com/Unknwon/com"
  15. git "github.com/libgit2/git2go"
  16. "github.com/gogits/gogs/modules/base"
  17. "github.com/gogits/gogs/modules/log"
  18. )
  19. type Repository struct {
  20. Id int64
  21. OwnerId int64 `xorm:"unique(s)"`
  22. ForkId int64
  23. LowerName string `xorm:"unique(s) index not null"`
  24. Name string `xorm:"index not null"`
  25. Description string
  26. Private bool
  27. NumWatchs int
  28. NumStars int
  29. NumForks int
  30. Created time.Time `xorm:"created"`
  31. Updated time.Time `xorm:"updated"`
  32. }
  33. type Star struct {
  34. Id int64
  35. RepoId int64
  36. UserId int64
  37. Created time.Time `xorm:"created"`
  38. }
  39. var (
  40. LanguageIgns, Licenses []string
  41. )
  42. var (
  43. ErrRepoAlreadyExist = errors.New("Repository already exist")
  44. ErrRepoNotExist = errors.New("Repository does not exist")
  45. )
  46. func init() {
  47. LanguageIgns = strings.Split(base.Cfg.MustValue("repository", "LANG_IGNS"), "|")
  48. Licenses = strings.Split(base.Cfg.MustValue("repository", "LICENSES"), "|")
  49. }
  50. // check if repository is exist
  51. func IsRepositoryExist(user *User, repoName string) (bool, error) {
  52. repo := Repository{OwnerId: user.Id}
  53. has, err := orm.Where("lower_name = ?", strings.ToLower(repoName)).Get(&repo)
  54. if err != nil {
  55. return has, err
  56. }
  57. s, err := os.Stat(RepoPath(user.Name, repoName))
  58. if err != nil {
  59. return false, nil
  60. }
  61. return s.IsDir(), nil
  62. }
  63. // CreateRepository creates a repository for given user or orgnaziation.
  64. func CreateRepository(user *User, repoName, desc, repoLang, license string, private bool, initReadme bool) (*Repository, error) {
  65. isExist, err := IsRepositoryExist(user, repoName)
  66. if err != nil {
  67. return nil, err
  68. } else if isExist {
  69. return nil, ErrRepoAlreadyExist
  70. }
  71. repo := &Repository{
  72. OwnerId: user.Id,
  73. Name: repoName,
  74. LowerName: strings.ToLower(repoName),
  75. Description: desc,
  76. Private: private,
  77. }
  78. f := RepoPath(user.Name, repoName)
  79. if err = initRepository(f, user, repo, initReadme, repoLang, license); err != nil {
  80. return nil, err
  81. }
  82. session := orm.NewSession()
  83. defer session.Close()
  84. session.Begin()
  85. if _, err = session.Insert(repo); err != nil {
  86. if err2 := os.RemoveAll(f); err2 != nil {
  87. return nil, errors.New(fmt.Sprintf(
  88. "delete repo directory %s/%s failed", user.Name, repoName))
  89. }
  90. session.Rollback()
  91. return nil, err
  92. }
  93. // TODO: RemoveAll may fail due to not root access.
  94. access := Access{
  95. UserName: user.Name,
  96. RepoName: repo.Name,
  97. Mode: AU_WRITABLE,
  98. }
  99. if _, err = session.Insert(&access); err != nil {
  100. session.Rollback()
  101. if err2 := os.RemoveAll(f); err2 != nil {
  102. return nil, errors.New(fmt.Sprintf(
  103. "delete repo directory %s/%s failed", user.Name, repoName))
  104. }
  105. return nil, err
  106. }
  107. if _, err = session.Exec("update user set num_repos = num_repos + 1 where id = ?", user.Id); err != nil {
  108. session.Rollback()
  109. if err2 := os.RemoveAll(f); err2 != nil {
  110. return nil, errors.New(fmt.Sprintf(
  111. "delete repo directory %s/%s failed", user.Name, repoName))
  112. }
  113. return nil, err
  114. }
  115. if err = session.Commit(); err != nil {
  116. session.Rollback()
  117. if err2 := os.RemoveAll(f); err2 != nil {
  118. return nil, errors.New(fmt.Sprintf(
  119. "delete repo directory %s/%s failed", user.Name, repoName))
  120. }
  121. return nil, err
  122. }
  123. return repo, NewRepoAction(user, repo)
  124. }
  125. // InitRepository initializes README and .gitignore if needed.
  126. func initRepository(f string, user *User, repo *Repository, initReadme bool, repoLang, license string) error {
  127. fileName := map[string]string{}
  128. if initReadme {
  129. fileName["readme"] = "README.md"
  130. }
  131. if repoLang != "" {
  132. fileName["gitign"] = ".gitignore"
  133. }
  134. if license != "" {
  135. fileName["license"] = "LICENSE"
  136. }
  137. workdir := filepath.Join(os.TempDir(), fmt.Sprintf("%d", time.Now().Nanosecond()))
  138. os.MkdirAll(workdir, os.ModePerm)
  139. sig := user.NewGitSig()
  140. // README
  141. if initReadme {
  142. defaultReadme := repo.Name + "\n" + strings.Repeat("=",
  143. utf8.RuneCountInString(repo.Name)) + "\n\n" + repo.Description
  144. if err := ioutil.WriteFile(filepath.Join(workdir, fileName["readme"]),
  145. []byte(defaultReadme), 0644); err != nil {
  146. return err
  147. }
  148. }
  149. if repoLang != "" {
  150. // .gitignore
  151. filePath := "conf/gitignore/" + repoLang
  152. if com.IsFile(filePath) {
  153. if _, err := com.Copy(filePath,
  154. filepath.Join(workdir, fileName["gitign"])); err != nil {
  155. return err
  156. }
  157. }
  158. }
  159. if license != "" {
  160. // LICENSE
  161. filePath := "conf/license/" + license
  162. if com.IsFile(filePath) {
  163. if _, err := com.Copy(filePath,
  164. filepath.Join(workdir, fileName["license"])); err != nil {
  165. return err
  166. }
  167. }
  168. }
  169. rp, err := git.InitRepository(f, true)
  170. if err != nil {
  171. return err
  172. }
  173. rp.SetWorkdir(workdir, false)
  174. idx, err := rp.Index()
  175. if err != nil {
  176. return err
  177. }
  178. for _, name := range fileName {
  179. if err = idx.AddByPath(name); err != nil {
  180. return err
  181. }
  182. }
  183. treeId, err := idx.WriteTree()
  184. if err != nil {
  185. return err
  186. }
  187. message := "Init commit"
  188. tree, err := rp.LookupTree(treeId)
  189. if err != nil {
  190. return err
  191. }
  192. if _, err = rp.CreateCommit("HEAD", sig, sig, message, tree); err != nil {
  193. return err
  194. }
  195. return nil
  196. }
  197. func GetRepositoryByName(user *User, repoName string) (*Repository, error) {
  198. repo := &Repository{
  199. OwnerId: user.Id,
  200. LowerName: strings.ToLower(repoName),
  201. }
  202. has, err := orm.Get(repo)
  203. if err != nil {
  204. return nil, err
  205. } else if !has {
  206. return nil, ErrRepoNotExist
  207. }
  208. return repo, err
  209. }
  210. func GetRepositoryById(id int64) (repo *Repository, err error) {
  211. has, err := orm.Id(id).Get(repo)
  212. if err != nil {
  213. return nil, err
  214. } else if !has {
  215. return nil, ErrRepoNotExist
  216. }
  217. return repo, err
  218. }
  219. // GetRepositories returns the list of repositories of given user.
  220. func GetRepositories(user *User) ([]Repository, error) {
  221. repos := make([]Repository, 0, 10)
  222. err := orm.Find(&repos, &Repository{OwnerId: user.Id})
  223. return repos, err
  224. }
  225. func GetRepositoryCount(user *User) (int64, error) {
  226. return orm.Count(&Repository{OwnerId: user.Id})
  227. }
  228. const (
  229. RFile = iota + 1
  230. RDir
  231. )
  232. type RepoFile struct {
  233. Type int
  234. Name string
  235. Created time.Time
  236. }
  237. func GetReposFiles(userName, reposName, treeName, rpath string) ([]RepoFile, error) {
  238. f := RepoPath(userName, reposName)
  239. repo, err := git.OpenRepository(f)
  240. if err != nil {
  241. return nil, err
  242. }
  243. obj, err := repo.RevparseSingle("HEAD")
  244. if err != nil {
  245. return nil, err
  246. }
  247. lastCommit := obj.(*git.Commit)
  248. var repofiles []RepoFile
  249. tree, err := lastCommit.Tree()
  250. if err != nil {
  251. return nil, err
  252. }
  253. var i uint64 = 0
  254. for ; i < tree.EntryCount(); i++ {
  255. entry := tree.EntryByIndex(i)
  256. repofiles = append(repofiles, RepoFile{
  257. entry.Filemode,
  258. entry.Name,
  259. time.Now(),
  260. })
  261. }
  262. return repofiles, nil
  263. }
  264. func StarReposiory(user *User, repoName string) error {
  265. return nil
  266. }
  267. func UnStarRepository() {
  268. }
  269. func WatchRepository() {
  270. }
  271. func UnWatchRepository() {
  272. }
  273. func ForkRepository(reposName string, userId int64) {
  274. }
  275. func RepoPath(userName, repoName string) string {
  276. return filepath.Join(UserPath(userName), repoName+".git")
  277. }
  278. // DeleteRepository deletes a repository for a user or orgnaztion.
  279. func DeleteRepository(user *User, reposName string) (err error) {
  280. session := orm.NewSession()
  281. if _, err = session.Delete(&Repository{OwnerId: user.Id, Name: reposName}); err != nil {
  282. session.Rollback()
  283. return err
  284. }
  285. if _, err = session.Exec("update user set num_repos = num_repos - 1 where id = ?", user.Id); err != nil {
  286. session.Rollback()
  287. return err
  288. }
  289. if err = session.Commit(); err != nil {
  290. session.Rollback()
  291. return err
  292. }
  293. if err = os.RemoveAll(RepoPath(user.Name, reposName)); err != nil {
  294. // TODO: log and delete manully
  295. log.Error("delete repo %s/%s failed", user.Name, reposName)
  296. return err
  297. }
  298. return nil
  299. }