commit
5c57a06c51
@ -0,0 +1,53 @@ |
||||
Gogs - Go Git Service [![wercker status](https://app.wercker.com/status/ad0bdb0bc450ac6f09bc56b9640a50aa/s/ "wercker status")](https://app.wercker.com/project/bykey/ad0bdb0bc450ac6f09bc56b9640a50aa) [![Build Status](https://drone.io/github.com/gogits/gogs/status.png)](https://drone.io/github.com/gogits/gogs/latest) |
||||
===================== |
||||
|
||||
Gogs(Go Git Service) 是一个由 Go 语言编写的自助 Git 托管服务。 |
||||
|
||||
![Demo](http://gowalker.org/public/gogs_demo.gif) |
||||
|
||||
##### 当前版本:0.1.6 Alpha |
||||
|
||||
## 开发目的 |
||||
|
||||
Gogs 完全使用 Go 语言来实现对 Git 数据的操作,实现 **零** 依赖,并且支持 Go 语言所支持的 **所有平台**,包括 Linux、Mac OS X 以及 Windows。 |
||||
|
||||
更重要的是,您只需要一个可执行文件就能借助 Gogs 快速搭建属于您自己的代码托管服务! |
||||
|
||||
## 项目概览 |
||||
|
||||
- 有关项目设计、开发说明、变更日志和路线图,请通过 [Wiki](https://github.com/gogits/gogs/wiki) 查看。 |
||||
- 您可以到 [Trello Broad](https://trello.com/b/uxAoeLUl/gogs-go-git-service) 跟随开发团队的脚步。 |
||||
- 想要先睹为快?通过 [在线体验](http://try.gogits.org/Unknown/gogs) 或查看 **安装部署 -> 二进制安装** 小节。 |
||||
- 使用过程中遇到问题?尝试从 [故障排查](https://github.com/gogits/gogs/wiki/Troubleshooting) 页面获取帮助。 |
||||
|
||||
## 功能特性 |
||||
|
||||
- 活动时间线 |
||||
- SSH 协议支持 |
||||
- 注册/删除用户 |
||||
- 创建/删除/关注公开仓库 |
||||
- 用户个人信息页面 |
||||
- 仓库浏览器 |
||||
- Gravatar 支持 |
||||
- 邮件服务(注册) |
||||
- 管理员面板 |
||||
- 支持 MySQL、PostgreSQL 以及 SQLite3(仅限二进制版本) |
||||
|
||||
## 安装部署 |
||||
|
||||
在安装 Gogs 之前,您需要先安装 [基本环境](https://github.com/gogits/gogs/wiki/Prerequirements)。 |
||||
|
||||
然后,您可以通过以下两种方式来安装 Gogs: |
||||
|
||||
- [二进制安装](https://github.com/gogits/gogs/wiki/Install-from-binary): **强烈推荐** 适合体验者和实际部署 |
||||
- [源码安装](https://github.com/gogits/gogs/wiki/Install-from-source) |
||||
|
||||
## 特别鸣谢 |
||||
|
||||
- Logo 基于 [martini](https://github.com/martini-contrib) 修改而来。 |
||||
- 邮件服务、模块设计基于 [WeTalk](https://github.com/beego/wetalk) 修改而来。 |
||||
- 系统监视状态基于 [GoBlog](https://github.com/fuxiaohei/goblog) 修改而来。 |
||||
|
||||
## 贡献成员 |
||||
|
||||
本项目最初由 [Unknown](https://github.com/Unknwon) 和 [lunny](https://github.com/lunny) 发起,随后 [fuxiaohei](https://github.com/fuxiaohei) 与 [slene](https://github.com/slene) 加入到开发团队。您可以通过查看 [贡献者页面](https://github.com/gogits/gogs/graphs/contributors) 获取完整的贡献者列表。 |
@ -0,0 +1,54 @@ |
||||
// 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 auth |
||||
|
||||
import ( |
||||
"net/http" |
||||
"reflect" |
||||
|
||||
"github.com/codegangsta/martini" |
||||
|
||||
"github.com/gogits/binding" |
||||
|
||||
"github.com/gogits/gogs/modules/base" |
||||
"github.com/gogits/gogs/modules/log" |
||||
) |
||||
|
||||
type CreateIssueForm struct { |
||||
IssueName string `form:"name" binding:"Required;MaxSize(50)"` |
||||
RepoId int64 `form:"repoid" binding:"Required"` |
||||
MilestoneId int64 `form:"milestoneid" binding:"Required"` |
||||
AssigneeId int64 `form:"assigneeid"` |
||||
Labels string `form:"labels"` |
||||
Content string `form:"content"` |
||||
} |
||||
|
||||
func (f *CreateIssueForm) Name(field string) string { |
||||
names := map[string]string{ |
||||
"IssueName": "Issue name", |
||||
"RepoId": "Repository ID", |
||||
"MilestoneId": "Milestone ID", |
||||
} |
||||
return names[field] |
||||
} |
||||
|
||||
func (f *CreateIssueForm) Validate(errors *binding.Errors, req *http.Request, context martini.Context) { |
||||
if req.Method == "GET" || errors.Count() == 0 { |
||||
return |
||||
} |
||||
|
||||
data := context.Get(reflect.TypeOf(base.TmplData{})).Interface().(base.TmplData) |
||||
data["HasError"] = true |
||||
AssignForm(f, data) |
||||
|
||||
if len(errors.Overall) > 0 { |
||||
for _, err := range errors.Overall { |
||||
log.Error("CreateIssueForm.Validate: %v", err) |
||||
} |
||||
return |
||||
} |
||||
|
||||
validate(errors, data, f) |
||||
} |
@ -0,0 +1,85 @@ |
||||
// 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 repo |
||||
|
||||
import ( |
||||
"fmt" |
||||
|
||||
"github.com/codegangsta/martini" |
||||
|
||||
"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" |
||||
) |
||||
|
||||
func Issues(ctx *middleware.Context, params martini.Params) { |
||||
ctx.Data["Title"] = "Issues" |
||||
ctx.Data["IsRepoToolbarIssues"] = true |
||||
|
||||
milestoneId, _ := base.StrTo(params["milestone"]).Int() |
||||
page, _ := base.StrTo(params["page"]).Int() |
||||
|
||||
var err error |
||||
ctx.Data["Issues"], err = models.GetIssues(0, ctx.Repo.Repository.Id, 0, |
||||
int64(milestoneId), page, params["state"] == "closed", false, params["labels"], params["sortType"]) |
||||
if err != nil { |
||||
ctx.Handle(200, "issue.Issues: %v", err) |
||||
return |
||||
} |
||||
|
||||
ctx.HTML(200, "repo/issues") |
||||
} |
||||
|
||||
func CreateIssue(ctx *middleware.Context, params martini.Params, form auth.CreateIssueForm) { |
||||
if !ctx.Repo.IsOwner { |
||||
ctx.Handle(404, "issue.CreateIssue", nil) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["Title"] = "Create issue" |
||||
|
||||
if ctx.Req.Method == "GET" { |
||||
ctx.HTML(200, "issue/create") |
||||
return |
||||
} |
||||
|
||||
if ctx.HasError() { |
||||
ctx.HTML(200, "issue/create") |
||||
return |
||||
} |
||||
|
||||
issue, err := models.CreateIssue(ctx.User.Id, form.RepoId, form.MilestoneId, form.AssigneeId, |
||||
form.IssueName, form.Labels, form.Content, false) |
||||
if err == nil { |
||||
log.Trace("%s Issue created: %d", form.RepoId, issue.Id) |
||||
ctx.Redirect(fmt.Sprintf("/%s/%s/issues/%d", params["username"], params["reponame"], issue.Index)) |
||||
return |
||||
} |
||||
ctx.Handle(200, "issue.CreateIssue", err) |
||||
} |
||||
|
||||
func ViewIssue(ctx *middleware.Context, params martini.Params) { |
||||
issueid, err := base.StrTo(params["issueid"]).Int() |
||||
if err != nil { |
||||
ctx.Handle(404, "issue.ViewIssue", err) |
||||
return |
||||
} |
||||
|
||||
issue, err := models.GetIssueById(int64(issueid)) |
||||
if err != nil { |
||||
if err == models.ErrIssueNotExist { |
||||
ctx.Handle(404, "issue.ViewIssue", err) |
||||
} else { |
||||
ctx.Handle(200, "issue.ViewIssue", err) |
||||
} |
||||
return |
||||
} |
||||
|
||||
ctx.Data["Title"] = issue.Name |
||||
ctx.Data["Issue"] = issue |
||||
ctx.HTML(200, "issue/view") |
||||
} |
@ -1,307 +0,0 @@ |
||||
// 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 repo |
||||
|
||||
import ( |
||||
"path" |
||||
"strings" |
||||
|
||||
"github.com/codegangsta/martini" |
||||
|
||||
"github.com/gogits/git" |
||||
"github.com/gogits/webdav" |
||||
|
||||
"github.com/gogits/gogs/models" |
||||
"github.com/gogits/gogs/modules/base" |
||||
"github.com/gogits/gogs/modules/log" |
||||
"github.com/gogits/gogs/modules/middleware" |
||||
) |
||||
|
||||
func Branches(ctx *middleware.Context, params martini.Params) { |
||||
if !ctx.Repo.IsValid { |
||||
return |
||||
} |
||||
|
||||
brs, err := models.GetBranches(params["username"], params["reponame"]) |
||||
if err != nil { |
||||
ctx.Handle(200, "repo.Branches", err) |
||||
return |
||||
} else if len(brs) == 0 { |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["Username"] = params["username"] |
||||
ctx.Data["Reponame"] = params["reponame"] |
||||
|
||||
ctx.Data["Branchname"] = brs[0] |
||||
ctx.Data["Branches"] = brs |
||||
ctx.Data["IsRepoToolbarBranches"] = true |
||||
|
||||
ctx.HTML(200, "repo/branches") |
||||
} |
||||
|
||||
func Single(ctx *middleware.Context, params martini.Params) { |
||||
if !ctx.Repo.IsValid { |
||||
return |
||||
} |
||||
|
||||
if len(params["branchname"]) == 0 { |
||||
params["branchname"] = "master" |
||||
} |
||||
|
||||
// Get tree path
|
||||
treename := params["_1"] |
||||
|
||||
if len(treename) > 0 && treename[len(treename)-1] == '/' { |
||||
ctx.Redirect("/"+ctx.Repo.Owner.LowerName+"/"+ |
||||
ctx.Repo.Repository.Name+"/src/"+params["branchname"]+"/"+treename[:len(treename)-1], 302) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["IsRepoToolbarSource"] = true |
||||
|
||||
// Branches.
|
||||
brs, err := models.GetBranches(params["username"], params["reponame"]) |
||||
if err != nil { |
||||
log.Error("repo.Single(GetBranches): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} else if len(brs) == 0 { |
||||
ctx.Data["IsBareRepo"] = true |
||||
ctx.HTML(200, "repo/single") |
||||
return |
||||
} |
||||
|
||||
ctx.Data["Branches"] = brs |
||||
|
||||
repoFile, err := models.GetTargetFile(params["username"], params["reponame"], |
||||
params["branchname"], params["commitid"], treename) |
||||
|
||||
if err != nil && err != models.ErrRepoFileNotExist { |
||||
log.Error("repo.Single(GetTargetFile): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
branchLink := "/" + ctx.Repo.Owner.LowerName + "/" + ctx.Repo.Repository.Name + "/src/" + params["branchname"] |
||||
|
||||
if len(treename) != 0 && repoFile == nil { |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
if repoFile != nil && repoFile.IsFile() { |
||||
if repoFile.Size > 1024*1024 || repoFile.Filemode != git.FileModeBlob { |
||||
ctx.Data["FileIsLarge"] = true |
||||
} else if blob, err := repoFile.LookupBlob(); err != nil { |
||||
log.Error("repo.Single(repoFile.LookupBlob): %v", err) |
||||
ctx.Error(404) |
||||
} else { |
||||
ctx.Data["IsFile"] = true |
||||
ctx.Data["FileName"] = repoFile.Name |
||||
ext := path.Ext(repoFile.Name) |
||||
if len(ext) > 0 { |
||||
ext = ext[1:] |
||||
} |
||||
ctx.Data["FileExt"] = ext |
||||
|
||||
readmeExist := base.IsMarkdownFile(repoFile.Name) || base.IsReadmeFile(repoFile.Name) |
||||
ctx.Data["ReadmeExist"] = readmeExist |
||||
if readmeExist { |
||||
ctx.Data["FileContent"] = string(base.RenderMarkdown(blob.Contents(), "")) |
||||
} else { |
||||
ctx.Data["FileContent"] = string(blob.Contents()) |
||||
} |
||||
} |
||||
|
||||
} else { |
||||
// Directory and file list.
|
||||
files, err := models.GetReposFiles(params["username"], params["reponame"], |
||||
params["branchname"], params["commitid"], treename) |
||||
if err != nil { |
||||
log.Error("repo.Single(GetReposFiles): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["Files"] = files |
||||
|
||||
var readmeFile *models.RepoFile |
||||
|
||||
for _, f := range files { |
||||
if !f.IsFile() || !base.IsReadmeFile(f.Name) { |
||||
continue |
||||
} else { |
||||
readmeFile = f |
||||
break |
||||
} |
||||
} |
||||
|
||||
if readmeFile != nil { |
||||
ctx.Data["ReadmeExist"] = true |
||||
// if file large than 1M not show it
|
||||
if readmeFile.Size > 1024*1024 || readmeFile.Filemode != git.FileModeBlob { |
||||
ctx.Data["FileIsLarge"] = true |
||||
} else if blob, err := readmeFile.LookupBlob(); err != nil { |
||||
log.Error("repo.Single(readmeFile.LookupBlob): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} else { |
||||
// current repo branch link
|
||||
|
||||
ctx.Data["FileName"] = readmeFile.Name |
||||
ctx.Data["FileContent"] = string(base.RenderMarkdown(blob.Contents(), branchLink)) |
||||
} |
||||
} |
||||
} |
||||
|
||||
ctx.Data["Username"] = params["username"] |
||||
ctx.Data["Reponame"] = params["reponame"] |
||||
ctx.Data["Branchname"] = params["branchname"] |
||||
|
||||
var treenames []string |
||||
Paths := make([]string, 0) |
||||
|
||||
if len(treename) > 0 { |
||||
treenames = strings.Split(treename, "/") |
||||
for i, _ := range treenames { |
||||
Paths = append(Paths, strings.Join(treenames[0:i+1], "/")) |
||||
} |
||||
|
||||
ctx.Data["HasParentPath"] = true |
||||
if len(Paths)-2 >= 0 { |
||||
ctx.Data["ParentPath"] = "/" + Paths[len(Paths)-2] |
||||
} |
||||
} |
||||
|
||||
// Get latest commit according username and repo name
|
||||
commit, err := models.GetCommit(params["username"], params["reponame"], |
||||
params["branchname"], params["commitid"]) |
||||
if err != nil { |
||||
log.Error("repo.Single(GetCommit): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
ctx.Data["LastCommit"] = commit |
||||
|
||||
ctx.Data["Paths"] = Paths |
||||
ctx.Data["Treenames"] = treenames |
||||
ctx.Data["BranchLink"] = branchLink |
||||
ctx.HTML(200, "repo/single") |
||||
} |
||||
|
||||
func Http(ctx *middleware.Context, params martini.Params) { |
||||
/*if !ctx.Repo.IsValid { |
||||
return |
||||
}*/ |
||||
|
||||
// TODO: access check
|
||||
|
||||
username := params["username"] |
||||
reponame := params["reponame"] |
||||
if strings.HasSuffix(reponame, ".git") { |
||||
reponame = reponame[:len(reponame)-4] |
||||
} |
||||
|
||||
prefix := path.Join("/", username, params["reponame"]) |
||||
server := &webdav.Server{ |
||||
Fs: webdav.Dir(models.RepoPath(username, reponame)), |
||||
TrimPrefix: prefix, |
||||
Listings: true, |
||||
} |
||||
|
||||
server.ServeHTTP(ctx.ResponseWriter, ctx.Req) |
||||
} |
||||
|
||||
func Setting(ctx *middleware.Context, params martini.Params) { |
||||
if !ctx.Repo.IsOwner { |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["IsRepoToolbarSetting"] = true |
||||
|
||||
// Branches.
|
||||
brs, err := models.GetBranches(params["username"], params["reponame"]) |
||||
if err != nil { |
||||
log.Error("repo.Setting(GetBranches): %v", err) |
||||
ctx.Error(404) |
||||
return |
||||
} else if len(brs) == 0 { |
||||
ctx.Data["IsBareRepo"] = true |
||||
ctx.HTML(200, "repo/setting") |
||||
return |
||||
} |
||||
|
||||
var title string |
||||
if t, ok := ctx.Data["Title"].(string); ok { |
||||
title = t |
||||
} |
||||
|
||||
if len(params["branchname"]) == 0 { |
||||
params["branchname"] = "master" |
||||
} |
||||
|
||||
ctx.Data["Branchname"] = params["branchname"] |
||||
ctx.Data["Title"] = title + " - settings" |
||||
ctx.HTML(200, "repo/setting") |
||||
} |
||||
|
||||
func Commits(ctx *middleware.Context, params martini.Params) { |
||||
brs, err := models.GetBranches(params["username"], params["reponame"]) |
||||
if err != nil { |
||||
ctx.Handle(200, "repo.Commits", err) |
||||
return |
||||
} else if len(brs) == 0 { |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
|
||||
ctx.Data["IsRepoToolbarCommits"] = true |
||||
commits, err := models.GetCommits(params["username"], |
||||
params["reponame"], params["branchname"]) |
||||
if err != nil { |
||||
ctx.Error(404) |
||||
return |
||||
} |
||||
ctx.Data["Username"] = params["username"] |
||||
ctx.Data["Reponame"] = params["reponame"] |
||||
ctx.Data["CommitCount"] = commits.Len() |
||||
ctx.Data["Commits"] = commits |
||||
ctx.HTML(200, "repo/commits") |
||||
} |
||||
|
||||
func Issues(ctx *middleware.Context) { |
||||
ctx.Data["IsRepoToolbarIssues"] = true |
||||
ctx.HTML(200, "repo/issues") |
||||
} |
||||
|
||||
func Pulls(ctx *middleware.Context) { |
||||
ctx.Data["IsRepoToolbarPulls"] = true |
||||
ctx.HTML(200, "repo/pulls") |
||||
} |
||||
|
||||
func Action(ctx *middleware.Context, params martini.Params) { |
||||
var err error |
||||
switch params["action"] { |
||||
case "watch": |
||||
err = models.WatchRepo(ctx.User.Id, ctx.Repo.Repository.Id, true) |
||||
case "unwatch": |
||||
err = models.WatchRepo(ctx.User.Id, ctx.Repo.Repository.Id, false) |
||||
} |
||||
|
||||
if err != nil { |
||||
log.Error("repo.Action(%s): %v", params["action"], err) |
||||
ctx.JSON(200, map[string]interface{}{ |
||||
"ok": false, |
||||
"err": err.Error(), |
||||
}) |
||||
return |
||||
} |
||||
ctx.JSON(200, map[string]interface{}{ |
||||
"ok": true, |
||||
}) |
||||
} |
@ -0,0 +1,7 @@ |
||||
{{template "base/head" .}} |
||||
{{template "base/navbar" .}} |
||||
<div id="gogs-body" class="container"> |
||||
<h4>This page is not found !</h4> |
||||
<p>Application Version: {{AppVer}}</p> |
||||
</div> |
||||
{{template "base/footer" .}} |
@ -0,0 +1,7 @@ |
||||
{{template "base/head" .}} |
||||
{{template "base/navbar" .}} |
||||
<div id="gogs-body" class="container"> |
||||
<p>An error is occurred : {{.ErrorMsg}}</p> |
||||
<p>Application Version: {{AppVer}}</p> |
||||
</div> |
||||
{{template "base/footer" .}} |
Loading…
Reference in new issue