![Demo](http://gogs.qiniudn.com/gogs_demo.gif) | ![Demo](http://gogs.qiniudn.com/gogs_demo.gif) | ||||
##### Current version: 0.5.13 Beta | |||||
##### Current version: 0.5.15 Beta | |||||
### NOTICES | ### NOTICES | ||||
- Due to testing purpose, data of [try.gogs.io](https://try.gogs.io) has been reset in **Jan 28, 2015** and will reset multiple times after. Please do **NOT** put your important data on the site. | - Due to testing purpose, data of [try.gogs.io](https://try.gogs.io) has been reset in **Jan 28, 2015** and will reset multiple times after. Please do **NOT** put your important data on the site. | ||||
- The demo site [try.gogs.io](https://try.gogs.io) is running under `dev` branch. | - The demo site [try.gogs.io](https://try.gogs.io) is running under `dev` branch. | ||||
- You **MUST** read [CONTRIBUTING.md](CONTRIBUTING.md) before you start filing a issue or making a Pull Request. | |||||
- You **MUST** read [CONTRIBUTING.md](CONTRIBUTING.md) before you start filing an issue or making a Pull Request. | |||||
- If you think there are vulnerabilities in the project, please talk privately to **u@gogs.io**. Thanks! | - If you think there are vulnerabilities in the project, please talk privately to **u@gogs.io**. Thanks! | ||||
#### Other language version | #### Other language version | ||||
## System Requirements | ## System Requirements | ||||
- A cheap Raspberry Pi is powerful enough for basic functionality. | - A cheap Raspberry Pi is powerful enough for basic functionality. | ||||
- At least 4 CPU cores and 1GB RAM would be the baseline for teamwork. | |||||
- At least 2 CPU cores and 1GB RAM would be the baseline for teamwork. | |||||
## Installation | ## Installation | ||||
![Demo](http://gogs.qiniudn.com/gogs_demo.gif) | ![Demo](http://gogs.qiniudn.com/gogs_demo.gif) | ||||
##### 当前版本:0.5.13 Beta | |||||
##### 当前版本:0.5.15 Beta | |||||
## 开发目的 | ## 开发目的 | ||||
## 系统要求 | ## 系统要求 | ||||
- 最低的系统硬件要求为一个廉价的树莓派 | - 最低的系统硬件要求为一个廉价的树莓派 | ||||
- 如果用于团队项目,建议使用 4 核 CPU 及 1GB 内存 | |||||
- 如果用于团队项目,建议使用 2 核 CPU 及 1GB 内存 | |||||
## 安装部署 | ## 安装部署 | ||||
## 授权许可 | ## 授权许可 | ||||
本项目采用 MIT 开源授权许可证,完整的授权说明已放置在 [LICENSE](https://github.com/gogits/gogs/blob/master/LICENSE) 文件中。 | |||||
本项目采用 MIT 开源授权许可证,完整的授权说明已放置在 [LICENSE](https://github.com/gogits/gogs/blob/master/LICENSE) 文件中。 |
"github.com/gogits/gogs/modules/setting" | "github.com/gogits/gogs/modules/setting" | ||||
) | ) | ||||
const APP_VER = "0.5.15.0223 Beta" | |||||
const APP_VER = "0.5.15.0224 Beta" | |||||
func init() { | func init() { | ||||
runtime.GOMAXPROCS(runtime.NumCPU()) | runtime.GOMAXPROCS(runtime.NumCPU()) |
package models | package models | ||||
import ( | |||||
"fmt" | |||||
) | |||||
type AccessMode int | type AccessMode int | ||||
const ( | const ( | ||||
} | } | ||||
if err = team.getMembers(e); err != nil { | if err = team.getMembers(e); err != nil { | ||||
return err | |||||
return fmt.Errorf("getMembers '%d': %v", team.ID, err) | |||||
} | } | ||||
for _, u := range team.Members { | for _, u := range team.Members { | ||||
accessMap[u.Id] = maxAccessMode(accessMap[u.Id], team.Authorize) | accessMap[u.Id] = maxAccessMode(accessMap[u.Id], team.Authorize) | ||||
// Delete old accesses and insert new ones for repository. | // Delete old accesses and insert new ones for repository. | ||||
if _, err = e.Delete(&Access{RepoID: repo.Id}); err != nil { | if _, err = e.Delete(&Access{RepoID: repo.Id}); err != nil { | ||||
return err | |||||
return fmt.Errorf("delete old accesses: %v", err) | |||||
} else if _, err = e.Insert(newAccesses); err != nil { | } else if _, err = e.Insert(newAccesses); err != nil { | ||||
return err | |||||
return fmt.Errorf("insert new accesses: %v", err) | |||||
} | } | ||||
return nil | return nil |
// IsOrgMember returns true if given user is member of organization. | // IsOrgMember returns true if given user is member of organization. | ||||
func (org *User) IsOrgMember(uid int64) bool { | func (org *User) IsOrgMember(uid int64) bool { | ||||
return IsOrganizationMember(org.Id, uid) | |||||
return org.IsOrganization() && IsOrganizationMember(org.Id, uid) | |||||
} | } | ||||
func (org *User) getTeam(e Engine, name string) (*Team, error) { | func (org *User) getTeam(e Engine, name string) (*Team, error) { | ||||
t.NumRepos++ | t.NumRepos++ | ||||
if _, err = e.Id(t.ID).AllCols().Update(t); err != nil { | if _, err = e.Id(t.ID).AllCols().Update(t); err != nil { | ||||
return err | |||||
return fmt.Errorf("update team: %v", err) | |||||
} | } | ||||
if err = repo.recalculateAccesses(e); err != nil { | if err = repo.recalculateAccesses(e); err != nil { | ||||
return err | |||||
return fmt.Errorf("recalculateAccesses: %v", err) | |||||
} | } | ||||
if err = t.getMembers(e); err != nil { | if err = t.getMembers(e); err != nil { | ||||
return fmt.Errorf("get team members: %v", err) | |||||
return fmt.Errorf("getMembers: %v", err) | |||||
} | } | ||||
for _, u := range t.Members { | for _, u := range t.Members { | ||||
if err = watchRepo(e, u.Id, repo.Id, true); err != nil { | if err = watchRepo(e, u.Id, repo.Id, true); err != nil { | ||||
return err | |||||
return fmt.Errorf("watchRepo: %v", err) | |||||
} | } | ||||
} | } | ||||
return nil | return nil | ||||
return isTeamMember(x, orgID, teamID, uid) | return isTeamMember(x, orgID, teamID, uid) | ||||
} | } | ||||
func getTeamMembers(e Engine, teamID int64) ([]*User, error) { | |||||
us := make([]*User, 0, 10) | |||||
err := e.Sql("SELECT * FROM `user` JOIN `team_user` ON `team_user`.`team_id` = ? AND `team_user`.`uid` = `user`.`id`", teamID).Find(&us) | |||||
return us, err | |||||
func getTeamMembers(e Engine, teamID int64) (_ []*User, err error) { | |||||
teamUsers := make([]*TeamUser, 0, 10) | |||||
if err = e.Where("team_id=?", teamID).Find(&teamUsers); err != nil { | |||||
return nil, fmt.Errorf("get team-users: %v", err) | |||||
} | |||||
members := make([]*User, 0, len(teamUsers)) | |||||
for i := range teamUsers { | |||||
member := new(User) | |||||
if _, err = e.Id(teamUsers[i].Uid).Get(member); err != nil { | |||||
return nil, fmt.Errorf("get user '%d': %v", teamUsers[i].Uid, err) | |||||
} | |||||
members = append(members, member) | |||||
} | |||||
return members, nil | |||||
} | } | ||||
// GetTeamMembers returns all members in given team of organization. | // GetTeamMembers returns all members in given team of organization. |
} | } | ||||
// IsRepositoryExist returns true if the repository with given name under user has already existed. | // IsRepositoryExist returns true if the repository with given name under user has already existed. | ||||
func IsRepositoryExist(u *User, repoName string) (bool, error) { | |||||
repo := Repository{OwnerId: u.Id} | |||||
has, err := x.Where("lower_name = ?", strings.ToLower(repoName)).Get(&repo) | |||||
if err != nil { | |||||
return has, err | |||||
} else if !has { | |||||
return false, nil | |||||
} | |||||
return com.IsDir(RepoPath(u.Name, repoName)), nil | |||||
func IsRepositoryExist(u *User, repoName string) bool { | |||||
has, _ := x.Get(&Repository{ | |||||
OwnerId: u.Id, | |||||
LowerName: strings.ToLower(repoName), | |||||
}) | |||||
return has && com.IsDir(RepoPath(u.Name, repoName)) | |||||
} | } | ||||
// CloneLink represents different types of clone URLs of repository. | // CloneLink represents different types of clone URLs of repository. | ||||
} | } | ||||
// CreateRepository creates a repository for given user or organization. | // CreateRepository creates a repository for given user or organization. | ||||
func CreateRepository(u *User, name, desc, lang, license string, isPrivate, isMirror, initReadme bool) (*Repository, error) { | |||||
func CreateRepository(u *User, name, desc, lang, license string, isPrivate, isMirror, initReadme bool) (_ *Repository, err error) { | |||||
if !IsLegalName(name) { | if !IsLegalName(name) { | ||||
return nil, ErrRepoNameIllegal | return nil, ErrRepoNameIllegal | ||||
} | } | ||||
isExist, err := IsRepositoryExist(u, name) | |||||
if err != nil { | |||||
return nil, err | |||||
} else if isExist { | |||||
if IsRepositoryExist(u, name) { | |||||
return nil, ErrRepoAlreadyExist | return nil, ErrRepoAlreadyExist | ||||
} | } | ||||
func TransferOwnership(u *User, newOwnerName string, repo *Repository) error { | func TransferOwnership(u *User, newOwnerName string, repo *Repository) error { | ||||
newOwner, err := GetUserByName(newOwnerName) | newOwner, err := GetUserByName(newOwnerName) | ||||
if err != nil { | if err != nil { | ||||
return fmt.Errorf("get new owner(%s): %v", newOwnerName, err) | |||||
return fmt.Errorf("get new owner '%s': %v", newOwnerName, err) | |||||
} | } | ||||
// Check if new owner has repository with same name. | // Check if new owner has repository with same name. | ||||
has, err := IsRepositoryExist(newOwner, repo.Name) | |||||
if err != nil { | |||||
return err | |||||
} else if has { | |||||
if IsRepositoryExist(newOwner, repo.Name) { | |||||
return ErrRepoAlreadyExist | return ErrRepoAlreadyExist | ||||
} | } | ||||
sess := x.NewSession() | sess := x.NewSession() | ||||
defer sessionRelease(sess) | defer sessionRelease(sess) | ||||
if err = sess.Begin(); err != nil { | if err = sess.Begin(); err != nil { | ||||
return err | |||||
return fmt.Errorf("sess.Begin: %v", err) | |||||
} | } | ||||
owner := repo.Owner | owner := repo.Owner | ||||
// Update repository. | |||||
// Note: we have to set value here to make sure recalculate accesses is based on | |||||
// new owner. | |||||
repo.OwnerId = newOwner.Id | repo.OwnerId = newOwner.Id | ||||
repo.Owner = newOwner | repo.Owner = newOwner | ||||
// Update repository. | |||||
if _, err := sess.Id(repo.Id).Update(repo); err != nil { | if _, err := sess.Id(repo.Id).Update(repo); err != nil { | ||||
return err | |||||
return fmt.Errorf("update owner: %v", err) | |||||
} | } | ||||
// Remove redundant collaborators | |||||
// Remove redundant collaborators. | |||||
collaborators, err := repo.GetCollaborators() | collaborators, err := repo.GetCollaborators() | ||||
if err != nil { | if err != nil { | ||||
return err | |||||
return fmt.Errorf("GetCollaborators: %v", err) | |||||
} | } | ||||
// Dummy object. | |||||
collaboration := &Collaboration{RepoID: repo.Id} | |||||
for _, c := range collaborators { | for _, c := range collaborators { | ||||
collaboration.UserID = c.Id | |||||
if c.Id == newOwner.Id || newOwner.IsOrgMember(c.Id) { | if c.Id == newOwner.Id || newOwner.IsOrgMember(c.Id) { | ||||
if _, err = sess.Delete(&Collaboration{RepoID: repo.Id, UserID: c.Id}); err != nil { | |||||
return err | |||||
if _, err = sess.Delete(collaboration); err != nil { | |||||
return fmt.Errorf("remove collaborator '%d': %v", c.Id, err) | |||||
} | } | ||||
} | } | ||||
} | } | ||||
if newOwner.IsOrganization() { | if newOwner.IsOrganization() { | ||||
// Update owner team info and count. | |||||
t, err := newOwner.GetOwnerTeam() | t, err := newOwner.GetOwnerTeam() | ||||
if err != nil { | if err != nil { | ||||
return err | |||||
return fmt.Errorf("GetOwnerTeam: %v", err) | |||||
} else if err = t.addRepository(sess, repo); err != nil { | } else if err = t.addRepository(sess, repo); err != nil { | ||||
return err | |||||
return fmt.Errorf("add to owner team: %v", err) | |||||
} | |||||
} else { | |||||
// Organization called this in addRepository method. | |||||
if err = repo.recalculateAccesses(sess); err != nil { | |||||
return fmt.Errorf("recalculateAccesses: %v", err) | |||||
} | } | ||||
} | } | ||||
// Update user repository number. | |||||
if _, err = sess.Exec("UPDATE `user` SET num_repos = num_repos + 1 WHERE id = ?", newOwner.Id); err != nil { | |||||
return err | |||||
} else if _, err = sess.Exec("UPDATE `user` SET num_repos = num_repos - 1 WHERE id = ?", owner.Id); err != nil { | |||||
return err | |||||
} else if err = repo.recalculateAccesses(sess); err != nil { | |||||
return err | |||||
} else if err = watchRepo(sess, newOwner.Id, repo.Id, true); err != nil { | |||||
return err | |||||
// Update repository count. | |||||
if _, err = sess.Exec("UPDATE `user` SET num_repos=num_repos+1 WHERE id=?", newOwner.Id); err != nil { | |||||
return fmt.Errorf("increase new owner repository count: %v", err) | |||||
} else if _, err = sess.Exec("UPDATE `user` SET num_repos=num_repos-1 WHERE id=?", owner.Id); err != nil { | |||||
return fmt.Errorf("decrease old owner repository count: %v", err) | |||||
} | |||||
if err = watchRepo(sess, newOwner.Id, repo.Id, true); err != nil { | |||||
return fmt.Errorf("watchRepo: %v", err) | |||||
} else if err = transferRepoAction(sess, u, owner, newOwner, repo); err != nil { | } else if err = transferRepoAction(sess, u, owner, newOwner, repo); err != nil { | ||||
return err | |||||
return fmt.Errorf("transferRepoAction: %v", err) | |||||
} | } | ||||
// Change repository directory name. | // Change repository directory name. | ||||
if err = os.Rename(RepoPath(owner.Name, repo.Name), RepoPath(newOwner.Name, repo.Name)); err != nil { | if err = os.Rename(RepoPath(owner.Name, repo.Name), RepoPath(newOwner.Name, repo.Name)); err != nil { | ||||
return err | |||||
return fmt.Errorf("rename directory: %v", err) | |||||
} | } | ||||
return sess.Commit() | return sess.Commit() | ||||
// \___ / \____/|__| |__|_ \ | // \___ / \____/|__| |__|_ \ | ||||
// \/ \/ | // \/ \/ | ||||
func ForkRepository(u *User, oldRepo *Repository, name, desc string) (*Repository, error) { | |||||
isExist, err := IsRepositoryExist(u, name) | |||||
if err != nil { | |||||
return nil, err | |||||
} else if isExist { | |||||
func ForkRepository(u *User, oldRepo *Repository, name, desc string) (_ *Repository, err error) { | |||||
if IsRepositoryExist(u, name) { | |||||
return nil, ErrRepoAlreadyExist | return nil, ErrRepoAlreadyExist | ||||
} | } | ||||
newRepoName := form.RepoName | newRepoName := form.RepoName | ||||
// Check if repository name has been changed. | // Check if repository name has been changed. | ||||
if ctx.Repo.Repository.Name != newRepoName { | if ctx.Repo.Repository.Name != newRepoName { | ||||
isExist, err := models.IsRepositoryExist(ctx.Repo.Owner, newRepoName) | |||||
if err != nil { | |||||
ctx.Handle(500, "IsRepositoryExist", err) | |||||
return | |||||
} else if isExist { | |||||
if models.IsRepositoryExist(ctx.Repo.Owner, newRepoName) { | |||||
ctx.Data["Err_RepoName"] = true | ctx.Data["Err_RepoName"] = true | ||||
ctx.RenderWithErr(ctx.Tr("form.repo_name_been_taken"), SETTINGS_OPTIONS, nil) | ctx.RenderWithErr(ctx.Tr("form.repo_name_been_taken"), SETTINGS_OPTIONS, nil) | ||||
return | return | ||||
} else if err = models.ChangeRepositoryName(ctx.Repo.Owner.Name, ctx.Repo.Repository.Name, newRepoName); err != nil { | |||||
} else if err := models.ChangeRepositoryName(ctx.Repo.Owner.Name, ctx.Repo.Repository.Name, newRepoName); err != nil { | |||||
if err == models.ErrRepoNameIllegal { | if err == models.ErrRepoNameIllegal { | ||||
ctx.Data["Err_RepoName"] = true | ctx.Data["Err_RepoName"] = true | ||||
ctx.RenderWithErr(ctx.Tr("form.illegal_repo_name"), SETTINGS_OPTIONS, nil) | ctx.RenderWithErr(ctx.Tr("form.illegal_repo_name"), SETTINGS_OPTIONS, nil) |
0.5.15.0223 Beta | |||||
0.5.15.0224 Beta |