2014-03-15 20:03:23 +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 middleware
import (
2014-03-16 10:28:24 +04:00
"errors"
2014-03-20 08:12:33 +04:00
"fmt"
2014-05-06 03:58:13 +04:00
"net/url"
2014-03-17 12:47:42 +04:00
"strings"
2014-03-16 10:28:24 +04:00
2014-03-30 20:11:28 +04:00
"github.com/go-martini/martini"
2014-03-15 20:03:23 +04:00
2014-03-30 06:09:59 +04:00
"github.com/gogits/git"
2014-03-15 20:03:23 +04:00
"github.com/gogits/gogs/models"
2014-04-11 08:01:38 +04:00
"github.com/gogits/gogs/modules/log"
2014-05-26 04:11:25 +04:00
"github.com/gogits/gogs/modules/setting"
2014-03-15 20:03:23 +04:00
)
2014-03-30 09:30:17 +04:00
func RepoAssignment ( redirect bool , args ... bool ) martini . Handler {
2014-03-15 20:03:23 +04:00
return func ( ctx * Context , params martini . Params ) {
2014-03-30 09:30:17 +04:00
// valid brachname
var validBranch bool
// display bare quick start if it is a bare repo
var displayBare bool
if len ( args ) >= 1 {
2014-06-13 01:47:23 +04:00
validBranch = args [ 0 ]
2014-03-30 09:30:17 +04:00
}
if len ( args ) >= 2 {
2014-06-13 01:47:23 +04:00
displayBare = args [ 1 ]
2014-03-30 09:30:17 +04:00
}
2014-03-15 20:03:23 +04:00
var (
2014-07-04 09:23:11 +04:00
user * models . User
err error
2014-03-15 20:03:23 +04:00
)
2014-03-30 06:09:59 +04:00
userName := params [ "username" ]
repoName := params [ "reponame" ]
2014-04-16 01:43:25 +04:00
refName := params [ "branchname" ]
2014-03-30 06:09:59 +04:00
2014-06-28 08:40:07 +04:00
// TODO: need more advanced onwership and access level check.
2014-05-08 20:24:11 +04:00
// Collaborators who have write access can be seen as owners.
if ctx . IsSigned {
2014-06-25 08:44:48 +04:00
ctx . Repo . IsOwner , err = models . HasAccess ( ctx . User . Name , userName + "/" + repoName , models . WRITABLE )
2014-05-08 20:24:11 +04:00
if err != nil {
ctx . Handle ( 500 , "RepoAssignment(HasAccess)" , err )
return
}
2014-07-04 09:23:11 +04:00
ctx . Repo . IsTrueOwner = ctx . User . LowerName == strings . ToLower ( userName )
2014-05-08 20:24:11 +04:00
}
2014-03-15 20:03:23 +04:00
2014-07-04 09:23:11 +04:00
if ! ctx . Repo . IsTrueOwner {
2014-05-08 20:24:11 +04:00
user , err = models . GetUserByName ( userName )
2014-03-15 20:03:23 +04:00
if err != nil {
2014-04-30 11:44:28 +04:00
if err == models . ErrUserNotExist {
2014-05-08 20:24:11 +04:00
ctx . Handle ( 404 , "RepoAssignment(GetUserByName)" , err )
2014-04-30 11:44:28 +04:00
return
} else if redirect {
2014-03-19 17:57:55 +04:00
ctx . Redirect ( "/" )
2014-03-15 20:03:23 +04:00
return
}
2014-05-08 20:24:11 +04:00
ctx . Handle ( 500 , "RepoAssignment(GetUserByName)" , err )
2014-03-15 20:03:23 +04:00
return
}
} else {
user = ctx . User
}
if user == nil {
if redirect {
2014-03-19 17:57:55 +04:00
ctx . Redirect ( "/" )
2014-03-15 20:03:23 +04:00
return
}
2014-04-30 11:44:28 +04:00
ctx . Handle ( 403 , "RepoAssignment" , errors . New ( "invliad user account for single repository" ) )
2014-03-15 20:03:23 +04:00
return
}
2014-04-12 05:47:39 +04:00
ctx . Repo . Owner = user
2014-03-15 20:03:23 +04:00
2014-07-04 09:23:11 +04:00
// Organization owner team members are true owners as well.
if ctx . Repo . Owner . IsOrganization ( ) && ctx . Repo . Owner . IsOrgOwner ( ctx . User . Id ) {
ctx . Repo . IsTrueOwner = true
}
2014-03-15 20:03:23 +04:00
// get repository
2014-03-30 06:09:59 +04:00
repo , err := models . GetRepositoryByName ( user . Id , repoName )
2014-03-15 20:03:23 +04:00
if err != nil {
2014-03-28 05:15:53 +04:00
if err == models . ErrRepoNotExist {
ctx . Handle ( 404 , "RepoAssignment" , err )
2014-04-12 05:47:39 +04:00
return
2014-03-28 05:15:53 +04:00
} else if redirect {
2014-03-19 17:57:55 +04:00
ctx . Redirect ( "/" )
2014-03-15 20:03:23 +04:00
return
}
2014-04-11 06:03:31 +04:00
ctx . Handle ( 500 , "RepoAssignment" , err )
2014-03-30 06:09:59 +04:00
return
}
2014-04-12 05:47:39 +04:00
2014-05-14 03:26:13 +04:00
// Check if the mirror repository owner(mirror repository doesn't have access).
2014-05-14 16:30:35 +04:00
if ctx . IsSigned && ! ctx . Repo . IsOwner && repo . OwnerId == ctx . User . Id {
2014-05-14 03:26:13 +04:00
ctx . Repo . IsOwner = true
}
2014-04-12 05:47:39 +04:00
// Check access.
2014-05-09 03:17:43 +04:00
if repo . IsPrivate && ! ctx . Repo . IsOwner {
2014-04-12 05:47:39 +04:00
if ctx . User == nil {
ctx . Handle ( 404 , "RepoAssignment(HasAccess)" , nil )
return
}
2014-06-25 08:44:48 +04:00
hasAccess , err := models . HasAccess ( ctx . User . Name , ctx . Repo . Owner . Name + "/" + repo . Name , models . READABLE )
2014-04-12 05:47:39 +04:00
if err != nil {
ctx . Handle ( 500 , "RepoAssignment(HasAccess)" , err )
return
} else if ! hasAccess {
ctx . Handle ( 404 , "RepoAssignment(HasAccess)" , nil )
return
}
}
ctx . Repo . HasAccess = true
ctx . Data [ "HasAccess" ] = true
2014-04-13 06:30:00 +04:00
if repo . IsMirror {
ctx . Repo . Mirror , err = models . GetMirror ( repo . Id )
if err != nil {
ctx . Handle ( 500 , "RepoAssignment(GetMirror)" , err )
return
}
ctx . Data [ "MirrorInterval" ] = ctx . Repo . Mirror . Interval
}
2014-04-02 20:43:31 +04:00
repo . NumOpenIssues = repo . NumIssues - repo . NumClosedIssues
2014-05-12 22:06:42 +04:00
repo . NumOpenMilestones = repo . NumMilestones - repo . NumClosedMilestones
2014-03-30 06:09:59 +04:00
ctx . Repo . Repository = repo
2014-03-30 09:30:17 +04:00
ctx . Data [ "IsBareRepo" ] = ctx . Repo . Repository . IsBare
2014-03-30 06:09:59 +04:00
gitRepo , err := git . OpenRepository ( models . RepoPath ( userName , repoName ) )
if err != nil {
2014-04-11 06:03:31 +04:00
ctx . Handle ( 500 , "RepoAssignment Invalid repo " + models . RepoPath ( userName , repoName ) , err )
2014-03-15 20:03:23 +04:00
return
}
2014-03-30 06:09:59 +04:00
ctx . Repo . GitRepo = gitRepo
2014-03-30 07:38:41 +04:00
ctx . Repo . RepoLink = "/" + user . Name + "/" + repo . Name
2014-04-14 05:00:12 +04:00
tags , err := ctx . Repo . GitRepo . GetTags ( )
if err != nil {
ctx . Handle ( 500 , "RepoAssignment(GetTags))" , err )
return
}
ctx . Repo . Repository . NumTags = len ( tags )
2014-03-30 07:38:41 +04:00
ctx . Data [ "Title" ] = user . Name + "/" + repo . Name
ctx . Data [ "Repository" ] = repo
ctx . Data [ "Owner" ] = user
ctx . Data [ "RepoLink" ] = ctx . Repo . RepoLink
ctx . Data [ "IsRepositoryOwner" ] = ctx . Repo . IsOwner
2014-07-04 09:23:11 +04:00
ctx . Data [ "IsRepositoryTrueOwner" ] = ctx . Repo . IsTrueOwner
2014-03-30 09:30:17 +04:00
ctx . Data [ "BranchName" ] = ""
2014-03-30 07:38:41 +04:00
2014-05-26 04:11:25 +04:00
if setting . SshPort != 22 {
ctx . Repo . CloneLink . SSH = fmt . Sprintf ( "ssh://%s@%s/%s/%s.git" , setting . RunUser , setting . Domain , user . LowerName , repo . LowerName )
2014-05-11 20:17:10 +04:00
} else {
2014-05-26 04:11:25 +04:00
ctx . Repo . CloneLink . SSH = fmt . Sprintf ( "%s@%s:%s/%s.git" , setting . RunUser , setting . Domain , user . LowerName , repo . LowerName )
2014-05-11 19:18:10 +04:00
}
2014-05-26 04:11:25 +04:00
ctx . Repo . CloneLink . HTTPS = fmt . Sprintf ( "%s%s/%s.git" , setting . AppUrl , user . LowerName , repo . LowerName )
2014-03-30 07:38:41 +04:00
ctx . Data [ "CloneLink" ] = ctx . Repo . CloneLink
2014-04-13 12:08:25 +04:00
if ctx . Repo . Repository . IsGoget {
2014-05-26 04:11:25 +04:00
ctx . Data [ "GoGetLink" ] = fmt . Sprintf ( "%s%s/%s" , setting . AppUrl , user . LowerName , repo . LowerName )
ctx . Data [ "GoGetImport" ] = fmt . Sprintf ( "%s/%s/%s" , setting . Domain , user . LowerName , repo . LowerName )
2014-04-13 12:08:25 +04:00
}
2014-03-30 09:30:17 +04:00
// when repo is bare, not valid branch
if ! ctx . Repo . Repository . IsBare && validBranch {
detect :
2014-04-16 01:43:25 +04:00
if len ( refName ) > 0 {
if gitRepo . IsBranchExist ( refName ) {
2014-03-30 09:30:17 +04:00
ctx . Repo . IsBranch = true
2014-04-16 01:43:25 +04:00
ctx . Repo . BranchName = refName
2014-03-30 09:30:17 +04:00
2014-04-16 01:43:25 +04:00
ctx . Repo . Commit , err = gitRepo . GetCommitOfBranch ( refName )
2014-03-30 09:30:17 +04:00
if err != nil {
ctx . Handle ( 404 , "RepoAssignment invalid branch" , nil )
return
}
2014-04-16 01:43:25 +04:00
ctx . Repo . CommitId = ctx . Repo . Commit . Id . String ( )
} else if gitRepo . IsTagExist ( refName ) {
2014-06-28 19:56:41 +04:00
ctx . Repo . IsTag = true
2014-04-16 01:43:25 +04:00
ctx . Repo . BranchName = refName
2014-03-30 09:30:17 +04:00
2014-06-28 19:56:41 +04:00
ctx . Repo . Tag , err = gitRepo . GetTag ( refName )
2014-04-16 01:43:25 +04:00
if err != nil {
ctx . Handle ( 404 , "RepoAssignment invalid tag" , nil )
return
}
2014-06-28 19:56:41 +04:00
ctx . Repo . Commit , _ = ctx . Repo . Tag . Commit ( )
2014-04-13 05:35:36 +04:00
ctx . Repo . CommitId = ctx . Repo . Commit . Id . String ( )
2014-04-16 01:43:25 +04:00
} else if len ( refName ) == 40 {
2014-03-30 09:30:17 +04:00
ctx . Repo . IsCommit = true
2014-04-16 01:43:25 +04:00
ctx . Repo . CommitId = refName
ctx . Repo . BranchName = refName
2014-03-30 09:30:17 +04:00
2014-04-16 01:43:25 +04:00
ctx . Repo . Commit , err = gitRepo . GetCommit ( refName )
2014-03-30 09:30:17 +04:00
if err != nil {
ctx . Handle ( 404 , "RepoAssignment invalid commit" , nil )
return
}
} else {
ctx . Handle ( 404 , "RepoAssignment invalid repo" , nil )
2014-03-30 06:09:59 +04:00
return
}
} else {
2014-04-16 01:43:25 +04:00
if len ( refName ) == 0 {
2014-05-01 20:03:10 +04:00
if gitRepo . IsBranchExist ( ctx . Repo . Repository . DefaultBranch ) {
refName = ctx . Repo . Repository . DefaultBranch
} else {
brs , err := gitRepo . GetBranches ( )
if err != nil {
ctx . Handle ( 500 , "RepoAssignment(GetBranches))" , err )
return
}
refName = brs [ 0 ]
}
2014-04-11 06:03:31 +04:00
}
2014-03-30 09:30:17 +04:00
goto detect
2014-03-30 06:09:59 +04:00
}
2014-03-30 13:09:19 +04:00
ctx . Data [ "IsBranch" ] = ctx . Repo . IsBranch
ctx . Data [ "IsCommit" ] = ctx . Repo . IsCommit
2014-03-30 09:30:17 +04:00
}
2014-03-30 06:09:59 +04:00
2014-04-20 07:37:04 +04:00
log . Debug ( "displayBare: %v; IsBare: %v" , displayBare , ctx . Repo . Repository . IsBare )
2014-03-30 09:30:17 +04:00
// repo is bare and display enable
if displayBare && ctx . Repo . Repository . IsBare {
2014-04-20 06:13:22 +04:00
log . Debug ( "Bare repository: %s" , ctx . Repo . RepoLink )
2014-03-30 09:30:17 +04:00
ctx . HTML ( 200 , "repo/single_bare" )
return
2014-03-30 06:09:59 +04:00
}
2014-03-15 20:03:23 +04:00
2014-03-30 06:09:59 +04:00
if ctx . IsSigned {
2014-03-20 10:25:21 +04:00
ctx . Repo . IsWatching = models . IsWatching ( ctx . User . Id , repo . Id )
}
2014-03-30 06:09:59 +04:00
ctx . Data [ "BranchName" ] = ctx . Repo . BranchName
2014-06-28 19:56:41 +04:00
ctx . Data [ "TagName" ] = ctx . Repo . TagName
2014-04-13 05:35:36 +04:00
brs , err := ctx . Repo . GitRepo . GetBranches ( )
2014-04-11 08:01:38 +04:00
if err != nil {
log . Error ( "RepoAssignment(GetBranches): %v" , err )
}
ctx . Data [ "Branches" ] = brs
2014-03-30 06:09:59 +04:00
ctx . Data [ "CommitId" ] = ctx . Repo . CommitId
2014-03-20 17:28:12 +04:00
ctx . Data [ "IsRepositoryWatching" ] = ctx . Repo . IsWatching
2014-03-15 20:03:23 +04:00
}
}
2014-05-06 03:58:13 +04:00
func RequireOwner ( ) martini . Handler {
return func ( ctx * Context ) {
2014-07-04 09:23:11 +04:00
if ! ctx . Repo . IsTrueOwner {
2014-05-06 03:58:13 +04:00
if ! ctx . IsSigned {
ctx . SetCookie ( "redirect_to" , "/" + url . QueryEscape ( ctx . Req . RequestURI ) )
ctx . Redirect ( "/user/login" )
return
}
ctx . Handle ( 404 , ctx . Req . RequestURI , nil )
return
}
}
}