2014-02-20 06:45:43 +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.
package repo
import (
2014-07-26 10:28:04 +04:00
"fmt"
2014-07-26 08:24:27 +04:00
"os"
2014-03-22 21:50:50 +04:00
"path"
2014-07-26 10:28:04 +04:00
"strings"
2014-03-22 21:50:50 +04:00
2014-07-26 08:24:27 +04:00
"github.com/Unknwon/com"
2014-05-06 03:58:13 +04:00
2015-12-16 01:25:45 +03:00
"github.com/gogits/git-module"
2015-12-10 04:46:05 +03:00
2014-02-20 06:45:43 +04:00
"github.com/gogits/gogs/models"
2014-03-08 01:05:18 +04:00
"github.com/gogits/gogs/modules/auth"
2014-03-22 21:50:50 +04:00
"github.com/gogits/gogs/modules/base"
2014-03-19 12:48:45 +04:00
"github.com/gogits/gogs/modules/log"
2014-03-15 17:17:16 +04:00
"github.com/gogits/gogs/modules/middleware"
2014-09-14 21:35:22 +04:00
"github.com/gogits/gogs/modules/setting"
2014-02-20 06:45:43 +04:00
)
2014-06-23 07:11:12 +04:00
const (
CREATE base . TplName = "repo/create"
MIGRATE base . TplName = "repo/migrate"
)
2016-01-07 06:07:17 +03:00
func MustBeNotBare ( ctx * middleware . Context ) {
if ctx . Repo . Repository . IsBare {
ctx . Handle ( 404 , "MustBeNotBare" , nil )
}
}
2015-07-19 12:11:16 +03:00
func checkContextUser ( ctx * middleware . Context , uid int64 ) * models . User {
2016-03-11 14:43:35 +03:00
orgs , err := models . GetOwnedOrgsByUserIDDesc ( ctx . User . Id , "updated_unix" )
2015-09-07 20:58:23 +03:00
if err != nil {
2015-09-07 21:02:09 +03:00
ctx . Handle ( 500 , "GetOwnedOrgsByUserIDDesc" , err )
2015-08-28 13:33:09 +03:00
return nil
}
2015-09-07 20:58:23 +03:00
ctx . Data [ "Orgs" ] = orgs
2015-08-28 13:33:09 +03:00
2015-07-19 12:11:16 +03:00
// Not equal means current user is an organization.
if uid == ctx . User . Id || uid == 0 {
return ctx . User
}
2015-08-08 17:43:14 +03:00
org , err := models . GetUserByID ( uid )
2015-08-05 06:14:17 +03:00
if models . IsErrUserNotExist ( err ) {
2015-07-19 12:11:16 +03:00
return ctx . User
}
if err != nil {
2015-10-25 11:26:26 +03:00
ctx . Handle ( 500 , "GetUserByID" , fmt . Errorf ( "[%d]: %v" , uid , err ) )
2015-07-19 12:11:16 +03:00
return nil
2015-08-28 13:33:09 +03:00
}
// Check ownership of organization.
2016-01-20 16:46:45 +03:00
if ! org . IsOrganization ( ) || ! ( ctx . User . IsAdmin || org . IsOwnedBy ( ctx . User . Id ) ) {
2015-07-19 12:11:16 +03:00
ctx . Error ( 403 )
return nil
2014-11-06 07:30:04 +03:00
}
2015-07-19 12:11:16 +03:00
return org
2014-11-06 07:30:04 +03:00
}
2014-04-11 02:09:57 +04:00
func Create ( ctx * middleware . Context ) {
2014-07-26 08:24:27 +04:00
ctx . Data [ "Title" ] = ctx . Tr ( "new_repo" )
// Give default value for template to render.
ctx . Data [ "Gitignores" ] = models . Gitignores
2014-03-19 21:14:56 +04:00
ctx . Data [ "Licenses" ] = models . Licenses
2015-08-28 11:44:04 +03:00
ctx . Data [ "Readmes" ] = models . Readmes
ctx . Data [ "readme" ] = "Default"
2015-08-28 13:45:25 +03:00
ctx . Data [ "private" ] = ctx . User . LastRepoVisibility
2015-10-25 11:26:26 +03:00
ctx . Data [ "IsForcedPrivate" ] = setting . Repository . ForcePrivate
2014-06-25 11:55:59 +04:00
2015-07-19 12:11:16 +03:00
ctxUser := checkContextUser ( ctx , ctx . QueryInt64 ( "org" ) )
if ctx . Written ( ) {
2014-11-06 07:30:04 +03:00
return
2014-07-26 10:28:04 +04:00
}
2014-06-28 23:43:25 +04:00
ctx . Data [ "ContextUser" ] = ctxUser
2014-06-23 07:11:12 +04:00
ctx . HTML ( 200 , CREATE )
2014-04-11 02:09:57 +04:00
}
2014-03-08 01:05:18 +04:00
2015-12-10 20:37:53 +03:00
func handleCreateError ( ctx * middleware . Context , owner * models . User , err error , name string , tpl base . TplName , form interface { } ) {
2015-08-28 13:33:09 +03:00
switch {
2015-12-10 20:37:53 +03:00
case models . IsErrReachLimitOfRepo ( err ) :
ctx . RenderWithErr ( ctx . Tr ( "repo.form.reach_limit_of_creation" , owner . RepoCreationNum ( ) ) , tpl , form )
2015-08-28 13:33:09 +03:00
case models . IsErrRepoAlreadyExist ( err ) :
ctx . Data [ "Err_RepoName" ] = true
ctx . RenderWithErr ( ctx . Tr ( "form.repo_name_been_taken" ) , tpl , form )
case models . IsErrNameReserved ( err ) :
ctx . Data [ "Err_RepoName" ] = true
ctx . RenderWithErr ( ctx . Tr ( "repo.form.name_reserved" , err . ( models . ErrNameReserved ) . Name ) , tpl , form )
case models . IsErrNamePatternNotAllowed ( err ) :
ctx . Data [ "Err_RepoName" ] = true
ctx . RenderWithErr ( ctx . Tr ( "repo.form.name_pattern_not_allowed" , err . ( models . ErrNamePatternNotAllowed ) . Pattern ) , tpl , form )
default :
ctx . Handle ( 500 , name , err )
}
}
2014-04-11 02:09:57 +04:00
func CreatePost ( ctx * middleware . Context , form auth . CreateRepoForm ) {
2014-07-26 08:24:27 +04:00
ctx . Data [ "Title" ] = ctx . Tr ( "new_repo" )
ctx . Data [ "Gitignores" ] = models . Gitignores
2014-04-11 02:09:57 +04:00
ctx . Data [ "Licenses" ] = models . Licenses
2015-08-28 11:44:04 +03:00
ctx . Data [ "Readmes" ] = models . Readmes
2014-02-20 06:45:43 +04:00
2015-07-19 12:11:16 +03:00
ctxUser := checkContextUser ( ctx , form . Uid )
if ctx . Written ( ) {
return
2014-07-26 10:28:04 +04:00
}
2014-07-26 08:24:27 +04:00
ctx . Data [ "ContextUser" ] = ctxUser
2014-03-23 00:00:46 +04:00
if ctx . HasError ( ) {
2014-06-23 07:11:12 +04:00
ctx . HTML ( 200 , CREATE )
2014-03-23 00:00:46 +04:00
return
}
2015-08-28 13:33:09 +03:00
repo , err := models . CreateRepository ( ctxUser , models . CreateRepoOptions {
Name : form . RepoName ,
Description : form . Description ,
Gitignores : form . Gitignores ,
License : form . License ,
Readme : form . Readme ,
2015-10-25 11:26:26 +03:00
IsPrivate : form . Private || setting . Repository . ForcePrivate ,
2015-08-28 13:33:09 +03:00
AutoInit : form . AutoInit ,
} )
2014-03-17 19:56:50 +04:00
if err == nil {
2015-10-25 11:26:26 +03:00
log . Trace ( "Repository created[%d]: %s/%s" , repo . ID , ctxUser . Name , repo . Name )
2014-11-06 07:30:04 +03:00
ctx . Redirect ( setting . AppSubUrl + "/" + ctxUser . Name + "/" + repo . Name )
2014-03-09 06:25:38 +04:00
return
2014-03-10 04:06:29 +04:00
}
2014-04-13 04:35:35 +04:00
if repo != nil {
2015-09-01 18:43:53 +03:00
if errDelete := models . DeleteRepository ( ctxUser . Id , repo . ID ) ; errDelete != nil {
2014-07-26 08:24:27 +04:00
log . Error ( 4 , "DeleteRepository: %v" , errDelete )
2014-04-13 04:35:35 +04:00
}
}
2015-03-27 00:11:47 +03:00
2015-12-10 20:37:53 +03:00
handleCreateError ( ctx , ctxUser , err , "CreatePost" , CREATE , & form )
2014-04-11 02:09:57 +04:00
}
2014-04-09 17:28:00 +04:00
2014-07-26 10:28:04 +04:00
func Migrate ( ctx * middleware . Context ) {
2014-08-01 08:06:19 +04:00
ctx . Data [ "Title" ] = ctx . Tr ( "new_migrate" )
2015-08-28 13:45:25 +03:00
ctx . Data [ "private" ] = ctx . User . LastRepoVisibility
2015-10-25 11:26:26 +03:00
ctx . Data [ "IsForcedPrivate" ] = setting . Repository . ForcePrivate
2015-12-09 19:24:56 +03:00
ctx . Data [ "mirror" ] = ctx . Query ( "mirror" ) == "1"
2014-08-01 08:06:19 +04:00
2015-07-19 12:11:16 +03:00
ctxUser := checkContextUser ( ctx , ctx . QueryInt64 ( "org" ) )
if ctx . Written ( ) {
2014-11-06 07:30:04 +03:00
return
2014-08-01 08:06:19 +04:00
}
ctx . Data [ "ContextUser" ] = ctxUser
2014-07-26 08:24:27 +04:00
2014-07-26 10:28:04 +04:00
ctx . HTML ( 200 , MIGRATE )
}
2014-07-26 08:24:27 +04:00
2014-07-26 10:28:04 +04:00
func MigratePost ( ctx * middleware . Context , form auth . MigrateRepoForm ) {
2014-08-01 08:06:19 +04:00
ctx . Data [ "Title" ] = ctx . Tr ( "new_migrate" )
2015-07-19 12:11:16 +03:00
ctxUser := checkContextUser ( ctx , form . Uid )
if ctx . Written ( ) {
return
2014-08-01 08:06:19 +04:00
}
ctx . Data [ "ContextUser" ] = ctxUser
2014-07-26 08:24:27 +04:00
2014-07-26 10:28:04 +04:00
if ctx . HasError ( ) {
ctx . HTML ( 200 , MIGRATE )
return
}
2014-07-26 08:24:27 +04:00
2015-11-04 02:40:52 +03:00
remoteAddr , err := form . ParseRemoteAddr ( ctx . User )
if err != nil {
if models . IsErrInvalidCloneAddr ( err ) {
2015-02-22 17:49:25 +03:00
ctx . Data [ "Err_CloneAddr" ] = true
2015-11-04 02:40:52 +03:00
addrErr := err . ( models . ErrInvalidCloneAddr )
switch {
case addrErr . IsURLError :
ctx . RenderWithErr ( ctx . Tr ( "form.url_error" ) , MIGRATE , & form )
case addrErr . IsPermissionDenied :
ctx . RenderWithErr ( ctx . Tr ( "repo.migrate.permission_denied" ) , MIGRATE , & form )
case addrErr . IsInvalidPath :
ctx . RenderWithErr ( ctx . Tr ( "repo.migrate.invalid_local_path" ) , MIGRATE , & form )
default :
ctx . Handle ( 500 , "Unknown error" , err )
}
} else {
ctx . Handle ( 500 , "ParseRemoteAddr" , err )
2015-02-22 17:49:25 +03:00
}
2015-01-02 13:14:11 +03:00
return
}
2015-10-25 11:26:26 +03:00
repo , err := models . MigrateRepository ( ctxUser , models . MigrateRepoOptions {
Name : form . RepoName ,
Description : form . Description ,
IsPrivate : form . Private || setting . Repository . ForcePrivate ,
IsMirror : form . Mirror ,
RemoteAddr : remoteAddr ,
} )
2014-07-26 10:28:04 +04:00
if err == nil {
2015-12-09 04:06:12 +03:00
log . Trace ( "Repository migrated [%d]: %s/%s" , repo . ID , ctxUser . Name , form . RepoName )
2014-09-20 04:11:34 +04:00
ctx . Redirect ( setting . AppSubUrl + "/" + ctxUser . Name + "/" + form . RepoName )
2014-07-26 10:28:04 +04:00
return
}
2014-07-26 08:24:27 +04:00
2014-07-26 10:28:04 +04:00
if repo != nil {
2015-09-01 18:43:53 +03:00
if errDelete := models . DeleteRepository ( ctxUser . Id , repo . ID ) ; errDelete != nil {
2014-07-26 10:28:04 +04:00
log . Error ( 4 , "DeleteRepository: %v" , errDelete )
}
}
2014-07-26 08:24:27 +04:00
2015-11-19 21:45:07 +03:00
if strings . Contains ( err . Error ( ) , "Authentication failed" ) ||
2015-09-06 15:54:08 +03:00
strings . Contains ( err . Error ( ) , "could not read Username" ) {
2014-08-01 08:06:19 +04:00
ctx . Data [ "Err_Auth" ] = true
2015-12-09 04:06:12 +03:00
ctx . RenderWithErr ( ctx . Tr ( "form.auth_failed" , models . HandleCloneUserCredentials ( err . Error ( ) , true ) ) , MIGRATE , & form )
2014-07-26 10:28:04 +04:00
return
2015-11-19 21:45:07 +03:00
} else if strings . Contains ( err . Error ( ) , "fatal:" ) {
ctx . Data [ "Err_CloneAddr" ] = true
2015-12-09 04:06:12 +03:00
ctx . RenderWithErr ( ctx . Tr ( "repo.migrate.failed" , models . HandleCloneUserCredentials ( err . Error ( ) , true ) ) , MIGRATE , & form )
2015-11-19 21:45:07 +03:00
return
2014-07-26 10:28:04 +04:00
}
2015-03-27 00:11:47 +03:00
2015-12-10 20:37:53 +03:00
handleCreateError ( ctx , ctxUser , err , "MigratePost" , MIGRATE , & form )
2014-07-26 10:28:04 +04:00
}
2014-07-26 08:24:27 +04:00
2014-08-11 07:11:18 +04:00
func Action ( ctx * middleware . Context ) {
var err error
switch ctx . Params ( ":action" ) {
case "watch" :
2015-08-08 17:43:14 +03:00
err = models . WatchRepo ( ctx . User . Id , ctx . Repo . Repository . ID , true )
2014-08-11 07:11:18 +04:00
case "unwatch" :
2015-08-08 17:43:14 +03:00
err = models . WatchRepo ( ctx . User . Id , ctx . Repo . Repository . ID , false )
2014-08-11 07:11:18 +04:00
case "star" :
2015-08-08 17:43:14 +03:00
err = models . StarRepo ( ctx . User . Id , ctx . Repo . Repository . ID , true )
2014-08-11 07:11:18 +04:00
case "unstar" :
2015-08-08 17:43:14 +03:00
err = models . StarRepo ( ctx . User . Id , ctx . Repo . Repository . ID , false )
2016-01-07 06:07:17 +03:00
case "desc" : // FIXME: this is not used
2015-02-16 13:51:56 +03:00
if ! ctx . Repo . IsOwner ( ) {
2014-08-11 07:11:18 +04:00
ctx . Error ( 404 )
return
}
ctx . Repo . Repository . Description = ctx . Query ( "desc" )
ctx . Repo . Repository . Website = ctx . Query ( "site" )
2015-03-16 11:52:11 +03:00
err = models . UpdateRepository ( ctx . Repo . Repository , false )
2014-08-11 07:11:18 +04:00
}
if err != nil {
2015-12-21 15:24:11 +03:00
ctx . Handle ( 500 , fmt . Sprintf ( "Action (%s)" , ctx . Params ( ":action" ) ) , err )
2014-08-11 07:11:18 +04:00
return
}
2015-07-23 23:50:05 +03:00
redirectTo := ctx . Query ( "redirect_to" )
if len ( redirectTo ) == 0 {
redirectTo = ctx . Repo . RepoLink
}
ctx . Redirect ( redirectTo )
2014-08-11 07:11:18 +04:00
}
2014-07-26 08:24:27 +04:00
func Download ( ctx * middleware . Context ) {
2014-09-25 01:43:33 +04:00
var (
uri = ctx . Params ( "*" )
refName string
ext string
archivePath string
2014-11-02 08:18:37 +03:00
archiveType git . ArchiveType
2014-09-25 01:43:33 +04:00
)
switch {
case strings . HasSuffix ( uri , ".zip" ) :
ext = ".zip"
2014-07-26 08:24:27 +04:00
archivePath = path . Join ( ctx . Repo . GitRepo . Path , "archives/zip" )
2014-11-02 08:18:37 +03:00
archiveType = git . ZIP
2014-09-25 01:43:33 +04:00
case strings . HasSuffix ( uri , ".tar.gz" ) :
ext = ".tar.gz"
2014-07-26 08:24:27 +04:00
archivePath = path . Join ( ctx . Repo . GitRepo . Path , "archives/targz" )
2014-11-02 08:18:37 +03:00
archiveType = git . TARGZ
2014-07-26 08:24:27 +04:00
default :
2015-12-14 02:20:39 +03:00
log . Trace ( "Unknown format: %s" , uri )
2014-07-26 08:24:27 +04:00
ctx . Error ( 404 )
2014-06-25 13:35:23 +04:00
return
}
2014-09-25 01:43:33 +04:00
refName = strings . TrimSuffix ( uri , ext )
2014-06-25 13:35:23 +04:00
2014-07-26 08:24:27 +04:00
if ! com . IsDir ( archivePath ) {
if err := os . MkdirAll ( archivePath , os . ModePerm ) ; err != nil {
ctx . Handle ( 500 , "Download -> os.MkdirAll(archivePath)" , err )
2014-06-25 13:35:23 +04:00
return
}
}
2014-09-25 01:43:33 +04:00
// Get corresponding commit.
var (
commit * git . Commit
err error
)
gitRepo := ctx . Repo . GitRepo
if gitRepo . IsBranchExist ( refName ) {
2015-12-10 04:46:05 +03:00
commit , err = gitRepo . GetBranchCommit ( refName )
2014-09-25 01:43:33 +04:00
if err != nil {
2015-12-10 04:46:05 +03:00
ctx . Handle ( 500 , "GetBranchCommit" , err )
2014-09-25 01:43:33 +04:00
return
}
} else if gitRepo . IsTagExist ( refName ) {
2015-12-10 04:46:05 +03:00
commit , err = gitRepo . GetTagCommit ( refName )
2014-09-25 01:43:33 +04:00
if err != nil {
2015-12-10 04:46:05 +03:00
ctx . Handle ( 500 , "GetTagCommit" , err )
2014-09-25 01:43:33 +04:00
return
}
} else if len ( refName ) == 40 {
commit , err = gitRepo . GetCommit ( refName )
if err != nil {
2015-12-10 04:46:05 +03:00
ctx . Handle ( 404 , "GetCommit" , nil )
2014-09-25 01:43:33 +04:00
return
}
} else {
2015-12-14 02:20:39 +03:00
ctx . Handle ( 404 , "Download" , nil )
2014-09-25 01:43:33 +04:00
return
}
2015-11-04 06:49:06 +03:00
archivePath = path . Join ( archivePath , base . ShortSha ( commit . ID . String ( ) ) + ext )
2014-07-26 08:24:27 +04:00
if ! com . IsFile ( archivePath ) {
2014-11-02 08:18:37 +03:00
if err := commit . CreateArchive ( archivePath , archiveType ) ; err != nil {
2014-07-26 08:24:27 +04:00
ctx . Handle ( 500 , "Download -> CreateArchive " + archivePath , err )
2014-03-22 21:50:50 +04:00
return
}
}
2016-01-07 23:28:38 +03:00
ctx . ServeFile ( archivePath , ctx . Repo . Repository . Name + "-" + refName + ext )
2014-03-22 21:50:50 +04:00
}