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.9 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
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
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  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"
  11. "path/filepath"
  12. "strings"
  13. "time"
  14. "unicode/utf8"
  15. "github.com/Unknwon/com"
  16. git "github.com/libgit2/git2go"
  17. "github.com/gogits/gogs/modules/base"
  18. "github.com/gogits/gogs/modules/log"
  19. )
  20. type Repository struct {
  21. Id int64
  22. OwnerId int64 `xorm:"unique(s)"`
  23. ForkId int64
  24. LowerName string `xorm:"unique(s) index not null"`
  25. Name string `xorm:"index not null"`
  26. Description string
  27. Private bool
  28. NumWatchs int
  29. NumStars int
  30. NumForks int
  31. Created time.Time `xorm:"created"`
  32. Updated time.Time `xorm:"updated"`
  33. }
  34. type Star struct {
  35. Id int64
  36. RepoId int64
  37. UserId int64
  38. Created time.Time `xorm:"created"`
  39. }
  40. var (
  41. LanguageIgns, Licenses []string
  42. )
  43. var (
  44. ErrRepoAlreadyExist = errors.New("Repository already exist")
  45. ErrRepoNotExist = errors.New("Repository does not exist")
  46. )
  47. func init() {
  48. LanguageIgns = strings.Split(base.Cfg.MustValue("repository", "LANG_IGNS"), "|")
  49. Licenses = strings.Split(base.Cfg.MustValue("repository", "LICENSES"), "|")
  50. }
  51. // check if repository is exist
  52. func IsRepositoryExist(user *User, repoName string) (bool, error) {
  53. repo := Repository{OwnerId: user.Id}
  54. has, err := orm.Where("lower_name = ?", strings.ToLower(repoName)).Get(&repo)
  55. if err != nil {
  56. return has, err
  57. }
  58. s, err := os.Stat(RepoPath(user.Name, repoName))
  59. if err != nil {
  60. return false, nil
  61. }
  62. return s.IsDir(), nil
  63. }
  64. // CreateRepository creates a repository for given user or orgnaziation.
  65. func CreateRepository(user *User, repoName, desc, repoLang, license string, private bool, initReadme bool) (*Repository, error) {
  66. isExist, err := IsRepositoryExist(user, repoName)
  67. if err != nil {
  68. return nil, err
  69. } else if isExist {
  70. return nil, ErrRepoAlreadyExist
  71. }
  72. repo := &Repository{
  73. OwnerId: user.Id,
  74. Name: repoName,
  75. LowerName: strings.ToLower(repoName),
  76. Description: desc,
  77. Private: private,
  78. }
  79. f := RepoPath(user.Name, repoName)
  80. if err = initRepository(f, user, repo, initReadme, repoLang, license); err != nil {
  81. return nil, err
  82. }
  83. session := orm.NewSession()
  84. defer session.Close()
  85. session.Begin()
  86. if _, err = session.Insert(repo); err != nil {
  87. if err2 := os.RemoveAll(f); err2 != nil {
  88. return nil, errors.New(fmt.Sprintf(
  89. "delete repo directory %s/%s failed", user.Name, repoName))
  90. }
  91. session.Rollback()
  92. return nil, err
  93. }
  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. Id *git.Oid
  234. Type int
  235. Name string
  236. Path string
  237. Message string
  238. Created time.Time
  239. }
  240. func (f *RepoFile) IsFile() bool {
  241. return f.Type == git.FilemodeBlob || f.Type == git.FilemodeBlobExecutable
  242. }
  243. func (f *RepoFile) IsDir() bool {
  244. return f.Type == git.FilemodeTree
  245. }
  246. func GetReposFiles(userName, reposName, branchName, rpath string) ([]*RepoFile, error) {
  247. f := RepoPath(userName, reposName)
  248. repo, err := git.OpenRepository(f)
  249. if err != nil {
  250. return nil, err
  251. }
  252. obj, err := repo.RevparseSingle("HEAD")
  253. if err != nil {
  254. return nil, err
  255. }
  256. lastCommit := obj.(*git.Commit)
  257. var repofiles []*RepoFile
  258. tree, err := lastCommit.Tree()
  259. if err != nil {
  260. return nil, err
  261. }
  262. //var i uint64 = 0
  263. if rpath != "" {
  264. rpath = rpath + "/"
  265. }
  266. fmt.Println("...", rpath, "...")
  267. tree.Walk(func(dirname string, entry *git.TreeEntry) int {
  268. if dirname == rpath {
  269. fmt.Println("====", dirname, "==", entry.Name)
  270. repofiles = append(repofiles, &RepoFile{
  271. entry.Id,
  272. entry.Filemode,
  273. entry.Name,
  274. path.Join(dirname, entry.Name),
  275. lastCommit.Message(),
  276. lastCommit.Committer().When,
  277. })
  278. }
  279. return 0
  280. })
  281. /*for ; i < tree.EntryCount(); i++ {
  282. entry := tree.EntryByIndex(i)
  283. repofiles = append(repofiles, &RepoFile{
  284. entry.Id,
  285. entry.Filemode,
  286. entry.Name,
  287. lastCommit.Message(),
  288. lastCommit.Committer().When,
  289. })
  290. }*/
  291. return repofiles, nil
  292. }
  293. func StarReposiory(user *User, repoName string) error {
  294. return nil
  295. }
  296. func UnStarRepository() {
  297. }
  298. func WatchRepository() {
  299. }
  300. func UnWatchRepository() {
  301. }
  302. func ForkRepository(reposName string, userId int64) {
  303. }
  304. func RepoPath(userName, repoName string) string {
  305. return filepath.Join(UserPath(userName), repoName+".git")
  306. }
  307. // DeleteRepository deletes a repository for a user or orgnaztion.
  308. func DeleteRepository(userId, repoId int64, userName string) (err error) {
  309. repo := &Repository{Id: repoId, OwnerId: userId}
  310. has, err := orm.Get(repo)
  311. if err != nil {
  312. return err
  313. } else if !has {
  314. return ErrRepoNotExist
  315. }
  316. session := orm.NewSession()
  317. if _, err = session.Delete(&Repository{Id: repoId}); err != nil {
  318. session.Rollback()
  319. return err
  320. }
  321. if _, err := session.Delete(&Access{UserName: userName, RepoName: repo.Name}); err != nil {
  322. session.Rollback()
  323. return err
  324. }
  325. if _, err = session.Exec("update user set num_repos = num_repos - 1 where id = ?", userId); err != nil {
  326. session.Rollback()
  327. return err
  328. }
  329. if err = session.Commit(); err != nil {
  330. session.Rollback()
  331. return err
  332. }
  333. if err = os.RemoveAll(RepoPath(userName, repo.Name)); err != nil {
  334. // TODO: log and delete manully
  335. log.Error("delete repo %s/%s failed", userName, repo.Name)
  336. return err
  337. }
  338. return nil
  339. }