2015-12-05 01:16:42 +03:00
// Copyright 2015 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 v1
import (
"strings"
"github.com/go-macaron/binding"
"gopkg.in/macaron.v1"
api "github.com/gogits/go-gogs-client"
"github.com/gogits/gogs/models"
"github.com/gogits/gogs/modules/auth"
2016-03-11 19:56:52 +03:00
"github.com/gogits/gogs/modules/context"
2015-12-06 01:13:13 +03:00
"github.com/gogits/gogs/routers/api/v1/admin"
2015-12-05 01:16:42 +03:00
"github.com/gogits/gogs/routers/api/v1/misc"
2015-12-17 10:28:47 +03:00
"github.com/gogits/gogs/routers/api/v1/org"
2015-12-05 01:16:42 +03:00
"github.com/gogits/gogs/routers/api/v1/repo"
"github.com/gogits/gogs/routers/api/v1/user"
)
2016-08-05 03:08:01 +03:00
func repoAssignment ( ) macaron . Handler {
2016-03-14 01:49:16 +03:00
return func ( ctx * context . APIContext ) {
2015-12-05 01:16:42 +03:00
userName := ctx . Params ( ":username" )
repoName := ctx . Params ( ":reponame" )
var (
owner * models . User
err error
)
// Check if the user is the same as the repository owner.
if ctx . IsSigned && ctx . User . LowerName == strings . ToLower ( userName ) {
owner = ctx . User
} else {
owner , err = models . GetUserByName ( userName )
if err != nil {
if models . IsErrUserNotExist ( err ) {
2016-03-14 01:49:16 +03:00
ctx . Status ( 404 )
2015-12-05 01:16:42 +03:00
} else {
2016-03-14 01:49:16 +03:00
ctx . Error ( 500 , "GetUserByName" , err )
2015-12-05 01:16:42 +03:00
}
return
}
}
ctx . Repo . Owner = owner
// Get repository.
2016-07-23 20:08:22 +03:00
repo , err := models . GetRepositoryByName ( owner . ID , repoName )
2015-12-05 01:16:42 +03:00
if err != nil {
if models . IsErrRepoNotExist ( err ) {
2016-03-14 01:49:16 +03:00
ctx . Status ( 404 )
2015-12-05 01:16:42 +03:00
} else {
2016-03-14 01:49:16 +03:00
ctx . Error ( 500 , "GetRepositoryByName" , err )
2015-12-05 01:16:42 +03:00
}
return
} else if err = repo . GetOwner ( ) ; err != nil {
2016-03-14 01:49:16 +03:00
ctx . Error ( 500 , "GetOwner" , err )
2015-12-05 01:16:42 +03:00
return
}
2016-03-14 06:20:22 +03:00
if ctx . IsSigned && ctx . User . IsAdmin {
ctx . Repo . AccessMode = models . ACCESS_MODE_OWNER
} else {
mode , err := models . AccessLevel ( ctx . User , repo )
if err != nil {
ctx . Error ( 500 , "AccessLevel" , err )
return
}
ctx . Repo . AccessMode = mode
2015-12-05 01:16:42 +03:00
}
2016-03-14 06:20:22 +03:00
if ! ctx . Repo . HasAccess ( ) {
2016-03-14 01:49:16 +03:00
ctx . Status ( 404 )
2015-12-05 01:16:42 +03:00
return
}
ctx . Repo . Repository = repo
}
}
// Contexter middleware already checks token for user sign in process.
2016-08-05 03:08:01 +03:00
func reqToken ( ) macaron . Handler {
2016-03-11 19:56:52 +03:00
return func ( ctx * context . Context ) {
2015-12-05 01:16:42 +03:00
if ! ctx . IsSigned {
ctx . Error ( 401 )
return
}
}
}
2016-08-05 03:08:01 +03:00
func reqBasicAuth ( ) macaron . Handler {
2016-03-11 19:56:52 +03:00
return func ( ctx * context . Context ) {
2015-12-05 01:16:42 +03:00
if ! ctx . IsBasicAuth {
ctx . Error ( 401 )
return
}
}
}
2016-08-05 03:08:01 +03:00
func reqAdmin ( ) macaron . Handler {
2016-03-11 19:56:52 +03:00
return func ( ctx * context . Context ) {
2016-07-23 12:56:37 +03:00
if ! ctx . IsSigned || ! ctx . User . IsAdmin {
2015-12-05 01:16:42 +03:00
ctx . Error ( 403 )
return
}
}
}
2016-08-05 03:08:01 +03:00
func orgAssignment ( args ... bool ) macaron . Handler {
2016-03-26 01:04:02 +03:00
var (
2016-04-05 02:41:34 +03:00
assignOrg bool
2016-03-26 01:04:02 +03:00
assignTeam bool
)
if len ( args ) > 0 {
2016-04-05 02:41:34 +03:00
assignOrg = args [ 0 ]
}
if len ( args ) > 1 {
assignTeam = args [ 1 ]
2016-03-26 01:04:02 +03:00
}
return func ( ctx * context . APIContext ) {
2016-04-05 02:41:34 +03:00
ctx . Org = new ( context . APIOrganization )
var err error
if assignOrg {
ctx . Org . Organization , err = models . GetUserByName ( ctx . Params ( ":orgname" ) )
if err != nil {
if models . IsErrUserNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetUserByName" , err )
}
return
2016-03-26 01:04:02 +03:00
}
}
if assignTeam {
ctx . Org . Team , err = models . GetTeamByID ( ctx . ParamsInt64 ( ":teamid" ) )
if err != nil {
if models . IsErrUserNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetTeamById" , err )
}
return
}
}
}
}
2016-08-05 03:08:01 +03:00
func mustEnableIssues ( ctx * context . APIContext ) {
2016-08-05 02:32:02 +03:00
if ! ctx . Repo . Repository . EnableIssues || ctx . Repo . Repository . EnableExternalTracker {
ctx . Status ( 404 )
return
}
}
2015-12-05 01:16:42 +03:00
// RegisterRoutes registers all v1 APIs routes to web application.
// FIXME: custom form error response
func RegisterRoutes ( m * macaron . Macaron ) {
bind := binding . Bind
m . Group ( "/v1" , func ( ) {
// Miscellaneous
m . Post ( "/markdown" , bind ( api . MarkdownOption { } ) , misc . Markdown )
m . Post ( "/markdown/raw" , misc . MarkdownRaw )
// Users
m . Group ( "/users" , func ( ) {
m . Get ( "/search" , user . Search )
m . Group ( "/:username" , func ( ) {
m . Get ( "" , user . GetInfo )
m . Group ( "/tokens" , func ( ) {
m . Combo ( "" ) . Get ( user . ListAccessTokens ) .
Post ( bind ( api . CreateAccessTokenOption { } ) , user . CreateAccessToken )
2016-08-05 03:08:01 +03:00
} , reqBasicAuth ( ) )
2015-12-05 01:16:42 +03:00
} )
} )
m . Group ( "/users" , func ( ) {
m . Group ( "/:username" , func ( ) {
2015-12-06 01:13:13 +03:00
m . Get ( "/keys" , user . ListPublicKeys )
2015-12-21 15:24:11 +03:00
m . Get ( "/followers" , user . ListFollowers )
m . Group ( "/following" , func ( ) {
m . Get ( "" , user . ListFollowing )
m . Get ( "/:target" , user . CheckFollowing )
} )
2015-12-05 01:16:42 +03:00
} )
2016-08-05 03:08:01 +03:00
} , reqToken ( ) )
2015-12-05 01:16:42 +03:00
m . Group ( "/user" , func ( ) {
2015-12-21 15:24:11 +03:00
m . Combo ( "/emails" ) . Get ( user . ListEmails ) .
Post ( bind ( api . CreateEmailOption { } ) , user . AddEmail ) .
Delete ( bind ( api . CreateEmailOption { } ) , user . DeleteEmail )
m . Get ( "/followers" , user . ListMyFollowers )
m . Group ( "/following" , func ( ) {
m . Get ( "" , user . ListMyFollowing )
m . Combo ( "/:username" ) . Get ( user . CheckMyFollowing ) . Put ( user . Follow ) . Delete ( user . Unfollow )
} )
2015-12-05 01:16:42 +03:00
m . Group ( "/keys" , func ( ) {
m . Combo ( "" ) . Get ( user . ListMyPublicKeys ) .
Post ( bind ( api . CreateKeyOption { } ) , user . CreatePublicKey )
m . Combo ( "/:id" ) . Get ( user . GetPublicKey ) .
Delete ( user . DeletePublicKey )
} )
2016-08-05 03:08:01 +03:00
} , reqToken ( ) )
2015-12-05 01:16:42 +03:00
// Repositories
2016-08-05 03:08:01 +03:00
m . Combo ( "/user/repos" , reqToken ( ) ) . Get ( repo . ListMyRepos ) .
2015-12-05 01:16:42 +03:00
Post ( bind ( api . CreateRepoOption { } ) , repo . Create )
2016-08-05 03:08:01 +03:00
m . Post ( "/org/:org/repos" , reqToken ( ) , bind ( api . CreateRepoOption { } ) , repo . CreateOrgRepo )
2015-12-05 01:16:42 +03:00
m . Group ( "/repos" , func ( ) {
m . Get ( "/search" , repo . Search )
} )
m . Group ( "/repos" , func ( ) {
m . Post ( "/migrate" , bind ( auth . MigrateRepoForm { } ) , repo . Migrate )
m . Combo ( "/:username/:reponame" ) . Get ( repo . Get ) .
Delete ( repo . Delete )
m . Group ( "/:username/:reponame" , func ( ) {
2016-07-17 03:08:38 +03:00
m . Group ( "/hooks" , func ( ) {
m . Combo ( "" ) . Get ( repo . ListHooks ) .
Post ( bind ( api . CreateHookOption { } ) , repo . CreateHook )
2016-07-17 04:25:30 +03:00
m . Combo ( "/:id" ) . Patch ( bind ( api . EditHookOption { } ) , repo . EditHook ) .
2016-07-17 03:08:38 +03:00
Delete ( repo . DeleteHook )
} )
2016-03-11 19:56:52 +03:00
m . Get ( "/raw/*" , context . RepoRef ( ) , repo . GetRawFile )
2015-12-05 01:16:42 +03:00
m . Get ( "/archive/*" , repo . GetArchive )
2016-01-15 21:24:03 +03:00
m . Group ( "/branches" , func ( ) {
2016-03-11 19:56:52 +03:00
m . Get ( "" , repo . ListBranches )
m . Get ( "/:branchname" , repo . GetBranch )
2016-01-15 21:24:03 +03:00
} )
2015-12-05 01:16:42 +03:00
m . Group ( "/keys" , func ( ) {
m . Combo ( "" ) . Get ( repo . ListDeployKeys ) .
Post ( bind ( api . CreateKeyOption { } ) , repo . CreateDeployKey )
m . Combo ( "/:id" ) . Get ( repo . GetDeployKey ) .
Delete ( repo . DeleteDeploykey )
} )
2016-03-14 06:20:22 +03:00
m . Group ( "/issues" , func ( ) {
m . Combo ( "" ) . Get ( repo . ListIssues ) . Post ( bind ( api . CreateIssueOption { } ) , repo . CreateIssue )
2016-08-03 19:24:16 +03:00
m . Group ( "/:index" , func ( ) {
m . Combo ( "" ) . Get ( repo . GetIssue ) . Patch ( bind ( api . EditIssueOption { } ) , repo . EditIssue )
m . Group ( "/labels" , func ( ) {
2016-08-03 21:51:22 +03:00
m . Combo ( "" ) . Get ( repo . ListIssueLabels ) .
2016-08-03 19:24:16 +03:00
Post ( bind ( api . IssueLabelsOption { } ) , repo . AddIssueLabels ) .
Put ( bind ( api . IssueLabelsOption { } ) , repo . ReplaceIssueLabels ) .
Delete ( repo . ClearIssueLabels )
m . Delete ( "/:id" , repo . DeleteIssueLabel )
} )
} )
2016-08-05 03:08:01 +03:00
} , mustEnableIssues )
2016-08-03 19:24:16 +03:00
m . Group ( "/labels" , func ( ) {
m . Combo ( "" ) . Get ( repo . ListLabels ) .
2016-08-03 21:51:22 +03:00
Post ( bind ( api . CreateLabelOption { } ) , repo . CreateLabel )
m . Combo ( "/:id" ) . Get ( repo . GetLabel ) . Patch ( bind ( api . EditLabelOption { } ) , repo . EditLabel ) .
2016-08-03 19:24:16 +03:00
Delete ( repo . DeleteLabel )
2016-03-14 01:49:16 +03:00
} )
2016-08-05 03:08:01 +03:00
} , repoAssignment ( ) )
} , reqToken ( ) )
2015-12-05 01:16:42 +03:00
2015-12-17 10:28:47 +03:00
// Organizations
2016-08-05 03:08:01 +03:00
m . Get ( "/user/orgs" , reqToken ( ) , org . ListMyOrgs )
2015-12-17 10:28:47 +03:00
m . Get ( "/users/:username/orgs" , org . ListUserOrgs )
2016-03-21 19:53:04 +03:00
m . Group ( "/orgs/:orgname" , func ( ) {
m . Combo ( "" ) . Get ( org . Get ) . Patch ( bind ( api . EditOrgOption { } ) , org . Edit )
m . Combo ( "/teams" ) . Get ( org . ListTeams )
2016-08-05 03:08:01 +03:00
} , orgAssignment ( true ) )
2015-12-17 10:28:47 +03:00
2016-03-11 19:56:52 +03:00
m . Any ( "/*" , func ( ctx * context . Context ) {
2015-12-05 01:16:42 +03:00
ctx . Error ( 404 )
} )
2015-12-06 01:13:13 +03:00
m . Group ( "/admin" , func ( ) {
m . Group ( "/users" , func ( ) {
m . Post ( "" , bind ( api . CreateUserOption { } ) , admin . CreateUser )
m . Group ( "/:username" , func ( ) {
m . Combo ( "" ) . Patch ( bind ( api . EditUserOption { } ) , admin . EditUser ) .
Delete ( admin . DeleteUser )
2016-01-08 03:49:03 +03:00
m . Post ( "/keys" , bind ( api . CreateKeyOption { } ) , admin . CreatePublicKey )
2015-12-17 10:28:47 +03:00
m . Post ( "/orgs" , bind ( api . CreateOrgOption { } ) , admin . CreateOrg )
2015-12-18 06:57:41 +03:00
m . Post ( "/repos" , bind ( api . CreateRepoOption { } ) , admin . CreateRepo )
2015-12-06 01:13:13 +03:00
} )
} )
2016-03-21 19:47:54 +03:00
m . Group ( "/orgs/:orgname" , func ( ) {
2016-03-26 01:04:02 +03:00
m . Group ( "/teams" , func ( ) {
2016-08-05 03:08:01 +03:00
m . Post ( "" , orgAssignment ( true ) , bind ( api . CreateTeamOption { } ) , admin . CreateTeam )
2016-03-26 01:04:02 +03:00
} )
2016-03-21 19:47:54 +03:00
} )
2016-04-05 02:41:34 +03:00
m . Group ( "/teams" , func ( ) {
m . Group ( "/:teamid" , func ( ) {
m . Combo ( "/members/:username" ) . Put ( admin . AddTeamMember ) . Delete ( admin . RemoveTeamMember )
m . Combo ( "/repos/:reponame" ) . Put ( admin . AddTeamRepository ) . Delete ( admin . RemoveTeamRepository )
2016-08-05 03:08:01 +03:00
} , orgAssignment ( false , true ) )
2016-04-05 02:41:34 +03:00
} )
2016-08-05 03:08:01 +03:00
} , reqAdmin ( ) )
2016-03-14 01:49:16 +03:00
} , context . APIContexter ( ) )
2015-12-05 01:16:42 +03:00
}