commit
8faa0dbcd7
33 changed files with 956 additions and 265 deletions
|
@ -2,12 +2,11 @@
|
||||||
"paths": ["."],
|
"paths": ["."],
|
||||||
"depth": 2,
|
"depth": 2,
|
||||||
"exclude": [],
|
"exclude": [],
|
||||||
"include": ["\\.go$"],
|
"include": ["\\.go$", "\\.ini$"],
|
||||||
"command": [
|
"command": [
|
||||||
"bash", "-c", "go build && ./gogs web"
|
"bash", "-c", "go build && ./gogs web"
|
||||||
],
|
],
|
||||||
"env": {
|
"env": {
|
||||||
"POWERED_BY": "github.com/shxsun/fswatch"
|
"POWERED_BY": "github.com/shxsun/fswatch"
|
||||||
},
|
}
|
||||||
"enable-restart": true
|
|
||||||
}
|
}
|
||||||
|
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -33,3 +33,4 @@ _testmain.go
|
||||||
*.exe~
|
*.exe~
|
||||||
gogs
|
gogs
|
||||||
__pycache__
|
__pycache__
|
||||||
|
*.pem
|
||||||
|
|
|
@ -5,7 +5,7 @@ Gogs(Go Git Service) is a Self Hosted Git Service in the Go Programming Language
|
||||||
|
|
||||||
![Demo](http://gowalker.org/public/gogs_demo.gif)
|
![Demo](http://gowalker.org/public/gogs_demo.gif)
|
||||||
|
|
||||||
##### Current version: 0.2.2 Alpha
|
##### Current version: 0.2.3 Alpha
|
||||||
|
|
||||||
#### Due to testing purpose, data of [try.gogits.org](http://try.gogits.org) has been reset in April 6, 2014 and will reset multiple times after. Please do NOT put your important data on the site.
|
#### Due to testing purpose, data of [try.gogits.org](http://try.gogits.org) has been reset in April 6, 2014 and will reset multiple times after. Please do NOT put your important data on the site.
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ More importantly, Gogs only needs one binary to setup your own project hosting o
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Activity timeline
|
- Activity timeline
|
||||||
- SSH/HTTPS(Clone only) protocol support.
|
- SSH/HTTP(S) protocol support.
|
||||||
- Register/delete/rename account.
|
- Register/delete/rename account.
|
||||||
- Create/delete/watch/rename/transfer public repository.
|
- Create/delete/watch/rename/transfer public repository.
|
||||||
- Repository viewer.
|
- Repository viewer.
|
||||||
|
|
|
@ -5,7 +5,7 @@ Gogs(Go Git Service) 是一个由 Go 语言编写的自助 Git 托管服务。
|
||||||
|
|
||||||
![Demo](http://gowalker.org/public/gogs_demo.gif)
|
![Demo](http://gowalker.org/public/gogs_demo.gif)
|
||||||
|
|
||||||
##### 当前版本:0.2.2 Alpha
|
##### 当前版本:0.2.3 Alpha
|
||||||
|
|
||||||
## 开发目的
|
## 开发目的
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ Gogs 完全使用 Go 语言来实现对 Git 数据的操作,实现 **零** 依
|
||||||
## 功能特性
|
## 功能特性
|
||||||
|
|
||||||
- 活动时间线
|
- 活动时间线
|
||||||
- SSH/HTTPS(仅限 Clone) 协议支持
|
- SSH/HTTP(S) 协议支持
|
||||||
- 注册/删除/重命名用户
|
- 注册/删除/重命名用户
|
||||||
- 创建/删除/关注/重命名/转移公开仓库
|
- 创建/删除/关注/重命名/转移公开仓库
|
||||||
- 仓库浏览器
|
- 仓库浏览器
|
||||||
|
|
2
gogs.go
2
gogs.go
|
@ -19,7 +19,7 @@ import (
|
||||||
// Test that go1.2 tag above is included in builds. main.go refers to this definition.
|
// Test that go1.2 tag above is included in builds. main.go refers to this definition.
|
||||||
const go12tag = true
|
const go12tag = true
|
||||||
|
|
||||||
const APP_VER = "0.2.2.0407 Alpha"
|
const APP_VER = "0.2.3.0409 Alpha"
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
base.AppVer = APP_VER
|
base.AppVer = APP_VER
|
||||||
|
|
|
@ -142,7 +142,8 @@ func GetReposFiles(userName, repoName, commitId, rpath string) ([]*RepoFile, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func getReposFiles(userName, repoName, commitId string, rpath string) ([]*RepoFile, error) {
|
func getReposFiles(userName, repoName, commitId string, rpath string) ([]*RepoFile, error) {
|
||||||
repo, err := git.OpenRepository(RepoPath(userName, repoName))
|
repopath := RepoPath(userName, repoName)
|
||||||
|
repo, err := git.OpenRepository(repopath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -162,77 +163,23 @@ func getReposFiles(userName, repoName, commitId string, rpath string) ([]*RepoFi
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
var cm = commit
|
cmd := exec.Command("git", "log", "-1", "--pretty=format:%H", commitId, "--", path.Join(dirname, entry.Name))
|
||||||
var i int
|
cmd.Dir = repopath
|
||||||
for {
|
out, err := cmd.Output()
|
||||||
i = i + 1
|
if err != nil {
|
||||||
//fmt.Println(".....", i, cm.Id(), cm.ParentCount())
|
return 0
|
||||||
if cm.ParentCount() == 0 {
|
|
||||||
break
|
|
||||||
} else if cm.ParentCount() == 1 {
|
|
||||||
pt, _ := repo.SubTree(cm.Parent(0).Tree, dirname)
|
|
||||||
if pt == nil {
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
pEntry := pt.EntryByName(entry.Name)
|
filecm, err := repo.GetCommit(string(out))
|
||||||
if pEntry == nil || !pEntry.Id.Equal(entry.Id) {
|
if err != nil {
|
||||||
break
|
return 0
|
||||||
} else {
|
|
||||||
cm = cm.Parent(0)
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
var emptyCnt = 0
|
|
||||||
var sameIdcnt = 0
|
|
||||||
var lastSameCm *git.Commit
|
|
||||||
//fmt.Println(".....", cm.ParentCount())
|
|
||||||
for i := 0; i < cm.ParentCount(); i++ {
|
|
||||||
//fmt.Println("parent", i, cm.Parent(i).Id())
|
|
||||||
p := cm.Parent(i)
|
|
||||||
pt, _ := repo.SubTree(p.Tree, dirname)
|
|
||||||
var pEntry *git.TreeEntry
|
|
||||||
if pt != nil {
|
|
||||||
pEntry = pt.EntryByName(entry.Name)
|
|
||||||
}
|
|
||||||
|
|
||||||
//fmt.Println("pEntry", pEntry)
|
|
||||||
|
|
||||||
if pEntry == nil {
|
|
||||||
emptyCnt = emptyCnt + 1
|
|
||||||
if emptyCnt+sameIdcnt == cm.ParentCount() {
|
|
||||||
if lastSameCm == nil {
|
|
||||||
goto loop
|
|
||||||
} else {
|
|
||||||
cm = lastSameCm
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
//fmt.Println(i, "pEntry", pEntry.Id, "entry", entry.Id)
|
|
||||||
if !pEntry.Id.Equal(entry.Id) {
|
|
||||||
goto loop
|
|
||||||
} else {
|
|
||||||
lastSameCm = cm.Parent(i)
|
|
||||||
sameIdcnt = sameIdcnt + 1
|
|
||||||
if emptyCnt+sameIdcnt == cm.ParentCount() {
|
|
||||||
// TODO: now follow the first parent commit?
|
|
||||||
cm = lastSameCm
|
|
||||||
//fmt.Println("sameId...")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
loop:
|
|
||||||
|
|
||||||
rp := &RepoFile{
|
rp := &RepoFile{
|
||||||
entry,
|
entry,
|
||||||
path.Join(dirname, entry.Name),
|
path.Join(dirname, entry.Name),
|
||||||
size,
|
size,
|
||||||
repo,
|
repo,
|
||||||
cm,
|
filecm,
|
||||||
}
|
}
|
||||||
|
|
||||||
if entry.IsFile() {
|
if entry.IsFile() {
|
||||||
|
|
|
@ -1,6 +1,10 @@
|
||||||
|
// 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 models
|
package models
|
||||||
|
|
||||||
import "fmt"
|
import "errors"
|
||||||
|
|
||||||
// OT: Oauth2 Type
|
// OT: Oauth2 Type
|
||||||
const (
|
const (
|
||||||
|
@ -9,12 +13,18 @@ const (
|
||||||
OT_TWITTER
|
OT_TWITTER
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrOauth2RecordNotExists = errors.New("not exists oauth2 record")
|
||||||
|
ErrOauth2NotAssociatedWithUser = errors.New("not associated with user")
|
||||||
|
)
|
||||||
|
|
||||||
type Oauth2 struct {
|
type Oauth2 struct {
|
||||||
Uid int64 `xorm:"pk"` // userId
|
Id int64
|
||||||
|
Uid int64 // userId
|
||||||
|
User *User `xorm:"-"`
|
||||||
Type int `xorm:"pk unique(oauth)"` // twitter,github,google...
|
Type int `xorm:"pk unique(oauth)"` // twitter,github,google...
|
||||||
Identity string `xorm:"pk unique(oauth)"` // id..
|
Identity string `xorm:"pk unique(oauth)"` // id..
|
||||||
Token string `xorm:"VARCHAR(200) not null"`
|
Token string `xorm:"VARCHAR(200) not null"`
|
||||||
//RefreshTime time.Time `xorm:"created"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func AddOauth2(oa *Oauth2) (err error) {
|
func AddOauth2(oa *Oauth2) (err error) {
|
||||||
|
@ -24,16 +34,16 @@ func AddOauth2(oa *Oauth2) (err error) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func GetOauth2User(identity string) (u *User, err error) {
|
func GetOauth2(identity string) (oa *Oauth2, err error) {
|
||||||
oa := &Oauth2{}
|
oa = &Oauth2{Identity: identity}
|
||||||
oa.Identity = identity
|
isExist, err := orm.Get(oa)
|
||||||
exists, err := orm.Get(oa)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
|
} else if !isExist {
|
||||||
|
return nil, ErrOauth2RecordNotExists
|
||||||
|
} else if oa.Uid == 0 {
|
||||||
|
return oa, ErrOauth2NotAssociatedWithUser
|
||||||
}
|
}
|
||||||
if !exists {
|
oa.User, err = GetUserById(oa.Uid)
|
||||||
err = fmt.Errorf("not exists oauth2: %s", identity)
|
return oa, err
|
||||||
return
|
|
||||||
}
|
|
||||||
return GetUserById(oa.Uid)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -79,6 +79,7 @@ type Repository struct {
|
||||||
NumOpenIssues int `xorm:"-"`
|
NumOpenIssues int `xorm:"-"`
|
||||||
IsPrivate bool
|
IsPrivate bool
|
||||||
IsBare bool
|
IsBare bool
|
||||||
|
IsGoget bool
|
||||||
Created time.Time `xorm:"created"`
|
Created time.Time `xorm:"created"`
|
||||||
Updated time.Time `xorm:"updated"`
|
Updated time.Time `xorm:"updated"`
|
||||||
}
|
}
|
||||||
|
@ -261,6 +262,13 @@ func createHookUpdate(hookPath, content string) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetRepoEnvs sets environment variables for command update.
|
||||||
|
func SetRepoEnvs(userId int64, userName, repoName string) {
|
||||||
|
os.Setenv("userId", base.ToStr(userId))
|
||||||
|
os.Setenv("userName", userName)
|
||||||
|
os.Setenv("repoName", repoName)
|
||||||
|
}
|
||||||
|
|
||||||
// InitRepository initializes README and .gitignore if needed.
|
// InitRepository initializes README and .gitignore if needed.
|
||||||
func initRepository(f string, user *User, repo *Repository, initReadme bool, repoLang, license string) error {
|
func initRepository(f string, user *User, repo *Repository, initReadme bool, repoLang, license string) error {
|
||||||
repoPath := RepoPath(user.Name, repo.Name)
|
repoPath := RepoPath(user.Name, repo.Name)
|
||||||
|
@ -333,10 +341,7 @@ func initRepository(f string, user *User, repo *Repository, initReadme bool, rep
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// for update use
|
SetRepoEnvs(user.Id, user.Name, repo.Name)
|
||||||
os.Setenv("userName", user.Name)
|
|
||||||
os.Setenv("userId", base.ToStr(user.Id))
|
|
||||||
os.Setenv("repoName", repo.Name)
|
|
||||||
|
|
||||||
// Apply changes and commit.
|
// Apply changes and commit.
|
||||||
return initRepoCommit(tmpDir, user.NewGitSig())
|
return initRepoCommit(tmpDir, user.NewGitSig())
|
||||||
|
|
|
@ -289,11 +289,21 @@ func DeleteUser(user *User) error {
|
||||||
|
|
||||||
// TODO: check issues, other repos' commits
|
// TODO: check issues, other repos' commits
|
||||||
|
|
||||||
|
// Delete all followers.
|
||||||
|
if _, err = orm.Delete(&Follow{FollowId: user.Id}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
// Delete all feeds.
|
// Delete all feeds.
|
||||||
if _, err = orm.Delete(&Action{UserId: user.Id}); err != nil {
|
if _, err = orm.Delete(&Action{UserId: user.Id}); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Delete all watches.
|
||||||
|
if _, err = orm.Delete(&Watch{UserId: user.Id}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
// Delete all accesses.
|
// Delete all accesses.
|
||||||
if _, err = orm.Delete(&Access{UserName: user.LowerName}); err != nil {
|
if _, err = orm.Delete(&Access{UserName: user.LowerName}); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -316,7 +326,6 @@ func DeleteUser(user *User) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = orm.Delete(user)
|
_, err = orm.Delete(user)
|
||||||
// TODO: delete and update follower information.
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,6 +43,7 @@ var (
|
||||||
AppName string
|
AppName string
|
||||||
AppLogo string
|
AppLogo string
|
||||||
AppUrl string
|
AppUrl string
|
||||||
|
IsProdMode bool
|
||||||
Domain string
|
Domain string
|
||||||
SecretKey string
|
SecretKey string
|
||||||
RunUser string
|
RunUser string
|
||||||
|
|
|
@ -133,14 +133,14 @@ func RenderSpecialLink(rawBytes []byte, urlPrefix string) []byte {
|
||||||
}
|
}
|
||||||
|
|
||||||
func RenderMarkdown(rawBytes []byte, urlPrefix string) []byte {
|
func RenderMarkdown(rawBytes []byte, urlPrefix string) []byte {
|
||||||
// body := RenderSpecialLink(rawBytes, urlPrefix)
|
body := RenderSpecialLink(rawBytes, urlPrefix)
|
||||||
// fmt.Println(string(body))
|
// fmt.Println(string(body))
|
||||||
htmlFlags := 0
|
htmlFlags := 0
|
||||||
// htmlFlags |= gfm.HTML_USE_XHTML
|
// htmlFlags |= gfm.HTML_USE_XHTML
|
||||||
// htmlFlags |= gfm.HTML_USE_SMARTYPANTS
|
// htmlFlags |= gfm.HTML_USE_SMARTYPANTS
|
||||||
// htmlFlags |= gfm.HTML_SMARTYPANTS_FRACTIONS
|
// htmlFlags |= gfm.HTML_SMARTYPANTS_FRACTIONS
|
||||||
// htmlFlags |= gfm.HTML_SMARTYPANTS_LATEX_DASHES
|
// htmlFlags |= gfm.HTML_SMARTYPANTS_LATEX_DASHES
|
||||||
htmlFlags |= gfm.HTML_SKIP_HTML
|
// htmlFlags |= gfm.HTML_SKIP_HTML
|
||||||
htmlFlags |= gfm.HTML_SKIP_STYLE
|
htmlFlags |= gfm.HTML_SKIP_STYLE
|
||||||
htmlFlags |= gfm.HTML_SKIP_SCRIPT
|
htmlFlags |= gfm.HTML_SKIP_SCRIPT
|
||||||
htmlFlags |= gfm.HTML_GITHUB_BLOCKCODE
|
htmlFlags |= gfm.HTML_GITHUB_BLOCKCODE
|
||||||
|
@ -162,7 +162,7 @@ func RenderMarkdown(rawBytes []byte, urlPrefix string) []byte {
|
||||||
extensions |= gfm.EXTENSION_SPACE_HEADERS
|
extensions |= gfm.EXTENSION_SPACE_HEADERS
|
||||||
extensions |= gfm.EXTENSION_NO_EMPTY_LINE_BEFORE_BLOCK
|
extensions |= gfm.EXTENSION_NO_EMPTY_LINE_BEFORE_BLOCK
|
||||||
|
|
||||||
body := gfm.Markdown(rawBytes, renderer, extensions)
|
body = gfm.Markdown(body, renderer, extensions)
|
||||||
// fmt.Println(string(body))
|
// fmt.Println(string(body))
|
||||||
return body
|
return body
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,6 +56,9 @@ var TemplateFuncs template.FuncMap = map[string]interface{}{
|
||||||
"AppDomain": func() string {
|
"AppDomain": func() string {
|
||||||
return Domain
|
return Domain
|
||||||
},
|
},
|
||||||
|
"IsProdMode": func() bool {
|
||||||
|
return IsProdMode
|
||||||
|
},
|
||||||
"LoadTimes": func(startTime time.Time) string {
|
"LoadTimes": func(startTime time.Time) string {
|
||||||
return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms"
|
return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms"
|
||||||
},
|
},
|
||||||
|
|
|
@ -146,7 +146,7 @@ func compile(options RenderOptions) *template.Template {
|
||||||
tmpl := t.New(filepath.ToSlash(name))
|
tmpl := t.New(filepath.ToSlash(name))
|
||||||
|
|
||||||
for _, funcs := range options.Funcs {
|
for _, funcs := range options.Funcs {
|
||||||
tmpl.Funcs(funcs)
|
tmpl = tmpl.Funcs(funcs)
|
||||||
}
|
}
|
||||||
|
|
||||||
template.Must(tmpl.Funcs(helperFuncs).Parse(string(buf)))
|
template.Must(tmpl.Funcs(helperFuncs).Parse(string(buf)))
|
||||||
|
|
|
@ -1,16 +1,7 @@
|
||||||
// Copyright 2014 Google Inc. All Rights Reserved.
|
// Copyright 2014 Google Inc. All Rights Reserved.
|
||||||
//
|
// Copyright 2014 The Gogs Authors. All rights reserved.
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
// Use of this source code is governed by a MIT-style
|
||||||
// you may not use this file except in compliance with the License.
|
// license that can be found in the LICENSE file.
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
// Package oauth2 contains Martini handlers to provide
|
// Package oauth2 contains Martini handlers to provide
|
||||||
// user login via an OAuth 2.0 backend.
|
// user login via an OAuth 2.0 backend.
|
||||||
|
|
|
@ -309,6 +309,18 @@ html, body {
|
||||||
height: 8em;
|
height: 8em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#repo-import-auth {
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 48px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
#repo-import-auth .form-group {
|
||||||
|
box-sizing: border-box;
|
||||||
|
margin-left: 0;
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* gogits user setting */
|
/* gogits user setting */
|
||||||
|
|
||||||
#user-setting-nav > h4, #user-setting-container > h4, #user-setting-container > div > h4,
|
#user-setting-nav > h4, #user-setting-container > h4, #user-setting-container > div > h4,
|
||||||
|
@ -444,6 +456,43 @@ html, body {
|
||||||
margin-right: 1em;
|
margin-right: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#user-dashboard-repo-new .btn-sm.dropdown-toggle {
|
||||||
|
padding: 3px 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#user-dashboard-repo-new .dropdown-menu, #nav-repo-new .dropdown-menu {
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#user-dashboard-repo-new ul, #nav-repo-new ul {
|
||||||
|
margin: 0;
|
||||||
|
width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#user-dashboard-repo-new li a, #nav-repo-new li a {
|
||||||
|
line-height: 36px;
|
||||||
|
display: block;
|
||||||
|
padding: 0 18px;
|
||||||
|
color: #444;
|
||||||
|
}
|
||||||
|
|
||||||
|
#user-dashboard-repo-new li a:hover, #nav-repo-new li a:hover {
|
||||||
|
background: #0093c4;
|
||||||
|
color: #FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nav-repo-new button {
|
||||||
|
border: none;
|
||||||
|
background: transparent;
|
||||||
|
padding: 0;
|
||||||
|
width: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nav-repo-new li .fa {
|
||||||
|
margin: 0 .5em;
|
||||||
|
}
|
||||||
|
|
||||||
/* gogits repo single page */
|
/* gogits repo single page */
|
||||||
|
|
||||||
#body-nav.repo-nav {
|
#body-nav.repo-nav {
|
||||||
|
|
|
@ -7,6 +7,7 @@ package routers
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"os"
|
"os"
|
||||||
|
"os/exec"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/Unknwon/goconfig"
|
"github.com/Unknwon/goconfig"
|
||||||
|
@ -27,6 +28,7 @@ func checkRunMode() {
|
||||||
switch base.Cfg.MustValue("", "RUN_MODE") {
|
switch base.Cfg.MustValue("", "RUN_MODE") {
|
||||||
case "prod":
|
case "prod":
|
||||||
martini.Env = martini.Prod
|
martini.Env = martini.Prod
|
||||||
|
base.IsProdMode = true
|
||||||
case "test":
|
case "test":
|
||||||
martini.Env = martini.Test
|
martini.Env = martini.Test
|
||||||
}
|
}
|
||||||
|
@ -102,6 +104,11 @@ func Install(ctx *middleware.Context, form auth.InstallForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if _, err := exec.LookPath("git"); err != nil {
|
||||||
|
ctx.RenderWithErr("Fail to test 'git' command: "+err.Error(), "install", &form)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// Pass basic check, now test configuration.
|
// Pass basic check, now test configuration.
|
||||||
// Test database setting.
|
// Test database setting.
|
||||||
dbTypes := map[string]string{"mysql": "mysql", "pgsql": "postgres", "sqlite": "sqlite3"}
|
dbTypes := map[string]string{"mysql": "mysql", "pgsql": "postgres", "sqlite": "sqlite3"}
|
||||||
|
|
55
routers/repo/git.go
Normal file
55
routers/repo/git.go
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
package repo
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
const advertise_refs = "--advertise-refs"
|
||||||
|
|
||||||
|
func command(cmd string, opts ...string) string {
|
||||||
|
return fmt.Sprintf("git %s %s", cmd, strings.Join(opts, " "))
|
||||||
|
}
|
||||||
|
|
||||||
|
/*func upload_pack(repository_path string, opts ...string) string {
|
||||||
|
cmd = "upload-pack"
|
||||||
|
opts = append(opts, "--stateless-rpc", repository_path)
|
||||||
|
return command(cmd, opts...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func receive_pack(repository_path string, opts ...string) string {
|
||||||
|
cmd = "receive-pack"
|
||||||
|
opts = append(opts, "--stateless-rpc", repository_path)
|
||||||
|
return command(cmd, opts...)
|
||||||
|
}*/
|
||||||
|
|
||||||
|
/*func update_server_info(repository_path, opts = {}, &block)
|
||||||
|
cmd = "update-server-info"
|
||||||
|
args = []
|
||||||
|
opts.each {|k,v| args << command_options[k] if command_options.has_key?(k) }
|
||||||
|
opts[:args] = args
|
||||||
|
Dir.chdir(repository_path) do # "git update-server-info" does not take a parameter to specify the repository, so set the working directory to the repository
|
||||||
|
self.command(cmd, opts, &block)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_config_setting(repository_path, key)
|
||||||
|
path = get_config_location(repository_path)
|
||||||
|
raise "Config file could not be found for repository in #{repository_path}." unless path
|
||||||
|
self.command("config", {:args => ["-f #{path}", key]}).chomp
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_config_location(repository_path)
|
||||||
|
non_bare = File.join(repository_path,'.git') # This is where the config file will be if the repository is non-bare
|
||||||
|
if File.exists?(non_bare) then # The repository is non-bare
|
||||||
|
non_bare_config = File.join(non_bare, 'config')
|
||||||
|
return non_bare_config if File.exists?(non_bare_config)
|
||||||
|
else # We are dealing with a bare repository
|
||||||
|
bare_config = File.join(repository_path, "config")
|
||||||
|
return bare_config if File.exists?(bare_config)
|
||||||
|
end
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
*/
|
471
routers/repo/http.go
Normal file
471
routers/repo/http.go
Normal file
|
@ -0,0 +1,471 @@
|
||||||
|
package repo
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-martini/martini"
|
||||||
|
"github.com/gogits/gogs/models"
|
||||||
|
"github.com/gogits/gogs/modules/base"
|
||||||
|
"github.com/gogits/gogs/modules/middleware"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Http(ctx *middleware.Context, params martini.Params) {
|
||||||
|
username := params["username"]
|
||||||
|
reponame := params["reponame"]
|
||||||
|
if strings.HasSuffix(reponame, ".git") {
|
||||||
|
reponame = reponame[:len(reponame)-4]
|
||||||
|
}
|
||||||
|
|
||||||
|
var isPull bool
|
||||||
|
service := ctx.Query("service")
|
||||||
|
if service == "git-receive-pack" ||
|
||||||
|
strings.HasSuffix(ctx.Req.URL.Path, "git-receive-pack") {
|
||||||
|
isPull = false
|
||||||
|
} else if service == "git-upload-pack" ||
|
||||||
|
strings.HasSuffix(ctx.Req.URL.Path, "git-upload-pack") {
|
||||||
|
isPull = true
|
||||||
|
} else {
|
||||||
|
isPull = (ctx.Req.Method == "GET")
|
||||||
|
}
|
||||||
|
|
||||||
|
repoUser, err := models.GetUserByName(username)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Handle(500, "repo.GetUserByName", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
repo, err := models.GetRepositoryByName(repoUser.Id, reponame)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Handle(500, "repo.GetRepositoryByName", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// only public pull don't need auth
|
||||||
|
var askAuth = !(!repo.IsPrivate && isPull)
|
||||||
|
|
||||||
|
// check access
|
||||||
|
if askAuth {
|
||||||
|
baHead := ctx.Req.Header.Get("Authorization")
|
||||||
|
if baHead == "" {
|
||||||
|
// ask auth
|
||||||
|
authRequired(ctx)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
auths := strings.Fields(baHead)
|
||||||
|
// currently check basic auth
|
||||||
|
// TODO: support digit auth
|
||||||
|
if len(auths) != 2 || auths[0] != "Basic" {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
authUsername, passwd, err := basicDecode(auths[1])
|
||||||
|
if err != nil {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
authUser, err := models.GetUserByName(authUsername)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
newUser := &models.User{Passwd: passwd, Salt: authUser.Salt}
|
||||||
|
|
||||||
|
newUser.EncodePasswd()
|
||||||
|
if authUser.Passwd != newUser.Passwd {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var tp = models.AU_WRITABLE
|
||||||
|
if isPull {
|
||||||
|
tp = models.AU_READABLE
|
||||||
|
}
|
||||||
|
|
||||||
|
has, err := models.HasAccess(authUsername, username+"/"+reponame, tp)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
} else if !has {
|
||||||
|
if tp == models.AU_READABLE {
|
||||||
|
has, err = models.HasAccess(authUsername, username+"/"+reponame, models.AU_WRITABLE)
|
||||||
|
if err != nil || !has {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ctx.Handle(401, "no basic auth and digit auth", nil)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
config := Config{base.RepoRootPath, "git", true, true, func(rpc string, input []byte) {
|
||||||
|
//fmt.Println("rpc:", rpc)
|
||||||
|
//fmt.Println("input:", string(input))
|
||||||
|
}}
|
||||||
|
|
||||||
|
handler := HttpBackend(&config)
|
||||||
|
handler(ctx.ResponseWriter, ctx.Req)
|
||||||
|
|
||||||
|
/* Webdav
|
||||||
|
dir := models.RepoPath(username, reponame)
|
||||||
|
|
||||||
|
prefix := path.Join("/", username, params["reponame"])
|
||||||
|
server := webdav.NewServer(
|
||||||
|
dir, prefix, true)
|
||||||
|
|
||||||
|
server.ServeHTTP(ctx.ResponseWriter, ctx.Req)
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
type route struct {
|
||||||
|
cr *regexp.Regexp
|
||||||
|
method string
|
||||||
|
handler func(handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
ReposRoot string
|
||||||
|
GitBinPath string
|
||||||
|
UploadPack bool
|
||||||
|
ReceivePack bool
|
||||||
|
OnSucceed func(rpc string, input []byte)
|
||||||
|
}
|
||||||
|
|
||||||
|
type handler struct {
|
||||||
|
*Config
|
||||||
|
w http.ResponseWriter
|
||||||
|
r *http.Request
|
||||||
|
Dir string
|
||||||
|
File string
|
||||||
|
}
|
||||||
|
|
||||||
|
var routes = []route{
|
||||||
|
{regexp.MustCompile("(.*?)/git-upload-pack$"), "POST", serviceUploadPack},
|
||||||
|
{regexp.MustCompile("(.*?)/git-receive-pack$"), "POST", serviceReceivePack},
|
||||||
|
{regexp.MustCompile("(.*?)/info/refs$"), "GET", getInfoRefs},
|
||||||
|
{regexp.MustCompile("(.*?)/HEAD$"), "GET", getTextFile},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/info/alternates$"), "GET", getTextFile},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/info/http-alternates$"), "GET", getTextFile},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/info/packs$"), "GET", getInfoPacks},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/info/[^/]*$"), "GET", getTextFile},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/[0-9a-f]{2}/[0-9a-f]{38}$"), "GET", getLooseObject},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/pack/pack-[0-9a-f]{40}\\.pack$"), "GET", getPackFile},
|
||||||
|
{regexp.MustCompile("(.*?)/objects/pack/pack-[0-9a-f]{40}\\.idx$"), "GET", getIdxFile},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Request handling function
|
||||||
|
func HttpBackend(config *Config) http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
//log.Printf("%s %s %s %s", r.RemoteAddr, r.Method, r.URL.Path, r.Proto)
|
||||||
|
for _, route := range routes {
|
||||||
|
if m := route.cr.FindStringSubmatch(r.URL.Path); m != nil {
|
||||||
|
if route.method != r.Method {
|
||||||
|
renderMethodNotAllowed(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
file := strings.Replace(r.URL.Path, m[1]+"/", "", 1)
|
||||||
|
dir, err := getGitDir(config, m[1])
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
renderNotFound(w)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
hr := handler{config, w, r, dir, file}
|
||||||
|
route.handler(hr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
renderNotFound(w)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Actual command handling functions
|
||||||
|
|
||||||
|
func serviceUploadPack(hr handler) {
|
||||||
|
serviceRpc("upload-pack", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func serviceReceivePack(hr handler) {
|
||||||
|
serviceRpc("receive-pack", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func serviceRpc(rpc string, hr handler) {
|
||||||
|
w, r, dir := hr.w, hr.r, hr.Dir
|
||||||
|
access := hasAccess(r, hr.Config, dir, rpc, true)
|
||||||
|
|
||||||
|
if access == false {
|
||||||
|
renderNoAccess(w)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
input, _ := ioutil.ReadAll(r.Body)
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", fmt.Sprintf("application/x-git-%s-result", rpc))
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
|
||||||
|
args := []string{rpc, "--stateless-rpc", dir}
|
||||||
|
cmd := exec.Command(hr.Config.GitBinPath, args...)
|
||||||
|
cmd.Dir = dir
|
||||||
|
in, err := cmd.StdinPipe()
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
err = cmd.Start()
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
in.Write(input)
|
||||||
|
io.Copy(w, stdout)
|
||||||
|
cmd.Wait()
|
||||||
|
|
||||||
|
if hr.Config.OnSucceed != nil {
|
||||||
|
hr.Config.OnSucceed(rpc, input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getInfoRefs(hr handler) {
|
||||||
|
w, r, dir := hr.w, hr.r, hr.Dir
|
||||||
|
serviceName := getServiceType(r)
|
||||||
|
access := hasAccess(r, hr.Config, dir, serviceName, false)
|
||||||
|
|
||||||
|
if access {
|
||||||
|
args := []string{serviceName, "--stateless-rpc", "--advertise-refs", "."}
|
||||||
|
refs := gitCommand(hr.Config.GitBinPath, dir, args...)
|
||||||
|
|
||||||
|
hdrNocache(w)
|
||||||
|
w.Header().Set("Content-Type", fmt.Sprintf("application/x-git-%s-advertisement", serviceName))
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write(packetWrite("# service=git-" + serviceName + "\n"))
|
||||||
|
w.Write(packetFlush())
|
||||||
|
w.Write(refs)
|
||||||
|
} else {
|
||||||
|
updateServerInfo(hr.Config.GitBinPath, dir)
|
||||||
|
hdrNocache(w)
|
||||||
|
sendFile("text/plain; charset=utf-8", hr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getInfoPacks(hr handler) {
|
||||||
|
hdrCacheForever(hr.w)
|
||||||
|
sendFile("text/plain; charset=utf-8", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLooseObject(hr handler) {
|
||||||
|
hdrCacheForever(hr.w)
|
||||||
|
sendFile("application/x-git-loose-object", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getPackFile(hr handler) {
|
||||||
|
hdrCacheForever(hr.w)
|
||||||
|
sendFile("application/x-git-packed-objects", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getIdxFile(hr handler) {
|
||||||
|
hdrCacheForever(hr.w)
|
||||||
|
sendFile("application/x-git-packed-objects-toc", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTextFile(hr handler) {
|
||||||
|
hdrNocache(hr.w)
|
||||||
|
sendFile("text/plain", hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logic helping functions
|
||||||
|
|
||||||
|
func sendFile(contentType string, hr handler) {
|
||||||
|
w, r := hr.w, hr.r
|
||||||
|
reqFile := path.Join(hr.Dir, hr.File)
|
||||||
|
|
||||||
|
//fmt.Println("sendFile:", reqFile)
|
||||||
|
|
||||||
|
f, err := os.Stat(reqFile)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
renderNotFound(w)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", contentType)
|
||||||
|
w.Header().Set("Content-Length", fmt.Sprintf("%d", f.Size()))
|
||||||
|
w.Header().Set("Last-Modified", f.ModTime().Format(http.TimeFormat))
|
||||||
|
http.ServeFile(w, r, reqFile)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGitDir(config *Config, filePath string) (string, error) {
|
||||||
|
root := config.ReposRoot
|
||||||
|
|
||||||
|
if root == "" {
|
||||||
|
cwd, err := os.Getwd()
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
root = cwd
|
||||||
|
}
|
||||||
|
|
||||||
|
f := path.Join(root, filePath)
|
||||||
|
if _, err := os.Stat(f); os.IsNotExist(err) {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return f, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getServiceType(r *http.Request) string {
|
||||||
|
serviceType := r.FormValue("service")
|
||||||
|
|
||||||
|
if s := strings.HasPrefix(serviceType, "git-"); !s {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
return strings.Replace(serviceType, "git-", "", 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func hasAccess(r *http.Request, config *Config, dir string, rpc string, checkContentType bool) bool {
|
||||||
|
if checkContentType {
|
||||||
|
if r.Header.Get("Content-Type") != fmt.Sprintf("application/x-git-%s-request", rpc) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !(rpc == "upload-pack" || rpc == "receive-pack") {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if rpc == "receive-pack" {
|
||||||
|
return config.ReceivePack
|
||||||
|
}
|
||||||
|
if rpc == "upload-pack" {
|
||||||
|
return config.UploadPack
|
||||||
|
}
|
||||||
|
|
||||||
|
return getConfigSetting(config.GitBinPath, rpc, dir)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getConfigSetting(gitBinPath, serviceName string, dir string) bool {
|
||||||
|
serviceName = strings.Replace(serviceName, "-", "", -1)
|
||||||
|
setting := getGitConfig(gitBinPath, "http."+serviceName, dir)
|
||||||
|
|
||||||
|
if serviceName == "uploadpack" {
|
||||||
|
return setting != "false"
|
||||||
|
}
|
||||||
|
|
||||||
|
return setting == "true"
|
||||||
|
}
|
||||||
|
|
||||||
|
func getGitConfig(gitBinPath, configName string, dir string) string {
|
||||||
|
args := []string{"config", configName}
|
||||||
|
out := string(gitCommand(gitBinPath, dir, args...))
|
||||||
|
return out[0 : len(out)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
func updateServerInfo(gitBinPath, dir string) []byte {
|
||||||
|
args := []string{"update-server-info"}
|
||||||
|
return gitCommand(gitBinPath, dir, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func gitCommand(gitBinPath, dir string, args ...string) []byte {
|
||||||
|
command := exec.Command(gitBinPath, args...)
|
||||||
|
command.Dir = dir
|
||||||
|
out, err := command.Output()
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTP error response handling functions
|
||||||
|
|
||||||
|
func renderMethodNotAllowed(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Proto == "HTTP/1.1" {
|
||||||
|
w.WriteHeader(http.StatusMethodNotAllowed)
|
||||||
|
w.Write([]byte("Method Not Allowed"))
|
||||||
|
} else {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte("Bad Request"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func renderNotFound(w http.ResponseWriter) {
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
w.Write([]byte("Not Found"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func renderNoAccess(w http.ResponseWriter) {
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
w.Write([]byte("Forbidden"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Packet-line handling function
|
||||||
|
|
||||||
|
func packetFlush() []byte {
|
||||||
|
return []byte("0000")
|
||||||
|
}
|
||||||
|
|
||||||
|
func packetWrite(str string) []byte {
|
||||||
|
s := strconv.FormatInt(int64(len(str)+4), 16)
|
||||||
|
|
||||||
|
if len(s)%4 != 0 {
|
||||||
|
s = strings.Repeat("0", 4-len(s)%4) + s
|
||||||
|
}
|
||||||
|
|
||||||
|
return []byte(s + str)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Header writing functions
|
||||||
|
|
||||||
|
func hdrNocache(w http.ResponseWriter) {
|
||||||
|
w.Header().Set("Expires", "Fri, 01 Jan 1980 00:00:00 GMT")
|
||||||
|
w.Header().Set("Pragma", "no-cache")
|
||||||
|
w.Header().Set("Cache-Control", "no-cache, max-age=0, must-revalidate")
|
||||||
|
}
|
||||||
|
|
||||||
|
func hdrCacheForever(w http.ResponseWriter) {
|
||||||
|
now := time.Now().Unix()
|
||||||
|
expires := now + 31536000
|
||||||
|
w.Header().Set("Date", fmt.Sprintf("%d", now))
|
||||||
|
w.Header().Set("Expires", fmt.Sprintf("%d", expires))
|
||||||
|
w.Header().Set("Cache-Control", "public, max-age=31536000")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Main
|
||||||
|
/*
|
||||||
|
func main() {
|
||||||
|
http.HandleFunc("/", requestHandler())
|
||||||
|
|
||||||
|
err := http.ListenAndServe(":8080", nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal("ListenAndServe: ", err)
|
||||||
|
}
|
||||||
|
}*/
|
|
@ -14,8 +14,6 @@ import (
|
||||||
|
|
||||||
"github.com/go-martini/martini"
|
"github.com/go-martini/martini"
|
||||||
|
|
||||||
"github.com/gogits/webdav"
|
|
||||||
|
|
||||||
"github.com/gogits/gogs/models"
|
"github.com/gogits/gogs/models"
|
||||||
"github.com/gogits/gogs/modules/auth"
|
"github.com/gogits/gogs/modules/auth"
|
||||||
"github.com/gogits/gogs/modules/base"
|
"github.com/gogits/gogs/modules/base"
|
||||||
|
@ -55,6 +53,36 @@ func Create(ctx *middleware.Context, form auth.CreateRepoForm) {
|
||||||
ctx.Handle(200, "repo.Create", err)
|
ctx.Handle(200, "repo.Create", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Mirror(ctx *middleware.Context, form auth.CreateRepoForm) {
|
||||||
|
ctx.Data["Title"] = "Mirror repository"
|
||||||
|
ctx.Data["PageIsNewRepo"] = true // For navbar arrow.
|
||||||
|
|
||||||
|
if ctx.Req.Method == "GET" {
|
||||||
|
ctx.HTML(200, "repo/mirror")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if ctx.HasError() {
|
||||||
|
ctx.HTML(200, "repo/mirror")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := models.CreateRepository(ctx.User, form.RepoName, form.Description,
|
||||||
|
"", form.License, form.Visibility == "private", false)
|
||||||
|
if err == nil {
|
||||||
|
log.Trace("%s Repository created: %s/%s", ctx.Req.RequestURI, ctx.User.LowerName, form.RepoName)
|
||||||
|
ctx.Redirect("/" + ctx.User.Name + "/" + form.RepoName)
|
||||||
|
return
|
||||||
|
} else if err == models.ErrRepoAlreadyExist {
|
||||||
|
ctx.RenderWithErr("Repository name has already been used", "repo/mirror", &form)
|
||||||
|
return
|
||||||
|
} else if err == models.ErrRepoNameIllegal {
|
||||||
|
ctx.RenderWithErr(models.ErrRepoNameIllegal.Error(), "repo/mirror", &form)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Handle(200, "repo.Mirror", err)
|
||||||
|
}
|
||||||
|
|
||||||
func Single(ctx *middleware.Context, params martini.Params) {
|
func Single(ctx *middleware.Context, params martini.Params) {
|
||||||
branchName := ctx.Repo.BranchName
|
branchName := ctx.Repo.BranchName
|
||||||
commitId := ctx.Repo.CommitId
|
commitId := ctx.Repo.CommitId
|
||||||
|
@ -266,89 +294,6 @@ func authRequired(ctx *middleware.Context) {
|
||||||
ctx.HTML(401, fmt.Sprintf("status/401"))
|
ctx.HTML(401, fmt.Sprintf("status/401"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func Http(ctx *middleware.Context, params martini.Params) {
|
|
||||||
username := params["username"]
|
|
||||||
reponame := params["reponame"]
|
|
||||||
if strings.HasSuffix(reponame, ".git") {
|
|
||||||
reponame = reponame[:len(reponame)-4]
|
|
||||||
}
|
|
||||||
|
|
||||||
//fmt.Println("req:", ctx.Req.Header)
|
|
||||||
|
|
||||||
repoUser, err := models.GetUserByName(username)
|
|
||||||
if err != nil {
|
|
||||||
ctx.Handle(500, "repo.GetUserByName", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
repo, err := models.GetRepositoryByName(repoUser.Id, reponame)
|
|
||||||
if err != nil {
|
|
||||||
ctx.Handle(500, "repo.GetRepositoryByName", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
isPull := webdav.IsPullMethod(ctx.Req.Method)
|
|
||||||
var askAuth = !(!repo.IsPrivate && isPull)
|
|
||||||
|
|
||||||
//authRequired(ctx)
|
|
||||||
//return
|
|
||||||
|
|
||||||
// check access
|
|
||||||
if askAuth {
|
|
||||||
// check digit auth
|
|
||||||
|
|
||||||
// check basic auth
|
|
||||||
baHead := ctx.Req.Header.Get("Authorization")
|
|
||||||
if baHead == "" {
|
|
||||||
authRequired(ctx)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
auths := strings.Fields(baHead)
|
|
||||||
if len(auths) != 2 || auths[0] != "Basic" {
|
|
||||||
ctx.Handle(401, "no basic auth and digit auth", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
authUsername, passwd, err := basicDecode(auths[1])
|
|
||||||
if err != nil {
|
|
||||||
ctx.Handle(401, "no basic auth and digit auth", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
authUser, err := models.GetUserByName(authUsername)
|
|
||||||
if err != nil {
|
|
||||||
ctx.Handle(401, "no basic auth and digit auth", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
newUser := &models.User{Passwd: passwd}
|
|
||||||
newUser.EncodePasswd()
|
|
||||||
if authUser.Passwd != newUser.Passwd {
|
|
||||||
ctx.Handle(401, "no basic auth and digit auth", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var tp = models.AU_WRITABLE
|
|
||||||
if isPull {
|
|
||||||
tp = models.AU_READABLE
|
|
||||||
}
|
|
||||||
|
|
||||||
has, err := models.HasAccess(authUsername, username+"/"+reponame, tp)
|
|
||||||
if err != nil || !has {
|
|
||||||
ctx.Handle(401, "no basic auth and digit auth", nil)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dir := models.RepoPath(username, reponame)
|
|
||||||
|
|
||||||
prefix := path.Join("/", username, params["reponame"])
|
|
||||||
server := webdav.NewServer(
|
|
||||||
dir, prefix, true)
|
|
||||||
|
|
||||||
server.ServeHTTP(ctx.ResponseWriter, ctx.Req)
|
|
||||||
}
|
|
||||||
|
|
||||||
func Setting(ctx *middleware.Context, params martini.Params) {
|
func Setting(ctx *middleware.Context, params martini.Params) {
|
||||||
if !ctx.Repo.IsOwner {
|
if !ctx.Repo.IsOwner {
|
||||||
ctx.Handle(404, "repo.Setting", nil)
|
ctx.Handle(404, "repo.Setting", nil)
|
||||||
|
@ -397,6 +342,7 @@ func SettingPost(ctx *middleware.Context) {
|
||||||
|
|
||||||
ctx.Repo.Repository.Description = ctx.Query("desc")
|
ctx.Repo.Repository.Description = ctx.Query("desc")
|
||||||
ctx.Repo.Repository.Website = ctx.Query("site")
|
ctx.Repo.Repository.Website = ctx.Query("site")
|
||||||
|
ctx.Repo.Repository.IsGoget = ctx.Query("goget") == "on"
|
||||||
if err := models.UpdateRepository(ctx.Repo.Repository); err != nil {
|
if err := models.UpdateRepository(ctx.Repo.Repository); err != nil {
|
||||||
ctx.Handle(404, "repo.SettingPost(update)", err)
|
ctx.Handle(404, "repo.SettingPost(update)", err)
|
||||||
return
|
return
|
||||||
|
|
|
@ -6,7 +6,10 @@ package user
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"code.google.com/p/goauth2/oauth"
|
"code.google.com/p/goauth2/oauth"
|
||||||
|
|
||||||
|
@ -70,53 +73,87 @@ func (s *SocialGithub) Update() error {
|
||||||
return json.NewDecoder(r.Body).Decode(&s.data)
|
return json.NewDecoder(r.Body).Decode(&s.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func extractPath(next string) string {
|
||||||
|
n, err := url.Parse(next)
|
||||||
|
if err != nil {
|
||||||
|
return "/"
|
||||||
|
}
|
||||||
|
return n.Path
|
||||||
|
}
|
||||||
|
|
||||||
// github && google && ...
|
// github && google && ...
|
||||||
func SocialSignIn(ctx *middleware.Context, tokens oauth2.Tokens) {
|
func SocialSignIn(ctx *middleware.Context, tokens oauth2.Tokens) {
|
||||||
gh := &SocialGithub{
|
var socid int64
|
||||||
WebToken: &oauth.Token{
|
var ok bool
|
||||||
AccessToken: tokens.Access(),
|
next := extractPath(ctx.Query("next"))
|
||||||
RefreshToken: tokens.Refresh(),
|
log.Debug("social signed check %s", next)
|
||||||
Expiry: tokens.ExpiryTime(),
|
if socid, ok = ctx.Session.Get("socialId").(int64); ok && socid != 0 {
|
||||||
Extra: tokens.ExtraData(),
|
// already login
|
||||||
},
|
ctx.Redirect(next)
|
||||||
}
|
log.Info("login soc id: %v", socid)
|
||||||
if len(tokens.Access()) == 0 {
|
|
||||||
log.Error("empty access")
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
var err error
|
config := &oauth.Config{
|
||||||
var u *models.User
|
//ClientId: base.OauthService.Github.ClientId,
|
||||||
|
//ClientSecret: base.OauthService.Github.ClientSecret, // FIXME: I don't know why compile error here
|
||||||
|
ClientId: "09383403ff2dc16daaa1",
|
||||||
|
ClientSecret: "0e4aa0c3630df396cdcea01a9d45cacf79925fea",
|
||||||
|
RedirectURL: strings.TrimSuffix(base.AppUrl, "/") + ctx.Req.URL.RequestURI(),
|
||||||
|
Scope: base.OauthService.GitHub.Scopes,
|
||||||
|
AuthURL: "https://github.com/login/oauth/authorize",
|
||||||
|
TokenURL: "https://github.com/login/oauth/access_token",
|
||||||
|
}
|
||||||
|
transport := &oauth.Transport{
|
||||||
|
Config: config,
|
||||||
|
Transport: http.DefaultTransport,
|
||||||
|
}
|
||||||
|
code := ctx.Query("code")
|
||||||
|
if code == "" {
|
||||||
|
// redirect to social login page
|
||||||
|
ctx.Redirect(config.AuthCodeURL(next))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle call back
|
||||||
|
tk, err := transport.Exchange(code)
|
||||||
|
if err != nil {
|
||||||
|
log.Error("oauth2 handle callback error: %v", err)
|
||||||
|
return // FIXME, need error page 501
|
||||||
|
}
|
||||||
|
next = extractPath(ctx.Query("state"))
|
||||||
|
log.Debug("success token: %v", tk)
|
||||||
|
|
||||||
|
gh := &SocialGithub{WebToken: tk}
|
||||||
if err = gh.Update(); err != nil {
|
if err = gh.Update(); err != nil {
|
||||||
// FIXME: handle error page
|
// FIXME: handle error page 501
|
||||||
log.Error("connect with github error: %s", err)
|
log.Error("connect with github error: %s", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
var soc SocialConnector = gh
|
var soc SocialConnector = gh
|
||||||
log.Info("login: %s", soc.Name())
|
log.Info("login: %s", soc.Name())
|
||||||
// FIXME: login here, user email to check auth, if not registe, then generate a uniq username
|
oa, err := models.GetOauth2(soc.Identity())
|
||||||
if u, err = models.GetOauth2User(soc.Identity()); err != nil {
|
switch err {
|
||||||
u = &models.User{
|
case nil:
|
||||||
Name: soc.Name(),
|
ctx.Session.Set("userId", oa.User.Id)
|
||||||
Email: soc.Email(),
|
ctx.Session.Set("userName", oa.User.Name)
|
||||||
Passwd: "123456",
|
case models.ErrOauth2RecordNotExists:
|
||||||
IsActive: !base.Service.RegisterEmailConfirm,
|
oa = &models.Oauth2{}
|
||||||
}
|
oa.Uid = 0
|
||||||
if u, err = models.RegisterUser(u); err != nil {
|
|
||||||
log.Error("register user: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
oa := &models.Oauth2{}
|
|
||||||
oa.Uid = u.Id
|
|
||||||
oa.Type = soc.Type()
|
oa.Type = soc.Type()
|
||||||
oa.Token = soc.Token()
|
oa.Token = soc.Token()
|
||||||
oa.Identity = soc.Identity()
|
oa.Identity = soc.Identity()
|
||||||
log.Info("oa: %v", oa)
|
log.Debug("oa: %v", oa)
|
||||||
if err = models.AddOauth2(oa); err != nil {
|
if err = models.AddOauth2(oa); err != nil {
|
||||||
log.Error("add oauth2 %v", err)
|
log.Error("add oauth2 %v", err) // 501
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
case models.ErrOauth2NotAssociatedWithUser:
|
||||||
|
// ignore it. judge in /usr/login page
|
||||||
|
default:
|
||||||
|
log.Error(err.Error()) // FIXME: handle error page
|
||||||
|
return
|
||||||
}
|
}
|
||||||
ctx.Session.Set("userId", u.Id)
|
ctx.Session.Set("socialId", oa.Id)
|
||||||
ctx.Session.Set("userName", u.Name)
|
log.Debug("socialId: %v", oa.Id)
|
||||||
ctx.Redirect("/")
|
ctx.Redirect(next)
|
||||||
}
|
}
|
||||||
|
|
|
@ -396,6 +396,10 @@ func Activate(ctx *middleware.Context) {
|
||||||
} else {
|
} else {
|
||||||
ctx.Data["Hours"] = base.Service.ActiveCodeLives / 60
|
ctx.Data["Hours"] = base.Service.ActiveCodeLives / 60
|
||||||
mailer.SendActiveMail(ctx.Render, ctx.User)
|
mailer.SendActiveMail(ctx.Render, ctx.User)
|
||||||
|
|
||||||
|
if err := ctx.Cache.Put("MailResendLimit_"+ctx.User.LowerName, ctx.User.LowerName, 180); err != nil {
|
||||||
|
log.Error("Set cache(MailResendLimit) fail: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ctx.Data["ServiceNotEnabled"] = true
|
ctx.Data["ServiceNotEnabled"] = true
|
||||||
|
@ -451,7 +455,17 @@ func ForgotPasswd(ctx *middleware.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ctx.Cache.IsExist("MailResendLimit_" + u.LowerName) {
|
||||||
|
ctx.Data["ResendLimited"] = true
|
||||||
|
ctx.HTML(200, "user/forgot_passwd")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
mailer.SendResetPasswdMail(ctx.Render, u)
|
mailer.SendResetPasswdMail(ctx.Render, u)
|
||||||
|
if err = ctx.Cache.Put("MailResendLimit_"+u.LowerName, u.LowerName, 180); err != nil {
|
||||||
|
log.Error("Set cache(MailResendLimit) fail: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
ctx.Data["Email"] = email
|
ctx.Data["Email"] = email
|
||||||
ctx.Data["Hours"] = base.Service.ActiveCodeLives / 60
|
ctx.Data["Hours"] = base.Service.ActiveCodeLives / 60
|
||||||
ctx.Data["IsResetSent"] = true
|
ctx.Data["IsResetSent"] = true
|
||||||
|
|
5
serve.go
5
serve.go
|
@ -177,10 +177,7 @@ func runServ(k *cli.Context) {
|
||||||
qlog.Fatal("Unknown command")
|
qlog.Fatal("Unknown command")
|
||||||
}
|
}
|
||||||
|
|
||||||
// for update use
|
models.SetRepoEnvs(user.Id, user.Name, repoName)
|
||||||
os.Setenv("userName", user.Name)
|
|
||||||
os.Setenv("userId", strconv.Itoa(int(user.Id)))
|
|
||||||
os.Setenv("repoName", repoName)
|
|
||||||
|
|
||||||
gitcmd := exec.Command(verb, repoPath)
|
gitcmd := exec.Command(verb, repoPath)
|
||||||
gitcmd.Dir = base.RepoRootPath
|
gitcmd.Dir = base.RepoRootPath
|
||||||
|
|
|
@ -9,16 +9,27 @@
|
||||||
<meta name="description" content="Gogs(Go Git Service) is a GitHub-like clone in the Go Programming Language" />
|
<meta name="description" content="Gogs(Go Git Service) is a GitHub-like clone in the Go Programming Language" />
|
||||||
<meta name="keywords" content="go, git">
|
<meta name="keywords" content="go, git">
|
||||||
<meta name="_csrf" content="{{.CsrfToken}}" />
|
<meta name="_csrf" content="{{.CsrfToken}}" />
|
||||||
|
{{if .Repository.IsGoget}}<meta name="go-import" content="{{AppDomain}} git {{.CloneLink.HTTPS}}">{{end}}
|
||||||
|
|
||||||
<!-- Stylesheets -->
|
<!-- Stylesheets -->
|
||||||
|
{{if IsProdMode}}
|
||||||
|
<link rel="stylesheet" href="//netdna.bootstrapcdn.com/bootstrap/3.1.1/css/bootstrap.min.css">
|
||||||
|
<link href="//netdna.bootstrapcdn.com/font-awesome/4.0.3/css/font-awesome.css" rel="stylesheet">
|
||||||
|
|
||||||
|
<script src="//code.jquery.com/jquery-1.11.0.min.js"></script>
|
||||||
|
<script src="//netdna.bootstrapcdn.com/bootstrap/3.1.1/js/bootstrap.min.js"></script>
|
||||||
|
{{else}}
|
||||||
<link href="/css/bootstrap.min.css" rel="stylesheet" />
|
<link href="/css/bootstrap.min.css" rel="stylesheet" />
|
||||||
<link href="/css/todc-bootstrap.min.css" rel="stylesheet" />
|
|
||||||
<link href="/css/font-awesome.min.css" rel="stylesheet" />
|
<link href="/css/font-awesome.min.css" rel="stylesheet" />
|
||||||
<link href="/css/markdown.css" rel="stylesheet" />
|
|
||||||
<link href="/css/gogs.css" rel="stylesheet" />
|
|
||||||
|
|
||||||
<script src="/js/jquery-1.10.1.min.js"></script>
|
<script src="/js/jquery-1.10.1.min.js"></script>
|
||||||
<script src="/js/bootstrap.min.js"></script>
|
<script src="/js/bootstrap.min.js"></script>
|
||||||
|
{{end}}
|
||||||
|
|
||||||
|
<link href="/css/todc-bootstrap.min.css" rel="stylesheet" />
|
||||||
|
<link href="/css/markdown.css" rel="stylesheet" />
|
||||||
|
<link href="/css/gogs.css" rel="stylesheet" />
|
||||||
|
|
||||||
<script src="/js/lib.js"></script>
|
<script src="/js/lib.js"></script>
|
||||||
<script src="/js/app.js"></script>
|
<script src="/js/app.js"></script>
|
||||||
<title>{{if .Title}}{{.Title}} - {{end}}{{AppName}}</title>
|
<title>{{if .Title}}{{.Title}} - {{end}}{{AppName}}</title>
|
||||||
|
|
|
@ -8,9 +8,18 @@
|
||||||
<a id="nav-avatar" class="nav-item navbar-right{{if .PageIsUserProfile}} active{{end}}" href="{{.SignedUser.HomeLink}}" data-toggle="tooltip" data-placement="bottom" title="{{.SignedUserName}}">
|
<a id="nav-avatar" class="nav-item navbar-right{{if .PageIsUserProfile}} active{{end}}" href="{{.SignedUser.HomeLink}}" data-toggle="tooltip" data-placement="bottom" title="{{.SignedUserName}}">
|
||||||
<img src="{{.SignedUser.AvatarLink}}?s=28" alt="user-avatar" title="username"/>
|
<img src="{{.SignedUser.AvatarLink}}?s=28" alt="user-avatar" title="username"/>
|
||||||
</a>
|
</a>
|
||||||
<a class="navbar-right nav-item{{if .PageIsNewRepo}} active{{end}}" href="/repo/create" data-toggle="tooltip" data-placement="bottom" title="New Repository"><i class="fa fa-plus fa-lg"></i></a>
|
|
||||||
<a class="navbar-right nav-item{{if .PageIsUserSetting}} active{{end}}" href="/user/setting" data-toggle="tooltip" data-placement="bottom" title="Setting"><i class="fa fa-cogs fa-lg"></i></a>
|
<a class="navbar-right nav-item{{if .PageIsUserSetting}} active{{end}}" href="/user/setting" data-toggle="tooltip" data-placement="bottom" title="Setting"><i class="fa fa-cogs fa-lg"></i></a>
|
||||||
{{if .IsAdmin}}<a class="navbar-right nav-item{{if .PageIsAdmin}} active{{end}}" href="/admin" data-toggle="tooltip" data-placement="bottom" title="Admin"><i class="fa fa-gear fa-lg"></i></a>{{end}}
|
{{if .IsAdmin}}<a class="navbar-right nav-item{{if .PageIsAdmin}} active{{end}}" href="/admin" data-toggle="tooltip" data-placement="bottom" title="Admin"><i class="fa fa-gear fa-lg"></i></a>{{end}}
|
||||||
|
<div class="navbar-right nav-item pull-right{{if .PageIsNewRepo}} active{{end}}" id="nav-repo-new" data-toggle="tooltip" data-placement="bottom" title="New Repo">
|
||||||
|
<button type="button" class="dropdown-toggle" data-toggle="dropdown"><i class="fa fa-plus-square fa-lg"></i></button>
|
||||||
|
<div class="dropdown-menu">
|
||||||
|
<ul class="list-unstyled">
|
||||||
|
<li><a href="/repo/create"><i class="fa fa-book"></i>Repository</a></li>
|
||||||
|
<li><a href="/repo/mirror"><i class="fa fa-clipboard"></i>Mirror</a></li>
|
||||||
|
<li><a href="#"><i class="fa fa-users"></i>Organization</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{{else}}<a id="nav-signin" class="nav-item navbar-right navbar-btn btn btn-danger" href="/user/login/">Sign In</a>
|
{{else}}<a id="nav-signin" class="nav-item navbar-right navbar-btn btn btn-danger" href="/user/login/">Sign In</a>
|
||||||
<a id="nav-signup" class="nav-item navbar-right" href="/user/sign_up/">Sign Up</a>{{end}}
|
<a id="nav-signup" class="nav-item navbar-right" href="/user/sign_up/">Sign Up</a>{{end}}
|
||||||
</nav>
|
</nav>
|
||||||
|
|
|
@ -156,11 +156,11 @@
|
||||||
<label class="col-md-3 control-label">SMTP Host: </label>
|
<label class="col-md-3 control-label">SMTP Host: </label>
|
||||||
|
|
||||||
<div class="col-md-8">
|
<div class="col-md-8">
|
||||||
<input name="smtp_host" type="text" class="form-control" placeholder="Type SMTP host address" value="{{.smtp_host}}">
|
<input name="smtp_host" type="text" class="form-control" placeholder="Type SMTP host address and port" value="{{.smtp_host}}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="col-md-3 control-label">Email: </label>
|
<label class="col-md-3 control-label">Username: </label>
|
||||||
|
|
||||||
<div class="col-md-8">
|
<div class="col-md-8">
|
||||||
<input name="mailer_user" type="text" class="form-control" placeholder="Type SMTP user e-mail address" value="{{.mailer_user}}">
|
<input name="mailer_user" type="text" class="form-control" placeholder="Type SMTP user e-mail address" value="{{.mailer_user}}">
|
||||||
|
|
81
templates/repo/mirror.tmpl
Normal file
81
templates/repo/mirror.tmpl
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
{{template "base/head" .}}
|
||||||
|
{{template "base/navbar" .}}
|
||||||
|
<div class="container" id="body">
|
||||||
|
<form action="/repo/create" method="post" class="form-horizontal card" id="repo-create">
|
||||||
|
{{.CsrfTokenHtml}}
|
||||||
|
<h3>Create Repository Mirror</h3>
|
||||||
|
<div class="alert alert-danger form-error{{if .HasError}}{{else}} hidden{{end}}">{{.ErrorMsg}}</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">From<strong class="text-danger">*</strong></label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<select class="form-control" name="from">
|
||||||
|
<option value="">GitHub</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">URL<strong class="text-danger">*</strong></label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<input name="url" type="text" class="form-control" placeholder="Type your mirror repository url link" required="required">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="col-md-offset-2 col-md-8">
|
||||||
|
<a class="btn btn-default" data-toggle="collapse" data-target="#repo-import-auth">Need Authorization</a>
|
||||||
|
</div>
|
||||||
|
<div id="repo-import-auth" class="collapse">
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">Username</label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<input name="auth-username" type="text" class="form-control">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">Password</label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<input name="auth-password" type="text" class="form-control">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<hr/>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">Owner<strong class="text-danger">*</strong></label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<p class="form-control-static">{{.SignedUserName}}</p>
|
||||||
|
<input type="hidden" value="{{.SignedUserId}}" name="userId"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group {{if .Err_RepoName}}has-error has-feedback{{end}}">
|
||||||
|
<label class="col-md-2 control-label">Repository<strong class="text-danger">*</strong></label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<input name="repo" type="text" class="form-control" placeholder="Type your repository name" value="{{.repo}}" required="required">
|
||||||
|
<span class="help-block">Great repository names are short and memorable. </span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="col-md-2 control-label">Visibility<strong class="text-danger">*</strong></label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<p class="form-control-static">Public</p>
|
||||||
|
<input type="hidden" value="public" name="visibility"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group {{if .Err_Description}}has-error has-feedback{{end}}">
|
||||||
|
<label class="col-md-2 control-label">Description</label>
|
||||||
|
<div class="col-md-8">
|
||||||
|
<textarea name="desc" class="form-control" placeholder="Type your repository description">{{.desc}}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="col-md-offset-2 col-md-8">
|
||||||
|
<button type="submit" class="btn btn-lg btn-primary">Mirror repository</button>
|
||||||
|
<a href="/" class="text-danger">Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
{{template "base/footer" .}}
|
|
@ -43,6 +43,7 @@
|
||||||
<input type="url" class="form-control" name="site" value="{{.Repository.Website}}" />
|
<input type="url" class="form-control" name="site" value="{{.Repository.Website}}" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<hr>
|
||||||
<!-- <div class="form-group">
|
<!-- <div class="form-group">
|
||||||
<label class="col-md-3 text-right">Default Branch</label>
|
<label class="col-md-3 text-right">Default Branch</label>
|
||||||
<div class="col-md-9">
|
<div class="col-md-9">
|
||||||
|
@ -51,6 +52,18 @@
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div> -->
|
</div> -->
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="col-md-offset-3 col-md-9">
|
||||||
|
<div class="checkbox">
|
||||||
|
<label style="line-height: 15px;">
|
||||||
|
<input type="checkbox" name="goget" {{if .Repository.IsGoget}}checked{{end}}>
|
||||||
|
<strong>Enable 'go get' meta</strong>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<div class="col-md-9 col-md-offset-3">
|
<div class="col-md-9 col-md-offset-3">
|
||||||
<button class="btn btn-primary" type="submit">Save Options</button>
|
<button class="btn btn-primary" type="submit">Save Options</button>
|
||||||
|
|
|
@ -9,6 +9,20 @@
|
||||||
<h4>Quick Guide</h4>
|
<h4>Quick Guide</h4>
|
||||||
</div>
|
</div>
|
||||||
<div class="panel-body guide-content text-center">
|
<div class="panel-body guide-content text-center">
|
||||||
|
<form action="{{.RepoLink}}/import" method="post">
|
||||||
|
{{.CsrfTokenHtml}}
|
||||||
|
<h3>Clone from existing repository</h3>
|
||||||
|
<div class="input-group col-md-6 col-md-offset-3">
|
||||||
|
<span class="input-group-btn">
|
||||||
|
<button class="btn btn-default" type="button">URL</button>
|
||||||
|
</span>
|
||||||
|
<input name="passwd" type="password" class="form-control" placeholder="Type existing repository address" required="required">
|
||||||
|
<span class="input-group-btn">
|
||||||
|
<button type="submit" class="btn btn-default" type="button">Clone</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
<h3>Clone this repository</h3>
|
<h3>Clone this repository</h3>
|
||||||
<div class="input-group col-md-8 col-md-offset-2 guide-buttons">
|
<div class="input-group col-md-8 col-md-offset-2 guide-buttons">
|
||||||
<span class="input-group-btn">
|
<span class="input-group-btn">
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
<li class="{{if .IsRepoToolbarIssues}}active{{end}}"><a href="{{.RepoLink}}/issues">{{if .Repository.NumOpenIssues}}<span class="badge">{{.Repository.NumOpenIssues}}</span> {{end}}Issues <!--<span class="badge">42</span>--></a></li>
|
<li class="{{if .IsRepoToolbarIssues}}active{{end}}"><a href="{{.RepoLink}}/issues">{{if .Repository.NumOpenIssues}}<span class="badge">{{.Repository.NumOpenIssues}}</span> {{end}}Issues <!--<span class="badge">42</span>--></a></li>
|
||||||
{{if .IsRepoToolbarIssues}}
|
{{if .IsRepoToolbarIssues}}
|
||||||
<li class="tmp">{{if .IsRepoToolbarIssuesList}}<a href="{{.RepoLink}}/issues/new"><button class="btn btn-primary btn-sm">New Issue</button>
|
<li class="tmp">{{if .IsRepoToolbarIssuesList}}<a href="{{.RepoLink}}/issues/new"><button class="btn btn-primary btn-sm">New Issue</button>
|
||||||
</a>{{else}}<a href="{{.RepoLink}}/issues"><button class="btn btn-primary btn-sm">Issues List</button></a>{{end}}</li>
|
</a>{{end}}</li>
|
||||||
{{end}}
|
{{end}}
|
||||||
<li class="{{if .IsRepoToolbarReleases}}active{{end}}"><a href="{{.RepoLink}}/releases">{{if .Repository.NumReleases}}<span class="badge">{{.Repository.NumReleases}}</span> {{end}}Releases</a></li>
|
<li class="{{if .IsRepoToolbarReleases}}active{{end}}"><a href="{{.RepoLink}}/releases">{{if .Repository.NumReleases}}<span class="badge">{{.Repository.NumReleases}}</span> {{end}}Releases</a></li>
|
||||||
{{if .IsRepoToolbarReleases}}
|
{{if .IsRepoToolbarReleases}}
|
||||||
|
|
|
@ -29,7 +29,16 @@
|
||||||
<div id="feed-right" class="col-md-4">
|
<div id="feed-right" class="col-md-4">
|
||||||
<div class="panel panel-default repo-panel">
|
<div class="panel panel-default repo-panel">
|
||||||
<div class="panel-heading">Your Repositories
|
<div class="panel-heading">Your Repositories
|
||||||
<a class="btn btn-success pull-right btn-sm" href="/repo/create"><i class="fa fa-plus-square"></i>New Repo</a>
|
<div class="btn-group pull-right" id="user-dashboard-repo-new">
|
||||||
|
<button type="button" class="btn btn-success btn-sm dropdown-toggle" data-toggle="dropdown"><i class="fa fa-plus-square"></i>New</button>
|
||||||
|
<div class="dropdown-menu dropdown-menu-right">
|
||||||
|
<ul class="list-unstyled">
|
||||||
|
<li><a href="/repo/create"><i class="fa fa-book"></i>Repository</a></li>
|
||||||
|
<li><a href="/repo/mirror"><i class="fa fa-clipboard"></i>Mirror</a></li>
|
||||||
|
<li><a href="#"><i class="fa fa-users"></i>Organization</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="panel-body">
|
<div class="panel-body">
|
||||||
<ul class="list-group">{{range .MyRepos}}
|
<ul class="list-group">{{range .MyRepos}}
|
||||||
|
|
|
@ -24,6 +24,8 @@
|
||||||
</div>
|
</div>
|
||||||
{{else if .IsResetDisable}}
|
{{else if .IsResetDisable}}
|
||||||
<p>Sorry, mail service is not enabled.</p>
|
<p>Sorry, mail service is not enabled.</p>
|
||||||
|
{{else if .ResendLimited}}
|
||||||
|
<p>Sorry, you are sending e-mail too frequently, please wait 3 minutes.</p>
|
||||||
{{end}}
|
{{end}}
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
56
update.go
56
update.go
|
@ -42,32 +42,7 @@ func newUpdateLogger(execDir string) {
|
||||||
qlog.Info("Start logging update...")
|
qlog.Info("Start logging update...")
|
||||||
}
|
}
|
||||||
|
|
||||||
// for command: ./gogs update
|
func update(refName, oldCommitId, newCommitId string) {
|
||||||
func runUpdate(c *cli.Context) {
|
|
||||||
execDir, _ := base.ExecDir()
|
|
||||||
newUpdateLogger(execDir)
|
|
||||||
|
|
||||||
base.NewConfigContext()
|
|
||||||
models.LoadModelsConfig()
|
|
||||||
|
|
||||||
if models.UseSQLite3 {
|
|
||||||
os.Chdir(execDir)
|
|
||||||
}
|
|
||||||
|
|
||||||
models.SetEngine()
|
|
||||||
|
|
||||||
args := c.Args()
|
|
||||||
if len(args) != 3 {
|
|
||||||
qlog.Fatal("received less 3 parameters")
|
|
||||||
}
|
|
||||||
|
|
||||||
refName := args[0]
|
|
||||||
if refName == "" {
|
|
||||||
qlog.Fatal("refName is empty, shouldn't use")
|
|
||||||
}
|
|
||||||
oldCommitId := args[1]
|
|
||||||
newCommitId := args[2]
|
|
||||||
|
|
||||||
isNew := strings.HasPrefix(oldCommitId, "0000000")
|
isNew := strings.HasPrefix(oldCommitId, "0000000")
|
||||||
if isNew &&
|
if isNew &&
|
||||||
strings.HasPrefix(newCommitId, "0000000") {
|
strings.HasPrefix(newCommitId, "0000000") {
|
||||||
|
@ -158,3 +133,32 @@ func runUpdate(c *cli.Context) {
|
||||||
qlog.Fatalf("runUpdate.models.CommitRepoAction: %v", err)
|
qlog.Fatalf("runUpdate.models.CommitRepoAction: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// for command: ./gogs update
|
||||||
|
func runUpdate(c *cli.Context) {
|
||||||
|
execDir, _ := base.ExecDir()
|
||||||
|
newUpdateLogger(execDir)
|
||||||
|
|
||||||
|
base.NewConfigContext()
|
||||||
|
models.LoadModelsConfig()
|
||||||
|
|
||||||
|
if models.UseSQLite3 {
|
||||||
|
os.Chdir(execDir)
|
||||||
|
}
|
||||||
|
|
||||||
|
models.SetEngine()
|
||||||
|
|
||||||
|
args := c.Args()
|
||||||
|
if len(args) != 3 {
|
||||||
|
qlog.Fatal("received less 3 parameters")
|
||||||
|
}
|
||||||
|
|
||||||
|
refName := args[0]
|
||||||
|
if refName == "" {
|
||||||
|
qlog.Fatal("refName is empty, shouldn't use")
|
||||||
|
}
|
||||||
|
oldCommitId := args[1]
|
||||||
|
newCommitId := args[2]
|
||||||
|
|
||||||
|
update(refName, oldCommitId, newCommitId)
|
||||||
|
}
|
||||||
|
|
12
web.go
12
web.go
|
@ -11,10 +11,10 @@ import (
|
||||||
|
|
||||||
"github.com/codegangsta/cli"
|
"github.com/codegangsta/cli"
|
||||||
"github.com/go-martini/martini"
|
"github.com/go-martini/martini"
|
||||||
|
|
||||||
qlog "github.com/qiniu/log"
|
qlog "github.com/qiniu/log"
|
||||||
|
|
||||||
"github.com/gogits/binding"
|
"github.com/gogits/binding"
|
||||||
|
|
||||||
"github.com/gogits/gogs/modules/auth"
|
"github.com/gogits/gogs/modules/auth"
|
||||||
"github.com/gogits/gogs/modules/avatar"
|
"github.com/gogits/gogs/modules/avatar"
|
||||||
"github.com/gogits/gogs/modules/base"
|
"github.com/gogits/gogs/modules/base"
|
||||||
|
@ -72,6 +72,11 @@ func runWeb(*cli.Context) {
|
||||||
|
|
||||||
reqSignIn := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: true})
|
reqSignIn := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: true})
|
||||||
ignSignIn := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: base.Service.RequireSignInView})
|
ignSignIn := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: base.Service.RequireSignInView})
|
||||||
|
ignSignInAndCsrf := middleware.Toggle(&middleware.ToggleOptions{
|
||||||
|
SignInRequire: base.Service.RequireSignInView,
|
||||||
|
DisableCsrf: true,
|
||||||
|
})
|
||||||
|
|
||||||
reqSignOut := middleware.Toggle(&middleware.ToggleOptions{SignOutRequire: true})
|
reqSignOut := middleware.Toggle(&middleware.ToggleOptions{SignOutRequire: true})
|
||||||
|
|
||||||
// Routers.
|
// Routers.
|
||||||
|
@ -91,7 +96,7 @@ func runWeb(*cli.Context) {
|
||||||
|
|
||||||
m.Group("/user", func(r martini.Router) {
|
m.Group("/user", func(r martini.Router) {
|
||||||
r.Any("/login", binding.BindIgnErr(auth.LogInForm{}), user.SignIn)
|
r.Any("/login", binding.BindIgnErr(auth.LogInForm{}), user.SignIn)
|
||||||
r.Any("/login/github", oauth2.LoginRequired, user.SocialSignIn)
|
r.Any("/login/github", user.SocialSignIn)
|
||||||
r.Any("/sign_up", binding.BindIgnErr(auth.RegisterForm{}), user.SignUp)
|
r.Any("/sign_up", binding.BindIgnErr(auth.RegisterForm{}), user.SignUp)
|
||||||
r.Any("/forget_password", user.ForgotPasswd)
|
r.Any("/forget_password", user.ForgotPasswd)
|
||||||
r.Any("/reset_password", user.ResetPasswd)
|
r.Any("/reset_password", user.ResetPasswd)
|
||||||
|
@ -116,6 +121,7 @@ func runWeb(*cli.Context) {
|
||||||
m.Get("/user/:username", ignSignIn, user.Profile)
|
m.Get("/user/:username", ignSignIn, user.Profile)
|
||||||
|
|
||||||
m.Any("/repo/create", reqSignIn, binding.BindIgnErr(auth.CreateRepoForm{}), repo.Create)
|
m.Any("/repo/create", reqSignIn, binding.BindIgnErr(auth.CreateRepoForm{}), repo.Create)
|
||||||
|
m.Any("/repo/mirror", reqSignIn, binding.BindIgnErr(auth.CreateRepoForm{}), repo.Mirror)
|
||||||
|
|
||||||
adminReq := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: true, AdminRequire: true})
|
adminReq := middleware.Toggle(&middleware.ToggleOptions{SignInRequire: true, AdminRequire: true})
|
||||||
|
|
||||||
|
@ -165,7 +171,7 @@ func runWeb(*cli.Context) {
|
||||||
m.Group("/:username", func(r martini.Router) {
|
m.Group("/:username", func(r martini.Router) {
|
||||||
r.Any("/:reponame/**", repo.Http)
|
r.Any("/:reponame/**", repo.Http)
|
||||||
r.Get("/:reponame", middleware.RepoAssignment(true, true, true), repo.Single)
|
r.Get("/:reponame", middleware.RepoAssignment(true, true, true), repo.Single)
|
||||||
}, ignSignIn)
|
}, ignSignInAndCsrf)
|
||||||
|
|
||||||
// Not found handler.
|
// Not found handler.
|
||||||
m.NotFound(routers.NotFound)
|
m.NotFound(routers.NotFound)
|
||||||
|
|
Reference in a new issue