commit
						8faa0dbcd7
					
				|  | @ -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 |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -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 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ##### 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 托管服务。 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ##### 当前版本: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"} | ||||||
|  |  | ||||||
|  | @ -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 | ||||||
|  | */ | ||||||
|  | @ -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}}"> | ||||||
|  |  | ||||||
|  | @ -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) | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue