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