mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 11:33:11 +00:00 
			
		
		
		
	Since `modules/context` has to depend on `models` and many other packages, it should be moved from `modules/context` to `services/context` according to design principles. There is no logic code change on this PR, only move packages. - Move `code.gitea.io/gitea/modules/context` to `code.gitea.io/gitea/services/context` - Move `code.gitea.io/gitea/modules/contexttest` to `code.gitea.io/gitea/services/contexttest` because of depending on context - Move `code.gitea.io/gitea/modules/upload` to `code.gitea.io/gitea/services/context/upload` because of depending on context (cherry picked from commit 29f149bd9f517225a3c9f1ca3fb0a7b5325af696) Conflicts: routers/api/packages/alpine/alpine.go routers/api/v1/repo/issue_reaction.go routers/install/install.go routers/web/admin/config.go routers/web/passkey.go routers/web/repo/search.go routers/web/repo/setting/default_branch.go routers/web/user/home.go routers/web/user/profile.go tests/integration/editor_test.go tests/integration/integration_test.go tests/integration/mirror_push_test.go trivial context conflicts also modified all other occurrences in Forgejo specific files
		
			
				
	
	
		
			300 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | |
| // Copyright 2020 The Gitea Authors.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package user
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 
 | |
| 	activities_model "code.gitea.io/gitea/models/activities"
 | |
| 	user_model "code.gitea.io/gitea/models/user"
 | |
| 	"code.gitea.io/gitea/routers/api/v1/utils"
 | |
| 	"code.gitea.io/gitea/services/context"
 | |
| 	"code.gitea.io/gitea/services/convert"
 | |
| )
 | |
| 
 | |
| // Search search users
 | |
| func Search(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /users/search user userSearch
 | |
| 	// ---
 | |
| 	// summary: Search for users
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: q
 | |
| 	//   in: query
 | |
| 	//   description: keyword
 | |
| 	//   type: string
 | |
| 	// - name: uid
 | |
| 	//   in: query
 | |
| 	//   description: ID of the user to search for
 | |
| 	//   type: integer
 | |
| 	//   format: int64
 | |
| 	// - name: page
 | |
| 	//   in: query
 | |
| 	//   description: page number of results to return (1-based)
 | |
| 	//   type: integer
 | |
| 	// - name: limit
 | |
| 	//   in: query
 | |
| 	//   description: page size of results
 | |
| 	//   type: integer
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     description: "SearchResults of a successful search"
 | |
| 	//     schema:
 | |
| 	//       type: object
 | |
| 	//       properties:
 | |
| 	//         ok:
 | |
| 	//           type: boolean
 | |
| 	//         data:
 | |
| 	//           type: array
 | |
| 	//           items:
 | |
| 	//             "$ref": "#/definitions/User"
 | |
| 
 | |
| 	listOptions := utils.GetListOptions(ctx)
 | |
| 
 | |
| 	uid := ctx.FormInt64("uid")
 | |
| 	var users []*user_model.User
 | |
| 	var maxResults int64
 | |
| 	var err error
 | |
| 
 | |
| 	switch uid {
 | |
| 	case user_model.GhostUserID:
 | |
| 		maxResults = 1
 | |
| 		users = []*user_model.User{user_model.NewGhostUser()}
 | |
| 	case user_model.ActionsUserID:
 | |
| 		maxResults = 1
 | |
| 		users = []*user_model.User{user_model.NewActionsUser()}
 | |
| 	default:
 | |
| 		users, maxResults, err = user_model.SearchUsers(ctx, &user_model.SearchUserOptions{
 | |
| 			Actor:       ctx.Doer,
 | |
| 			Keyword:     ctx.FormTrim("q"),
 | |
| 			UID:         uid,
 | |
| 			Type:        user_model.UserTypeIndividual,
 | |
| 			ListOptions: listOptions,
 | |
| 		})
 | |
| 		if err != nil {
 | |
| 			ctx.JSON(http.StatusInternalServerError, map[string]any{
 | |
| 				"ok":    false,
 | |
| 				"error": err.Error(),
 | |
| 			})
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ctx.SetLinkHeader(int(maxResults), listOptions.PageSize)
 | |
| 	ctx.SetTotalCountHeader(maxResults)
 | |
| 
 | |
| 	ctx.JSON(http.StatusOK, map[string]any{
 | |
| 		"ok":   true,
 | |
| 		"data": convert.ToUsers(ctx, ctx.Doer, users),
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // GetInfo get user's information
 | |
| func GetInfo(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /users/{username} user userGet
 | |
| 	// ---
 | |
| 	// summary: Get a user
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: username
 | |
| 	//   in: path
 | |
| 	//   description: username of user to get
 | |
| 	//   type: string
 | |
| 	//   required: true
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     "$ref": "#/responses/User"
 | |
| 	//   "404":
 | |
| 	//     "$ref": "#/responses/notFound"
 | |
| 
 | |
| 	if !user_model.IsUserVisibleToViewer(ctx, ctx.ContextUser, ctx.Doer) {
 | |
| 		// fake ErrUserNotExist error message to not leak information about existence
 | |
| 		ctx.NotFound("GetUserByName", user_model.ErrUserNotExist{Name: ctx.Params(":username")})
 | |
| 		return
 | |
| 	}
 | |
| 	ctx.JSON(http.StatusOK, convert.ToUser(ctx, ctx.ContextUser, ctx.Doer))
 | |
| }
 | |
| 
 | |
| // GetAuthenticatedUser get current user's information
 | |
| func GetAuthenticatedUser(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /user user userGetCurrent
 | |
| 	// ---
 | |
| 	// summary: Get the authenticated user
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     "$ref": "#/responses/User"
 | |
| 
 | |
| 	ctx.JSON(http.StatusOK, convert.ToUser(ctx, ctx.Doer, ctx.Doer))
 | |
| }
 | |
| 
 | |
| // GetUserHeatmapData is the handler to get a users heatmap
 | |
| func GetUserHeatmapData(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /users/{username}/heatmap user userGetHeatmapData
 | |
| 	// ---
 | |
| 	// summary: Get a user's heatmap
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: username
 | |
| 	//   in: path
 | |
| 	//   description: username of user to get
 | |
| 	//   type: string
 | |
| 	//   required: true
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     "$ref": "#/responses/UserHeatmapData"
 | |
| 	//   "404":
 | |
| 	//     "$ref": "#/responses/notFound"
 | |
| 
 | |
| 	heatmap, err := activities_model.GetUserHeatmapDataByUser(ctx, ctx.ContextUser, ctx.Doer)
 | |
| 	if err != nil {
 | |
| 		ctx.Error(http.StatusInternalServerError, "GetUserHeatmapDataByUser", err)
 | |
| 		return
 | |
| 	}
 | |
| 	ctx.JSON(http.StatusOK, heatmap)
 | |
| }
 | |
| 
 | |
| func ListUserActivityFeeds(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /users/{username}/activities/feeds user userListActivityFeeds
 | |
| 	// ---
 | |
| 	// summary: List a user's activity feeds
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: username
 | |
| 	//   in: path
 | |
| 	//   description: username of user
 | |
| 	//   type: string
 | |
| 	//   required: true
 | |
| 	// - name: only-performed-by
 | |
| 	//   in: query
 | |
| 	//   description: if true, only show actions performed by the requested user
 | |
| 	//   type: boolean
 | |
| 	// - name: date
 | |
| 	//   in: query
 | |
| 	//   description: the date of the activities to be found
 | |
| 	//   type: string
 | |
| 	//   format: date
 | |
| 	// - name: page
 | |
| 	//   in: query
 | |
| 	//   description: page number of results to return (1-based)
 | |
| 	//   type: integer
 | |
| 	// - name: limit
 | |
| 	//   in: query
 | |
| 	//   description: page size of results
 | |
| 	//   type: integer
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     "$ref": "#/responses/ActivityFeedsList"
 | |
| 	//   "404":
 | |
| 	//     "$ref": "#/responses/notFound"
 | |
| 
 | |
| 	includePrivate := ctx.IsSigned && (ctx.Doer.IsAdmin || ctx.Doer.ID == ctx.ContextUser.ID)
 | |
| 	listOptions := utils.GetListOptions(ctx)
 | |
| 
 | |
| 	opts := activities_model.GetFeedsOptions{
 | |
| 		RequestedUser:   ctx.ContextUser,
 | |
| 		Actor:           ctx.Doer,
 | |
| 		IncludePrivate:  includePrivate,
 | |
| 		OnlyPerformedBy: ctx.FormBool("only-performed-by"),
 | |
| 		Date:            ctx.FormString("date"),
 | |
| 		ListOptions:     listOptions,
 | |
| 	}
 | |
| 
 | |
| 	feeds, count, err := activities_model.GetFeeds(ctx, opts)
 | |
| 	if err != nil {
 | |
| 		ctx.Error(http.StatusInternalServerError, "GetFeeds", err)
 | |
| 		return
 | |
| 	}
 | |
| 	ctx.SetTotalCountHeader(count)
 | |
| 
 | |
| 	ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
 | |
| }
 | |
| 
 | |
| // ListBlockedUsers list the authenticated user's blocked users.
 | |
| func ListBlockedUsers(ctx *context.APIContext) {
 | |
| 	// swagger:operation GET /user/list_blocked user userListBlockedUsers
 | |
| 	// ---
 | |
| 	// summary: List the authenticated user's blocked users
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: page
 | |
| 	//   in: query
 | |
| 	//   description: page number of results to return (1-based)
 | |
| 	//   type: integer
 | |
| 	// - name: limit
 | |
| 	//   in: query
 | |
| 	//   description: page size of results
 | |
| 	//   type: integer
 | |
| 	// responses:
 | |
| 	//   "200":
 | |
| 	//     "$ref": "#/responses/BlockedUserList"
 | |
| 
 | |
| 	utils.ListUserBlockedUsers(ctx, ctx.Doer)
 | |
| }
 | |
| 
 | |
| // BlockUser blocks a user from the doer.
 | |
| func BlockUser(ctx *context.APIContext) {
 | |
| 	// swagger:operation PUT /user/block/{username} user userBlockUser
 | |
| 	// ---
 | |
| 	// summary: Blocks a user from the doer.
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: username
 | |
| 	//   in: path
 | |
| 	//   description: username of the user
 | |
| 	//   type: string
 | |
| 	//   required: true
 | |
| 	// responses:
 | |
| 	//   "204":
 | |
| 	//     "$ref": "#/responses/empty"
 | |
| 	//   "404":
 | |
| 	//     "$ref": "#/responses/notFound"
 | |
| 	//   "422":
 | |
| 	//     "$ref": "#/responses/validationError"
 | |
| 
 | |
| 	if ctx.ContextUser.IsOrganization() {
 | |
| 		ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("%s is an organization not a user", ctx.ContextUser.Name))
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	utils.BlockUser(ctx, ctx.Doer, ctx.ContextUser)
 | |
| }
 | |
| 
 | |
| // UnblockUser unblocks a user from the doer.
 | |
| func UnblockUser(ctx *context.APIContext) {
 | |
| 	// swagger:operation PUT /user/unblock/{username} user userUnblockUser
 | |
| 	// ---
 | |
| 	// summary: Unblocks a user from the doer.
 | |
| 	// produces:
 | |
| 	// - application/json
 | |
| 	// parameters:
 | |
| 	// - name: username
 | |
| 	//   in: path
 | |
| 	//   description: username of the user
 | |
| 	//   type: string
 | |
| 	//   required: true
 | |
| 	// responses:
 | |
| 	//   "204":
 | |
| 	//     "$ref": "#/responses/empty"
 | |
| 	//   "404":
 | |
| 	//     "$ref": "#/responses/notFound"
 | |
| 	//   "422":
 | |
| 	//     "$ref": "#/responses/validationError"
 | |
| 
 | |
| 	if ctx.ContextUser.IsOrganization() {
 | |
| 		ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("%s is an organization not a user", ctx.ContextUser.Name))
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	utils.UnblockUser(ctx, ctx.Doer, ctx.ContextUser)
 | |
| }
 |