2014-02-19 13:50:53 +04:00
// 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.
2014-05-02 05:21:46 +04:00
package cmd
2014-02-19 13:50:53 +04:00
import (
2014-12-11 13:14:41 +03:00
"crypto/tls"
2014-02-19 13:50:53 +04:00
"fmt"
2014-05-28 08:06:31 +04:00
"io/ioutil"
2016-08-12 00:46:33 +03:00
"net"
2014-02-19 13:50:53 +04:00
"net/http"
2014-11-04 04:46:53 +03:00
"net/http/fcgi"
2014-04-16 04:01:20 +04:00
"os"
2014-05-26 04:11:25 +04:00
"path"
2014-09-29 13:38:46 +04:00
"strings"
2014-02-19 13:50:53 +04:00
2015-10-16 04:28:12 +03:00
"github.com/go-macaron/binding"
"github.com/go-macaron/cache"
"github.com/go-macaron/captcha"
"github.com/go-macaron/csrf"
"github.com/go-macaron/gzip"
"github.com/go-macaron/i18n"
"github.com/go-macaron/session"
"github.com/go-macaron/toolbox"
2015-08-06 18:03:04 +03:00
"github.com/go-xorm/xorm"
"github.com/mcuadros/go-version"
2016-08-30 14:57:58 +03:00
"github.com/urfave/cli"
2015-01-02 15:14:43 +03:00
"gopkg.in/ini.v1"
2015-10-16 04:28:12 +03:00
"gopkg.in/macaron.v1"
2014-02-19 13:50:53 +04:00
2015-12-16 01:25:45 +03:00
"github.com/gogits/git-module"
2016-01-06 08:34:34 +03:00
"github.com/gogits/go-gogs-client"
2015-12-10 04:46:05 +03:00
2016-11-03 15:29:56 +03:00
"github.com/go-gitea/gitea/models"
"github.com/go-gitea/gitea/modules/auth"
"github.com/go-gitea/gitea/modules/bindata"
"github.com/go-gitea/gitea/modules/context"
"github.com/go-gitea/gitea/modules/log"
"github.com/go-gitea/gitea/modules/setting"
"github.com/go-gitea/gitea/modules/template"
"github.com/go-gitea/gitea/routers"
"github.com/go-gitea/gitea/routers/admin"
apiv1 "github.com/go-gitea/gitea/routers/api/v1"
"github.com/go-gitea/gitea/routers/dev"
"github.com/go-gitea/gitea/routers/org"
"github.com/go-gitea/gitea/routers/repo"
"github.com/go-gitea/gitea/routers/user"
2014-02-19 13:50:53 +04:00
)
var CmdWeb = cli . Command {
Name : "web" ,
2014-05-02 05:21:46 +04:00
Usage : "Start Gogs web server" ,
2014-08-30 17:12:53 +04:00
Description : ` Gogs web server is the only thing you need to run ,
2014-03-24 15:36:38 +04:00
and it takes care of all the other things for you ` ,
2014-02-19 13:50:53 +04:00
Action : runWeb ,
2015-02-01 20:41:03 +03:00
Flags : [ ] cli . Flag {
2015-11-16 01:07:44 +03:00
stringFlag ( "port, p" , "3000" , "Temporary port number to prevent conflict" ) ,
stringFlag ( "config, c" , "custom/conf/app.ini" , "Custom configuration file path" ) ,
2015-02-01 20:41:03 +03:00
} ,
2014-02-19 13:50:53 +04:00
}
2014-12-15 09:49:59 +03:00
type VerChecker struct {
ImportPath string
Version func ( ) string
Expected string
}
2014-08-24 17:09:05 +04:00
// checkVersion checks if binary matches the version of templates files.
2014-05-26 04:57:01 +04:00
func checkVersion ( ) {
2014-09-28 09:38:25 +04:00
// Templates.
2015-08-02 06:36:18 +03:00
data , err := ioutil . ReadFile ( setting . StaticRootPath + "/templates/.VERSION" )
2014-05-26 04:57:01 +04:00
if err != nil {
2014-07-26 08:24:27 +04:00
log . Fatal ( 4 , "Fail to read 'templates/.VERSION': %v" , err )
2014-05-26 04:57:01 +04:00
}
2016-08-31 14:31:53 +03:00
tplVer := string ( data )
if tplVer != setting . AppVer {
if version . Compare ( tplVer , setting . AppVer , ">" ) {
log . Fatal ( 4 , "Binary version is lower than template file version, did you forget to recompile Gogs?" )
} else {
log . Fatal ( 4 , "Binary version is higher than template file version, did you forget to update template files?" )
}
2014-05-26 04:57:01 +04:00
}
2014-09-28 09:38:25 +04:00
2014-10-07 03:12:52 +04:00
// Check dependency version.
2014-12-15 09:49:59 +03:00
checkers := [ ] VerChecker {
2016-07-15 19:36:39 +03:00
{ "github.com/go-xorm/xorm" , func ( ) string { return xorm . Version } , "0.5.5" } ,
2016-07-12 02:07:57 +03:00
{ "github.com/go-macaron/binding" , binding . Version , "0.3.2" } ,
2015-11-03 20:16:43 +03:00
{ "github.com/go-macaron/cache" , cache . Version , "0.1.2" } ,
2016-03-13 04:56:03 +03:00
{ "github.com/go-macaron/csrf" , csrf . Version , "0.1.0" } ,
2016-06-12 12:54:22 +03:00
{ "github.com/go-macaron/i18n" , i18n . Version , "0.3.0" } ,
2015-11-03 20:16:43 +03:00
{ "github.com/go-macaron/session" , session . Version , "0.1.6" } ,
2015-11-14 21:21:31 +03:00
{ "github.com/go-macaron/toolbox" , toolbox . Version , "0.1.0" } ,
2015-12-12 02:52:28 +03:00
{ "gopkg.in/ini.v1" , ini . Version , "1.8.4" } ,
2016-08-06 04:34:13 +03:00
{ "gopkg.in/macaron.v1" , macaron . Version , "1.1.7" } ,
2016-08-30 09:57:22 +03:00
{ "github.com/gogits/git-module" , git . Version , "0.4.1" } ,
2016-08-16 20:19:09 +03:00
{ "github.com/gogits/go-gogs-client" , gogs . Version , "0.12.1" } ,
2014-10-10 14:15:27 +04:00
}
2014-12-15 09:49:59 +03:00
for _ , c := range checkers {
2015-08-06 18:03:04 +03:00
if ! version . Compare ( c . Version ( ) , c . Expected , ">=" ) {
2016-07-11 02:57:15 +03:00
log . Fatal ( 4 , ` Dependency outdated !
2016-11-03 11:37:24 +03:00
Package ' % s ' current version ( % s ) is below requirement ( % s ) ,
2016-07-11 02:57:15 +03:00
please use following command to update this package and recompile Gogs :
go get - u % [ 1 ] s ` , c . ImportPath , c . Version ( ) , c . Expected )
2014-12-15 09:49:59 +03:00
}
2014-09-28 09:38:25 +04:00
}
2014-05-26 04:57:01 +04:00
}
2014-07-26 08:24:27 +04:00
// newMacaron initializes Macaron instance.
func newMacaron ( ) * macaron . Macaron {
m := macaron . New ( )
2015-07-31 10:03:30 +03:00
if ! setting . DisableRouterLog {
m . Use ( macaron . Logger ( ) )
}
2014-07-26 08:24:27 +04:00
m . Use ( macaron . Recovery ( ) )
2014-10-14 02:04:07 +04:00
if setting . EnableGzip {
2015-10-16 04:28:12 +03:00
m . Use ( gzip . Gziper ( ) )
2014-10-14 02:04:07 +04:00
}
2014-11-16 08:06:09 +03:00
if setting . Protocol == setting . FCGI {
m . SetURLPrefix ( setting . AppSubUrl )
}
2014-09-16 01:23:58 +04:00
m . Use ( macaron . Static (
path . Join ( setting . StaticRootPath , "public" ) ,
2014-07-26 08:24:27 +04:00
macaron . StaticOptions {
2015-07-31 10:03:30 +03:00
SkipLogging : setting . DisableRouterLog ,
2014-07-26 08:24:27 +04:00
} ,
) )
2014-11-21 18:58:08 +03:00
m . Use ( macaron . Static (
setting . AvatarUploadPath ,
macaron . StaticOptions {
Prefix : "avatars" ,
2015-07-31 10:03:30 +03:00
SkipLogging : setting . DisableRouterLog ,
2014-11-21 18:58:08 +03:00
} ,
) )
2016-07-15 19:36:39 +03:00
funcMap := template . NewFuncMap ( )
2014-07-26 08:24:27 +04:00
m . Use ( macaron . Renderer ( macaron . RenderOptions {
2016-03-12 21:48:34 +03:00
Directory : path . Join ( setting . StaticRootPath , "templates" ) ,
AppendDirectories : [ ] string { path . Join ( setting . CustomPath , "templates" ) } ,
2016-07-15 19:36:39 +03:00
Funcs : funcMap ,
2016-03-12 21:48:34 +03:00
IndentJSON : macaron . Env != macaron . PROD ,
2014-07-26 08:24:27 +04:00
} ) )
2016-07-15 19:36:39 +03:00
models . InitMailRender ( path . Join ( setting . StaticRootPath , "templates/mail" ) ,
path . Join ( setting . CustomPath , "templates/mail" ) , funcMap )
2015-03-18 13:37:44 +03:00
localeNames , err := bindata . AssetDir ( "conf/locale" )
if err != nil {
log . Fatal ( 4 , "Fail to list locale files: %v" , err )
}
localFiles := make ( map [ string ] [ ] byte )
for _ , name := range localeNames {
localFiles [ name ] = bindata . MustAsset ( "conf/locale/" + name )
}
2014-08-07 01:21:24 +04:00
m . Use ( i18n . I18n ( i18n . Options {
2014-10-10 02:35:09 +04:00
SubURL : setting . AppSubUrl ,
2015-03-18 13:37:44 +03:00
Files : localFiles ,
2014-10-10 02:35:09 +04:00
CustomDirectory : path . Join ( setting . CustomPath , "conf/locale" ) ,
Langs : setting . Langs ,
Names : setting . Names ,
2015-12-05 09:20:11 +03:00
DefaultLang : "en-US" ,
2014-10-10 02:35:09 +04:00
Redirect : true ,
2014-07-26 08:24:27 +04:00
} ) )
2014-08-01 01:25:34 +04:00
m . Use ( cache . Cacher ( cache . Options {
2014-12-31 12:08:57 +03:00
Adapter : setting . CacheAdapter ,
AdapterConfig : setting . CacheConn ,
2016-08-11 15:59:11 +03:00
Interval : setting . CacheInterval ,
2014-08-01 01:25:34 +04:00
} ) )
2014-09-20 04:11:34 +04:00
m . Use ( captcha . Captchaer ( captcha . Options {
SubURL : setting . AppSubUrl ,
} ) )
2014-12-28 15:40:35 +03:00
m . Use ( session . Sessioner ( setting . SessionConfig ) )
2014-12-21 06:51:16 +03:00
m . Use ( csrf . Csrfer ( csrf . Options {
2014-09-21 20:22:50 +04:00
Secret : setting . SecretKey ,
2016-03-13 04:56:03 +03:00
Cookie : setting . CSRFCookieName ,
2014-09-21 20:22:50 +04:00
SetCookie : true ,
Header : "X-Csrf-Token" ,
CookiePath : setting . AppSubUrl ,
2014-08-01 01:25:34 +04:00
} ) )
2014-08-07 01:21:24 +04:00
m . Use ( toolbox . Toolboxer ( m , toolbox . Options {
HealthCheckFuncs : [ ] * toolbox . HealthCheckFuncDesc {
& toolbox . HealthCheckFuncDesc {
Desc : "Database connection" ,
Func : models . Ping ,
} ,
} ,
} ) )
2016-03-11 19:56:52 +03:00
m . Use ( context . Contexter ( ) )
2014-07-26 08:24:27 +04:00
return m
2014-03-19 13:31:38 +04:00
}
2016-05-12 21:32:28 +03:00
func runWeb ( ctx * cli . Context ) error {
2015-02-05 13:12:37 +03:00
if ctx . IsSet ( "config" ) {
setting . CustomConf = ctx . String ( "config" )
}
2014-03-30 01:50:51 +04:00
routers . GlobalInit ( )
2015-02-15 01:01:33 +03:00
checkVersion ( )
2014-02-19 13:50:53 +04:00
2014-07-26 08:24:27 +04:00
m := newMacaron ( )
2014-03-15 15:01:50 +04:00
2016-03-11 19:56:52 +03:00
reqSignIn := context . Toggle ( & context . ToggleOptions { SignInRequired : true } )
ignSignIn := context . Toggle ( & context . ToggleOptions { SignInRequired : setting . Service . RequireSignInView } )
ignSignInAndCsrf := context . Toggle ( & context . ToggleOptions { DisableCSRF : true } )
reqSignOut := context . Toggle ( & context . ToggleOptions { SignOutRequired : true } )
2014-03-22 21:44:02 +04:00
2014-05-05 10:42:52 +04:00
bindIgnErr := binding . BindIgnErr
2014-04-10 22:37:43 +04:00
2016-03-06 02:08:42 +03:00
// FIXME: not all routes need go through same middlewares.
// Especially some AJAX requests, we can reduce middleware number to improve performance.
2014-02-19 13:50:53 +04:00
// Routers.
2014-03-24 19:58:46 +04:00
m . Get ( "/" , ignSignIn , routers . Home )
2016-03-11 23:33:12 +03:00
m . Group ( "/explore" , func ( ) {
2016-03-12 01:12:37 +03:00
m . Get ( "" , func ( ctx * context . Context ) {
ctx . Redirect ( setting . AppSubUrl + "/explore/repos" )
} )
2016-03-11 23:33:12 +03:00
m . Get ( "/repos" , routers . ExploreRepos )
m . Get ( "/users" , routers . ExploreUsers )
2016-09-01 16:08:05 +03:00
m . Get ( "/organizations" , routers . ExploreOrganizations )
2016-03-11 23:33:12 +03:00
} , ignSignIn )
2015-09-02 23:18:09 +03:00
m . Combo ( "/install" , routers . InstallInit ) . Get ( routers . Install ) .
2015-02-01 20:41:03 +03:00
Post ( bindIgnErr ( auth . InstallForm { } ) , routers . InstallPost )
2015-09-24 21:20:07 +03:00
m . Get ( "/^:type(issues|pulls)$" , reqSignIn , user . Issues )
2014-03-29 18:01:52 +04:00
2015-08-18 21:49:44 +03:00
// ***** START: User *****
2014-10-25 02:43:17 +04:00
m . Group ( "/user" , func ( ) {
m . Get ( "/login" , user . SignIn )
m . Post ( "/login" , bindIgnErr ( auth . SignInForm { } ) , user . SignInPost )
m . Get ( "/sign_up" , user . SignUp )
m . Post ( "/sign_up" , bindIgnErr ( auth . RegisterForm { } ) , user . SignUpPost )
m . Get ( "/reset_password" , user . ResetPasswd )
m . Post ( "/reset_password" , user . ResetPasswdPost )
2014-03-23 00:00:46 +04:00
} , reqSignOut )
2015-08-18 21:49:44 +03:00
2014-10-25 02:43:17 +04:00
m . Group ( "/user/settings" , func ( ) {
m . Get ( "" , user . Settings )
m . Post ( "" , bindIgnErr ( auth . UpdateProfileForm { } ) , user . SettingsPost )
2016-08-07 20:27:38 +03:00
m . Combo ( "/avatar" ) . Get ( user . SettingsAvatar ) .
Post ( binding . MultipartForm ( auth . AvatarForm { } ) , user . SettingsAvatarPost )
2016-03-05 08:51:51 +03:00
m . Post ( "/avatar/delete" , user . SettingsDeleteAvatar )
2015-09-10 18:40:34 +03:00
m . Combo ( "/email" ) . Get ( user . SettingsEmails ) .
Post ( bindIgnErr ( auth . AddEmailForm { } ) , user . SettingsEmailPost )
m . Post ( "/email/delete" , user . DeleteEmail )
2014-10-25 02:43:17 +04:00
m . Get ( "/password" , user . SettingsPassword )
m . Post ( "/password" , bindIgnErr ( auth . ChangePasswordForm { } ) , user . SettingsPasswordPost )
2015-08-20 12:11:29 +03:00
m . Combo ( "/ssh" ) . Get ( user . SettingsSSHKeys ) .
Post ( bindIgnErr ( auth . AddSSHKeyForm { } ) , user . SettingsSSHKeysPost )
m . Post ( "/ssh/delete" , user . DeleteSSHKey )
2015-08-18 22:36:16 +03:00
m . Combo ( "/applications" ) . Get ( user . SettingsApplications ) .
Post ( bindIgnErr ( auth . NewAccessTokenForm { } ) , user . SettingsApplicationsPost )
m . Post ( "/applications/delete" , user . SettingsDeleteApplication )
2014-10-25 02:43:17 +04:00
m . Route ( "/delete" , "GET,POST" , user . SettingsDelete )
2016-03-11 19:56:52 +03:00
} , reqSignIn , func ( ctx * context . Context ) {
2015-08-18 21:49:44 +03:00
ctx . Data [ "PageIsUserSettings" ] = true
} )
2014-10-25 02:43:17 +04:00
m . Group ( "/user" , func ( ) {
2014-07-26 08:24:27 +04:00
// r.Get("/feeds", binding.Bind(auth.FeedsForm{}), user.Feeds)
2014-10-25 02:43:17 +04:00
m . Any ( "/activate" , user . Activate )
2014-12-17 18:42:54 +03:00
m . Any ( "/activate_email" , user . ActivateEmail )
2014-10-25 02:43:17 +04:00
m . Get ( "/email2user" , user . Email2User )
m . Get ( "/forget_password" , user . ForgotPasswd )
m . Post ( "/forget_password" , user . ForgotPasswdPost )
m . Get ( "/logout" , user . SignOut )
2014-03-23 00:00:46 +04:00
} )
2015-08-18 21:49:44 +03:00
// ***** END: User *****
2014-03-10 12:54:52 +04:00
2016-03-11 19:56:52 +03:00
adminReq := context . Toggle ( & context . ToggleOptions { SignInRequired : true , AdminRequired : true } )
2014-03-22 21:44:02 +04:00
2015-08-26 16:45:51 +03:00
// ***** START: Admin *****
2014-10-25 02:43:17 +04:00
m . Group ( "/admin" , func ( ) {
2014-08-29 16:50:43 +04:00
m . Get ( "" , adminReq , admin . Dashboard )
2014-10-25 02:43:17 +04:00
m . Get ( "/config" , admin . Config )
2016-02-25 07:59:17 +03:00
m . Post ( "/config/test_mail" , admin . SendTestMail )
2014-10-25 02:43:17 +04:00
m . Get ( "/monitor" , admin . Monitor )
m . Group ( "/users" , func ( ) {
m . Get ( "" , admin . Users )
2015-12-12 03:24:57 +03:00
m . Combo ( "/new" ) . Get ( admin . NewUser ) . Post ( bindIgnErr ( auth . AdminCrateUserForm { } ) , admin . NewUserPost )
m . Combo ( "/:userid" ) . Get ( admin . EditUser ) . Post ( bindIgnErr ( auth . AdminEditUserForm { } ) , admin . EditUserPost )
2014-10-25 02:43:17 +04:00
m . Post ( "/:userid/delete" , admin . DeleteUser )
2014-08-29 16:50:43 +04:00
} )
2014-10-25 02:43:17 +04:00
m . Group ( "/orgs" , func ( ) {
m . Get ( "" , admin . Organizations )
2014-08-29 16:50:43 +04:00
} )
2014-10-25 02:43:17 +04:00
m . Group ( "/repos" , func ( ) {
2015-12-06 01:39:29 +03:00
m . Get ( "" , admin . Repos )
m . Post ( "/delete" , admin . DeleteRepo )
2014-08-29 16:50:43 +04:00
} )
2014-10-25 02:43:17 +04:00
m . Group ( "/auths" , func ( ) {
m . Get ( "" , admin . Authentications )
2015-12-12 03:24:57 +03:00
m . Combo ( "/new" ) . Get ( admin . NewAuthSource ) . Post ( bindIgnErr ( auth . AuthenticationForm { } ) , admin . NewAuthSourcePost )
2015-09-11 19:03:08 +03:00
m . Combo ( "/:authid" ) . Get ( admin . EditAuthSource ) .
Post ( bindIgnErr ( auth . AuthenticationForm { } ) , admin . EditAuthSourcePost )
2014-10-25 02:43:17 +04:00
m . Post ( "/:authid/delete" , admin . DeleteAuthSource )
2014-08-29 16:50:43 +04:00
} )
2014-10-09 02:29:18 +04:00
2014-10-25 02:43:17 +04:00
m . Group ( "/notices" , func ( ) {
m . Get ( "" , admin . Notices )
2015-12-05 09:09:14 +03:00
m . Post ( "/delete" , admin . DeleteNotices )
2015-12-02 07:33:08 +03:00
m . Get ( "/empty" , admin . EmptyNotices )
2014-10-09 02:29:18 +04:00
} )
2014-05-03 06:48:14 +04:00
} , adminReq )
2015-08-26 16:45:51 +03:00
// ***** END: Admin *****
2014-05-03 06:48:14 +04:00
2015-08-11 12:54:00 +03:00
m . Group ( "" , func ( ) {
2015-12-21 15:24:11 +03:00
m . Group ( "/:username" , func ( ) {
m . Get ( "" , user . Profile )
m . Get ( "/followers" , user . Followers )
m . Get ( "/following" , user . Following )
m . Get ( "/stars" , user . Stars )
} )
2016-03-11 19:56:52 +03:00
m . Get ( "/attachments/:uuid" , func ( ctx * context . Context ) {
2015-08-11 18:24:40 +03:00
attach , err := models . GetAttachmentByUUID ( ctx . Params ( ":uuid" ) )
if err != nil {
if models . IsErrAttachmentNotExist ( err ) {
ctx . Error ( 404 )
} else {
ctx . Handle ( 500 , "GetAttachmentByUUID" , err )
}
return
}
2015-08-11 23:49:51 +03:00
fr , err := os . Open ( attach . LocalPath ( ) )
if err != nil {
ctx . Handle ( 500 , "Open" , err )
return
}
defer fr . Close ( )
2015-08-11 23:58:39 +03:00
ctx . Header ( ) . Set ( "Cache-Control" , "public,max-age=86400" )
2016-08-12 02:53:40 +03:00
ctx . Header ( ) . Set ( "Content-Disposition" , fmt . Sprintf ( ` inline; filename="%s" ` , attach . Name ) )
2015-08-11 18:24:40 +03:00
// Fix #312. Attachments with , in their name are not handled correctly by Google Chrome.
// We must put the name in " manually.
2015-08-11 23:49:51 +03:00
if err = repo . ServeData ( ctx , "\"" + attach . Name + "\"" , fr ) ; err != nil {
ctx . Handle ( 500 , "ServeData" , err )
return
}
2015-08-11 18:24:40 +03:00
} )
m . Post ( "/issues/attachments" , repo . UploadIssueAttachment )
2015-08-11 12:54:00 +03:00
} , ignSignIn )
2014-07-12 08:55:19 +04:00
2015-12-21 15:24:11 +03:00
m . Group ( "/:username" , func ( ) {
m . Get ( "/action/:action" , user . Action )
} , reqSignIn )
2014-07-26 08:24:27 +04:00
if macaron . Env == macaron . DEV {
2014-08-01 01:25:34 +04:00
m . Get ( "/template/*" , dev . TemplatePreview )
2014-03-27 19:37:33 +04:00
}
2016-03-11 19:56:52 +03:00
reqRepoAdmin := context . RequireRepoAdmin ( )
reqRepoWriter := context . RequireRepoWriter ( )
2014-07-26 10:28:04 +04:00
2015-08-26 16:45:51 +03:00
// ***** START: Organization *****
2014-10-25 02:43:17 +04:00
m . Group ( "/org" , func ( ) {
m . Get ( "/create" , org . Create )
m . Post ( "/create" , bindIgnErr ( auth . CreateOrgForm { } ) , org . CreatePost )
m . Group ( "/:org" , func ( ) {
m . Get ( "/dashboard" , user . Dashboard )
2015-09-24 21:20:07 +03:00
m . Get ( "/^:type(issues|pulls)$" , user . Issues )
2014-10-25 02:43:17 +04:00
m . Get ( "/members" , org . Members )
m . Get ( "/members/action/:action" , org . MembersAction )
m . Get ( "/teams" , org . Teams )
2016-03-11 19:56:52 +03:00
} , context . OrgAssignment ( true ) )
2016-01-31 16:28:42 +03:00
m . Group ( "/:org" , func ( ) {
2014-10-25 02:43:17 +04:00
m . Get ( "/teams/:team" , org . TeamMembers )
m . Get ( "/teams/:team/repositories" , org . TeamRepositories )
2015-11-22 09:32:09 +03:00
m . Route ( "/teams/:team/action/:action" , "GET,POST" , org . TeamsAction )
m . Route ( "/teams/:team/action/repo/:action" , "GET,POST" , org . TeamsRepoAction )
2016-03-11 19:56:52 +03:00
} , context . OrgAssignment ( true , false , true ) )
2014-07-26 10:28:04 +04:00
2014-10-25 02:43:17 +04:00
m . Group ( "/:org" , func ( ) {
m . Get ( "/teams/new" , org . NewTeam )
m . Post ( "/teams/new" , bindIgnErr ( auth . CreateTeamForm { } ) , org . NewTeamPost )
m . Get ( "/teams/:team/edit" , org . EditTeam )
m . Post ( "/teams/:team/edit" , bindIgnErr ( auth . CreateTeamForm { } ) , org . EditTeamPost )
m . Post ( "/teams/:team/delete" , org . DeleteTeam )
m . Group ( "/settings" , func ( ) {
2015-08-26 16:45:51 +03:00
m . Combo ( "" ) . Get ( org . Settings ) .
Post ( bindIgnErr ( auth . UpdateOrgSettingForm { } ) , org . SettingsPost )
2016-08-07 20:27:38 +03:00
m . Post ( "/avatar" , binding . MultipartForm ( auth . AvatarForm { } ) , org . SettingsAvatar )
2016-03-06 19:36:30 +03:00
m . Post ( "/avatar/delete" , org . SettingsDeleteAvatar )
2015-08-26 16:45:51 +03:00
m . Group ( "/hooks" , func ( ) {
m . Get ( "" , org . Webhooks )
m . Post ( "/delete" , org . DeleteWebhook )
2015-08-26 19:30:06 +03:00
m . Get ( "/:type/new" , repo . WebhooksNew )
m . Post ( "/gogs/new" , bindIgnErr ( auth . NewWebhookForm { } ) , repo . WebHooksNewPost )
m . Post ( "/slack/new" , bindIgnErr ( auth . NewSlackHookForm { } ) , repo . SlackHooksNewPost )
2015-08-26 20:04:23 +03:00
m . Get ( "/:id" , repo . WebHooksEdit )
m . Post ( "/gogs/:id" , bindIgnErr ( auth . NewWebhookForm { } ) , repo . WebHooksEditPost )
m . Post ( "/slack/:id" , bindIgnErr ( auth . NewSlackHookForm { } ) , repo . SlackHooksEditPost )
2015-08-26 16:45:51 +03:00
} )
2014-10-25 02:43:17 +04:00
m . Route ( "/delete" , "GET,POST" , org . SettingsDelete )
2014-08-14 10:12:21 +04:00
} )
2014-08-15 14:29:41 +04:00
2014-10-25 02:43:17 +04:00
m . Route ( "/invitations/new" , "GET,POST" , org . Invitation )
2016-03-11 19:56:52 +03:00
} , context . OrgAssignment ( true , true ) )
2014-07-26 10:28:04 +04:00
} , reqSignIn )
2015-08-26 16:45:51 +03:00
// ***** END: Organization *****
2014-07-26 10:28:04 +04:00
2015-08-26 16:45:51 +03:00
// ***** START: Repository *****
2014-10-25 02:43:17 +04:00
m . Group ( "/repo" , func ( ) {
m . Get ( "/create" , repo . Create )
m . Post ( "/create" , bindIgnErr ( auth . CreateRepoForm { } ) , repo . CreatePost )
m . Get ( "/migrate" , repo . Migrate )
m . Post ( "/migrate" , bindIgnErr ( auth . MigrateRepoForm { } ) , repo . MigratePost )
2015-08-08 12:10:34 +03:00
m . Combo ( "/fork/:repoid" ) . Get ( repo . Fork ) .
Post ( bindIgnErr ( auth . CreateRepoForm { } ) , repo . ForkPost )
2014-07-27 07:53:16 +04:00
} , reqSignIn )
2014-10-25 02:43:17 +04:00
m . Group ( "/:username/:reponame" , func ( ) {
m . Group ( "/settings" , func ( ) {
2015-08-26 16:45:51 +03:00
m . Combo ( "" ) . Get ( repo . Settings ) .
Post ( bindIgnErr ( auth . RepoSettingForm { } ) , repo . SettingsPost )
2016-03-06 02:08:42 +03:00
m . Group ( "/collaboration" , func ( ) {
m . Combo ( "" ) . Get ( repo . Collaboration ) . Post ( repo . CollaborationPost )
m . Post ( "/access_mode" , repo . ChangeCollaborationAccessMode )
m . Post ( "/delete" , repo . DeleteCollaboration )
} )
2015-08-26 16:45:51 +03:00
m . Group ( "/hooks" , func ( ) {
m . Get ( "" , repo . Webhooks )
m . Post ( "/delete" , repo . DeleteWebhook )
2015-08-26 19:30:06 +03:00
m . Get ( "/:type/new" , repo . WebhooksNew )
m . Post ( "/gogs/new" , bindIgnErr ( auth . NewWebhookForm { } ) , repo . WebHooksNewPost )
m . Post ( "/slack/new" , bindIgnErr ( auth . NewSlackHookForm { } ) , repo . SlackHooksNewPost )
2015-08-26 20:04:23 +03:00
m . Get ( "/:id" , repo . WebHooksEdit )
2015-12-05 21:24:13 +03:00
m . Post ( "/:id/test" , repo . TestWebhook )
2015-08-26 20:04:23 +03:00
m . Post ( "/gogs/:id" , bindIgnErr ( auth . NewWebhookForm { } ) , repo . WebHooksEditPost )
m . Post ( "/slack/:id" , bindIgnErr ( auth . NewSlackHookForm { } ) , repo . SlackHooksEditPost )
m . Group ( "/git" , func ( ) {
m . Get ( "" , repo . GitHooks )
m . Combo ( "/:name" ) . Get ( repo . GitHooksEdit ) .
Post ( repo . GitHooksEditPost )
2016-03-11 19:56:52 +03:00
} , context . GitHookService ( ) )
2015-08-26 16:45:51 +03:00
} )
2015-08-26 19:30:06 +03:00
2015-08-06 17:48:11 +03:00
m . Group ( "/keys" , func ( ) {
2015-08-26 16:45:51 +03:00
m . Combo ( "" ) . Get ( repo . DeployKeys ) .
Post ( bindIgnErr ( auth . AddSSHKeyForm { } ) , repo . DeployKeysPost )
2015-08-06 17:48:11 +03:00
m . Post ( "/delete" , repo . DeleteDeployKey )
} )
2016-03-11 19:56:52 +03:00
} , func ( ctx * context . Context ) {
2015-11-16 19:16:52 +03:00
ctx . Data [ "PageIsSettings" ] = true
2014-07-26 10:28:04 +04:00
} )
2016-03-11 19:56:52 +03:00
} , reqSignIn , context . RepoAssignment ( ) , reqRepoAdmin , context . RepoRef ( ) )
2014-07-26 10:28:04 +04:00
2016-03-11 19:56:52 +03:00
m . Get ( "/:username/:reponame/action/:action" , reqSignIn , context . RepoAssignment ( ) , repo . Action )
2014-10-25 02:43:17 +04:00
m . Group ( "/:username/:reponame" , func ( ) {
2016-08-09 19:39:55 +03:00
// FIXME: should use different URLs but mostly same logic for comments of issue and pull reuqest.
// So they can apply their own enable/disable logic on routers.
2014-10-25 02:43:17 +04:00
m . Group ( "/issues" , func ( ) {
2016-08-09 19:39:55 +03:00
m . Combo ( "/new" , repo . MustEnableIssues ) . Get ( context . RepoRef ( ) , repo . NewIssue ) .
2015-08-09 10:23:02 +03:00
Post ( bindIgnErr ( auth . CreateIssueForm { } ) , repo . NewIssuePost )
2015-08-13 11:07:11 +03:00
m . Group ( "/:index" , func ( ) {
m . Post ( "/label" , repo . UpdateIssueLabel )
m . Post ( "/milestone" , repo . UpdateIssueMilestone )
2015-08-14 19:42:43 +03:00
m . Post ( "/assignee" , repo . UpdateIssueAssignee )
2016-03-06 04:45:23 +03:00
} , reqRepoWriter )
2015-08-19 23:31:28 +03:00
m . Group ( "/:index" , func ( ) {
m . Post ( "/title" , repo . UpdateIssueTitle )
m . Post ( "/content" , repo . UpdateIssueContent )
2016-08-11 16:22:56 +03:00
m . Combo ( "/comments" ) . Post ( bindIgnErr ( auth . CreateCommentForm { } ) , repo . NewComment )
2015-08-19 23:31:28 +03:00
} )
2016-08-09 19:39:55 +03:00
} )
2016-07-25 21:48:17 +03:00
m . Group ( "/comments/:id" , func ( ) {
m . Post ( "" , repo . UpdateCommentContent )
m . Post ( "/delete" , repo . DeleteComment )
} )
2015-07-24 11:42:47 +03:00
m . Group ( "/labels" , func ( ) {
m . Post ( "/new" , bindIgnErr ( auth . CreateLabelForm { } ) , repo . NewLabel )
m . Post ( "/edit" , bindIgnErr ( auth . CreateLabelForm { } ) , repo . UpdateLabel )
m . Post ( "/delete" , repo . DeleteLabel )
2016-08-30 05:02:49 +03:00
m . Post ( "/initialize" , bindIgnErr ( auth . InitializeLabelsForm { } ) , repo . InitializeLabels )
2016-08-30 15:30:47 +03:00
} , reqRepoWriter , context . RepoRef ( ) )
2015-07-24 11:42:47 +03:00
m . Group ( "/milestones" , func ( ) {
2015-12-03 04:56:26 +03:00
m . Combo ( "/new" ) . Get ( repo . NewMilestone ) .
Post ( bindIgnErr ( auth . CreateMilestoneForm { } ) , repo . NewMilestonePost )
2015-08-06 18:25:35 +03:00
m . Get ( "/:id/edit" , repo . EditMilestone )
m . Post ( "/:id/edit" , bindIgnErr ( auth . CreateMilestoneForm { } ) , repo . EditMilestonePost )
m . Get ( "/:id/:action" , repo . ChangeMilestonStatus )
2015-08-05 15:23:08 +03:00
m . Post ( "/delete" , repo . DeleteMilestone )
2016-08-30 15:30:47 +03:00
} , reqRepoWriter , context . RepoRef ( ) )
2014-07-26 10:28:04 +04:00
2014-12-11 00:37:54 +03:00
m . Group ( "/releases" , func ( ) {
m . Get ( "/new" , repo . NewRelease )
m . Post ( "/new" , bindIgnErr ( auth . NewReleaseForm { } ) , repo . NewReleasePost )
2015-11-20 10:38:41 +03:00
m . Post ( "/delete" , repo . DeleteRelease )
2016-03-11 19:56:52 +03:00
} , reqRepoWriter , context . RepoRef ( ) )
2015-08-08 17:43:14 +03:00
2016-09-03 13:00:59 +03:00
m . Group ( "/releases" , func ( ) {
m . Get ( "/edit/*" , repo . EditRelease )
m . Post ( "/edit/*" , bindIgnErr ( auth . EditReleaseForm { } ) , repo . EditReleasePost )
} , reqRepoWriter , func ( ctx * context . Context ) {
var err error
ctx . Repo . Commit , err = ctx . Repo . GitRepo . GetBranchCommit ( ctx . Repo . Repository . DefaultBranch )
if err != nil {
ctx . Handle ( 500 , "GetBranchCommit" , err )
return
}
ctx . Repo . CommitsCount , err = ctx . Repo . Commit . CommitsCount ( )
if err != nil {
ctx . Handle ( 500 , "CommitsCount" , err )
return
}
ctx . Data [ "CommitsCount" ] = ctx . Repo . CommitsCount
} )
2016-02-19 22:48:32 +03:00
m . Combo ( "/compare/*" , repo . MustAllowPulls ) . Get ( repo . CompareAndPullRequest ) .
2015-09-02 02:07:02 +03:00
Post ( bindIgnErr ( auth . CreateIssueForm { } ) , repo . CompareAndPullRequestPost )
2016-08-11 15:48:08 +03:00
m . Group ( "" , func ( ) {
m . Combo ( "/_edit/*" ) . Get ( repo . EditFile ) .
Post ( bindIgnErr ( auth . EditRepoFileForm { } ) , repo . EditFilePost )
m . Combo ( "/_new/*" ) . Get ( repo . NewFile ) .
Post ( bindIgnErr ( auth . EditRepoFileForm { } ) , repo . NewFilePost )
2016-08-15 09:38:35 +03:00
m . Post ( "/_preview/*" , bindIgnErr ( auth . EditPreviewDiffForm { } ) , repo . DiffPreviewPost )
2016-08-28 11:41:44 +03:00
m . Combo ( "/_delete/*" ) . Get ( repo . DeleteFile ) .
Post ( bindIgnErr ( auth . DeleteRepoFileForm { } ) , repo . DeleteFilePost )
2016-08-30 15:07:50 +03:00
m . Group ( "" , func ( ) {
m . Combo ( "/_upload/*" ) . Get ( repo . UploadFile ) .
Post ( bindIgnErr ( auth . UploadRepoFileForm { } ) , repo . UploadFilePost )
m . Post ( "/upload-file" , repo . UploadFileToServer )
m . Post ( "/upload-remove" , bindIgnErr ( auth . RemoveUploadFileForm { } ) , repo . RemoveUploadFileFromServer )
} , func ( ctx * context . Context ) {
if ! setting . Repository . Upload . Enabled {
ctx . Handle ( 404 , "" , nil )
return
}
} )
2016-08-15 09:38:35 +03:00
} , reqRepoWriter , context . RepoRef ( ) , func ( ctx * context . Context ) {
2016-08-28 14:56:41 +03:00
if ! ctx . Repo . Repository . CanEnableEditor ( ) || ctx . Repo . IsViewCommit {
2016-08-15 09:38:35 +03:00
ctx . Handle ( 404 , "" , nil )
return
}
} )
2016-03-11 19:56:52 +03:00
} , reqSignIn , context . RepoAssignment ( ) , repo . MustBeNotBare )
2014-07-26 10:28:04 +04:00
2014-10-25 02:43:17 +04:00
m . Group ( "/:username/:reponame" , func ( ) {
2015-11-16 18:14:12 +03:00
m . Group ( "" , func ( ) {
m . Get ( "/releases" , repo . Releases )
m . Get ( "/^:type(issues|pulls)$" , repo . RetrieveLabels , repo . Issues )
2015-12-03 22:31:31 +03:00
m . Get ( "/^:type(issues|pulls)$/:index" , repo . ViewIssue )
2015-11-16 18:14:12 +03:00
m . Get ( "/labels/" , repo . RetrieveLabels , repo . Labels )
m . Get ( "/milestones" , repo . Milestones )
2016-03-11 19:56:52 +03:00
} , context . RepoRef ( ) )
2015-11-16 18:14:12 +03:00
2015-11-26 04:10:25 +03:00
// m.Get("/branches", repo.Branches)
m . Group ( "/wiki" , func ( ) {
m . Get ( "/?:page" , repo . Wiki )
2015-11-27 09:50:38 +03:00
m . Get ( "/_pages" , repo . WikiPages )
2015-11-26 04:10:25 +03:00
m . Group ( "" , func ( ) {
2015-11-27 01:33:45 +03:00
m . Combo ( "/_new" ) . Get ( repo . NewWiki ) .
Post ( bindIgnErr ( auth . NewWikiForm { } ) , repo . NewWikiPost )
2015-11-27 09:50:38 +03:00
m . Combo ( "/:page/_edit" ) . Get ( repo . EditWiki ) .
Post ( bindIgnErr ( auth . NewWikiForm { } ) , repo . EditWikiPost )
2016-03-04 21:32:17 +03:00
m . Post ( "/:page/delete" , repo . DeleteWikiPagePost )
2016-03-06 04:45:23 +03:00
} , reqSignIn , reqRepoWriter )
2016-03-11 19:56:52 +03:00
} , repo . MustEnableWiki , context . RepoRef ( ) )
2015-11-26 04:10:25 +03:00
2014-10-25 02:43:17 +04:00
m . Get ( "/archive/*" , repo . Download )
2014-07-26 08:24:27 +04:00
2015-09-02 11:08:05 +03:00
m . Group ( "/pulls/:index" , func ( ) {
2016-03-11 19:56:52 +03:00
m . Get ( "/commits" , context . RepoRef ( ) , repo . ViewPullCommits )
m . Get ( "/files" , context . RepoRef ( ) , repo . ViewPullFiles )
2016-03-06 04:45:23 +03:00
m . Post ( "/merge" , reqRepoWriter , repo . MergePullRequest )
2016-02-19 22:48:32 +03:00
} , repo . MustAllowPulls )
2015-09-02 11:08:05 +03:00
2014-11-07 06:06:41 +03:00
m . Group ( "" , func ( ) {
m . Get ( "/src/*" , repo . Home )
m . Get ( "/raw/*" , repo . SingleDownload )
m . Get ( "/commits/*" , repo . RefCommits )
2016-11-03 11:32:55 +03:00
m . Get ( "/commit/:sha([a-f0-9]{7,40})$" , repo . Diff )
2015-10-02 11:48:31 +03:00
m . Get ( "/forks" , repo . Forks )
2016-03-11 19:56:52 +03:00
} , context . RepoRef ( ) )
2016-11-03 11:32:55 +03:00
m . Get ( "/commit/:sha([a-f0-9]{7,40})\\.:ext(patch|diff)" , repo . RawDiff )
2014-11-07 22:46:13 +03:00
2016-11-03 11:32:55 +03:00
m . Get ( "/compare/:before([a-z0-9]{40})\\.\\.\\.:after([a-z0-9]{40})" , repo . CompareDiff )
2016-03-11 19:56:52 +03:00
} , ignSignIn , context . RepoAssignment ( ) , repo . MustBeNotBare )
2016-01-07 06:07:17 +03:00
m . Group ( "/:username/:reponame" , func ( ) {
m . Get ( "/stars" , repo . Stars )
m . Get ( "/watchers" , repo . Watchers )
2016-03-11 19:56:52 +03:00
} , ignSignIn , context . RepoAssignment ( ) , context . RepoRef ( ) )
2014-03-23 00:00:46 +04:00
2014-10-25 02:43:17 +04:00
m . Group ( "/:username" , func ( ) {
2015-07-31 09:25:12 +03:00
m . Group ( "/:reponame" , func ( ) {
m . Get ( "" , repo . Home )
2015-09-07 18:56:06 +03:00
m . Get ( "\\.git$" , repo . Home )
2016-03-11 19:56:52 +03:00
} , ignSignIn , context . RepoAssignment ( true ) , context . RepoRef ( ) )
2015-07-31 09:25:12 +03:00
2015-08-07 20:04:12 +03:00
m . Group ( "/:reponame" , func ( ) {
2015-10-24 10:36:47 +03:00
m . Any ( "/*" , ignSignInAndCsrf , repo . HTTP )
m . Head ( "/tasks/trigger" , repo . TriggerTask )
2015-08-07 20:04:12 +03:00
} )
2014-09-15 18:09:17 +04:00
} )
2015-08-26 16:45:51 +03:00
// ***** END: Repository *****
2014-03-21 20:48:26 +04:00
2016-03-14 01:49:16 +03:00
m . Group ( "/api" , func ( ) {
apiv1 . RegisterRoutes ( m )
} , ignSignIn )
2014-09-22 03:39:10 +04:00
// robots.txt
2016-03-11 19:56:52 +03:00
m . Get ( "/robots.txt" , func ( ctx * context . Context ) {
2014-09-22 03:39:10 +04:00
if setting . HasRobotsTxt {
2015-03-18 08:29:54 +03:00
ctx . ServeFileContent ( path . Join ( setting . CustomPath , "robots.txt" ) )
2014-09-22 03:39:10 +04:00
} else {
ctx . Error ( 404 )
}
} )
2014-03-23 14:27:01 +04:00
// Not found handler.
2014-03-23 09:48:01 +04:00
m . NotFound ( routers . NotFound )
2015-02-01 20:41:03 +03:00
// Flag for port number in case first time run conflict.
if ctx . IsSet ( "port" ) {
2016-08-12 00:55:10 +03:00
setting . AppUrl = strings . Replace ( setting . AppUrl , setting . HTTPPort , ctx . String ( "port" ) , 1 )
setting . HTTPPort = ctx . String ( "port" )
2015-02-01 20:41:03 +03:00
}
2016-08-12 00:55:10 +03:00
var listenAddr string
2016-08-12 00:46:33 +03:00
if setting . Protocol == setting . UNIX_SOCKET {
2016-08-12 00:55:10 +03:00
listenAddr = fmt . Sprintf ( "%s" , setting . HTTPAddr )
} else {
listenAddr = fmt . Sprintf ( "%s:%s" , setting . HTTPAddr , setting . HTTPPort )
2016-08-12 00:46:33 +03:00
}
2014-09-20 04:11:34 +04:00
log . Info ( "Listen: %v://%s%s" , setting . Protocol , listenAddr , setting . AppSubUrl )
2016-08-12 00:55:10 +03:00
var err error
2014-05-26 04:11:25 +04:00
switch setting . Protocol {
case setting . HTTP :
err = http . ListenAndServe ( listenAddr , m )
case setting . HTTPS :
2014-12-12 02:55:09 +03:00
server := & http . Server { Addr : listenAddr , TLSConfig : & tls . Config { MinVersion : tls . VersionTLS10 } , Handler : m }
2014-12-11 13:14:41 +03:00
err = server . ListenAndServeTLS ( setting . CertFile , setting . KeyFile )
2014-11-04 04:46:53 +03:00
case setting . FCGI :
err = fcgi . Serve ( nil , m )
2016-08-12 00:46:33 +03:00
case setting . UNIX_SOCKET :
os . Remove ( listenAddr )
2016-08-12 00:55:10 +03:00
var listener * net . UnixListener
listener , err = net . ListenUnix ( "unix" , & net . UnixAddr { listenAddr , "unix" } )
2016-08-12 00:46:33 +03:00
if err != nil {
2016-08-12 00:55:10 +03:00
break // Handle error after switch
2016-08-12 00:46:33 +03:00
}
2016-08-12 00:55:10 +03:00
// FIXME: add proper implementation of signal capture on all protocols
2016-08-12 00:46:33 +03:00
// execute this on SIGTERM or SIGINT: listener.Close()
2016-08-12 00:55:10 +03:00
if err = os . Chmod ( listenAddr , os . FileMode ( setting . UnixSocketPermission ) ) ; err != nil {
2016-08-12 00:46:33 +03:00
log . Fatal ( 4 , "Failed to set permission of unix socket: %v" , err )
}
err = http . Serve ( listener , m )
2014-05-26 04:11:25 +04:00
default :
2014-07-26 08:24:27 +04:00
log . Fatal ( 4 , "Invalid protocol: %s" , setting . Protocol )
2014-05-26 04:11:25 +04:00
}
if err != nil {
2014-07-26 08:24:27 +04:00
log . Fatal ( 4 , "Fail to start server: %v" , err )
2014-03-18 17:58:58 +04:00
}
2016-05-12 21:32:28 +03:00
return nil
2014-02-19 13:50:53 +04:00
}