Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
229 lines
6.0 KiB
229 lines
6.0 KiB
// Copyright 2014 The Gogs Authors. All rights reserved. |
|
// Use of this source code is governed by a MIT-style |
|
// license that can be found in the LICENSE file. |
|
|
|
package v1 |
|
|
|
import ( |
|
"fmt" |
|
"path" |
|
"strings" |
|
|
|
"github.com/Unknwon/com" |
|
|
|
api "github.com/gogits/go-gogs-client" |
|
|
|
"github.com/gogits/gogs/models" |
|
"github.com/gogits/gogs/modules/auth" |
|
"github.com/gogits/gogs/modules/base" |
|
"github.com/gogits/gogs/modules/log" |
|
"github.com/gogits/gogs/modules/middleware" |
|
"github.com/gogits/gogs/modules/setting" |
|
) |
|
|
|
func SearchRepos(ctx *middleware.Context) { |
|
opt := models.SearchOption{ |
|
Keyword: path.Base(ctx.Query("q")), |
|
Uid: com.StrTo(ctx.Query("uid")).MustInt64(), |
|
Limit: com.StrTo(ctx.Query("limit")).MustInt(), |
|
} |
|
if opt.Limit == 0 { |
|
opt.Limit = 10 |
|
} |
|
|
|
// Check visibility. |
|
if ctx.IsSigned && opt.Uid > 0 { |
|
if ctx.User.Id == opt.Uid { |
|
opt.Private = true |
|
} else { |
|
u, err := models.GetUserById(opt.Uid) |
|
if err != nil { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
return |
|
} |
|
if u.IsOrganization() && u.IsOrgOwner(ctx.User.Id) { |
|
opt.Private = true |
|
} |
|
// FIXME: how about collaborators? |
|
} |
|
} |
|
|
|
repos, err := models.SearchRepositoryByName(opt) |
|
if err != nil { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
return |
|
} |
|
|
|
results := make([]*api.Repository, len(repos)) |
|
for i := range repos { |
|
if err = repos[i].GetOwner(); err != nil { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
return |
|
} |
|
results[i] = &api.Repository{ |
|
Id: repos[i].Id, |
|
FullName: path.Join(repos[i].Owner.Name, repos[i].Name), |
|
} |
|
} |
|
|
|
ctx.Render.JSON(200, map[string]interface{}{ |
|
"ok": true, |
|
"data": results, |
|
}) |
|
} |
|
|
|
func Migrate(ctx *middleware.Context, form auth.MigrateRepoForm) { |
|
u, err := models.GetUserByName(ctx.Query("username")) |
|
if err != nil { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
return |
|
} |
|
if !u.ValidtePassword(ctx.Query("password")) { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": "username or password is not correct", |
|
}) |
|
return |
|
} |
|
|
|
ctxUser := u |
|
// Not equal means current user is an organization. |
|
if form.Uid != u.Id { |
|
org, err := models.GetUserById(form.Uid) |
|
if err != nil { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
return |
|
} |
|
ctxUser = org |
|
} |
|
|
|
if ctx.HasError() { |
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": ctx.GetErrMsg(), |
|
}) |
|
return |
|
} |
|
|
|
if ctxUser.IsOrganization() { |
|
// Check ownership of organization. |
|
if !ctxUser.IsOrgOwner(u.Id) { |
|
ctx.JSON(403, map[string]interface{}{ |
|
"ok": false, |
|
"error": "given user is not owner of organization", |
|
}) |
|
return |
|
} |
|
} |
|
|
|
authStr := strings.Replace(fmt.Sprintf("://%s:%s", |
|
form.AuthUserName, form.AuthPasswd), "@", "%40", -1) |
|
url := strings.Replace(form.HttpsUrl, "://", authStr+"@", 1) |
|
repo, err := models.MigrateRepository(ctxUser, form.RepoName, form.Description, form.Private, |
|
form.Mirror, url) |
|
if err == nil { |
|
log.Trace("Repository migrated: %s/%s", ctxUser.Name, form.RepoName) |
|
ctx.JSON(200, map[string]interface{}{ |
|
"ok": true, |
|
"data": "/" + ctxUser.Name + "/" + form.RepoName, |
|
}) |
|
return |
|
} |
|
|
|
if repo != nil { |
|
if errDelete := models.DeleteRepository(ctxUser.Id, repo.Id, ctxUser.Name); errDelete != nil { |
|
log.Error(4, "DeleteRepository: %v", errDelete) |
|
} |
|
} |
|
|
|
ctx.JSON(500, map[string]interface{}{ |
|
"ok": false, |
|
"error": err.Error(), |
|
}) |
|
} |
|
|
|
// GET /user/repos |
|
// https://developer.github.com/v3/repos/#list-your-repositories |
|
func ListMyRepos(ctx *middleware.Context) { |
|
ownRepos, err := models.GetRepositories(ctx.User.Id, true) |
|
if err != nil { |
|
ctx.JSON(500, &base.ApiJsonErr{"GetRepositories: " + err.Error(), base.DOC_URL}) |
|
return |
|
} |
|
numOwnRepos := len(ownRepos) |
|
|
|
collaRepos, err := models.GetCollaborativeRepos(ctx.User.Name) |
|
if err != nil { |
|
ctx.JSON(500, &base.ApiJsonErr{"GetCollaborativeRepos: " + err.Error(), base.DOC_URL}) |
|
return |
|
} |
|
|
|
sshUrlFmt := "%s@%s:%s/%s.git" |
|
if setting.SshPort != 22 { |
|
sshUrlFmt = "ssh://%s@%s:%d/%s/%s.git" |
|
} |
|
|
|
repos := make([]*api.Repository, numOwnRepos+len(collaRepos)) |
|
// FIXME: make only one loop |
|
for i := range ownRepos { |
|
repos[i] = &api.Repository{ |
|
Id: ownRepos[i].Id, |
|
Owner: api.User{ |
|
Id: ctx.User.Id, |
|
UserName: ctx.User.Name, |
|
AvatarUrl: string(setting.Protocol) + ctx.User.AvatarLink(), |
|
}, |
|
FullName: ctx.User.Name + "/" + ownRepos[i].Name, |
|
Private: ownRepos[i].IsPrivate, |
|
Fork: ownRepos[i].IsFork, |
|
HtmlUrl: setting.AppUrl + ctx.User.Name + "/" + ownRepos[i].Name, |
|
SshUrl: fmt.Sprintf(sshUrlFmt, setting.RunUser, setting.Domain, ctx.User.LowerName, ownRepos[i].LowerName), |
|
Permissions: api.Permission{true, true, true}, |
|
} |
|
repos[i].CloneUrl = repos[i].HtmlUrl + ".git" |
|
} |
|
for i := range collaRepos { |
|
if err = collaRepos[i].GetOwner(); err != nil { |
|
ctx.JSON(500, &base.ApiJsonErr{"GetOwner: " + err.Error(), base.DOC_URL}) |
|
return |
|
} |
|
j := i + numOwnRepos |
|
repos[j] = &api.Repository{ |
|
Id: collaRepos[i].Id, |
|
Owner: api.User{ |
|
Id: collaRepos[i].Owner.Id, |
|
UserName: collaRepos[i].Owner.Name, |
|
AvatarUrl: string(setting.Protocol) + collaRepos[i].Owner.AvatarLink(), |
|
}, |
|
FullName: collaRepos[i].Owner.Name + "/" + collaRepos[i].Name, |
|
Private: collaRepos[i].IsPrivate, |
|
Fork: collaRepos[i].IsFork, |
|
HtmlUrl: setting.AppUrl + collaRepos[i].Owner.Name + "/" + collaRepos[i].Name, |
|
SshUrl: fmt.Sprintf(sshUrlFmt, setting.RunUser, setting.Domain, collaRepos[i].Owner.LowerName, collaRepos[i].LowerName), |
|
Permissions: api.Permission{false, collaRepos[i].CanPush, true}, |
|
} |
|
repos[j].CloneUrl = repos[j].HtmlUrl + ".git" |
|
|
|
// FIXME: cache result to reduce DB query? |
|
if collaRepos[i].Owner.IsOrganization() && collaRepos[i].Owner.IsOrgOwner(ctx.User.Id) { |
|
repos[j].Permissions.Admin = true |
|
} |
|
} |
|
|
|
ctx.JSON(200, &repos) |
|
}
|
|
|