mirror of
				https://github.com/go-gitea/gitea
				synced 2025-09-28 03:28:13 +00:00 
			
		
		
		
	Include file extension checks in attachment API (#32151)
From testing, I found that issue posters and users with repository write access are able to edit attachment names in a way that circumvents the instance-level file extension restrictions using the edit attachment APIs. This snapshot adds checks for these endpoints.
This commit is contained in:
		| @@ -12,7 +12,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/web" | 	"code.gitea.io/gitea/modules/web" | ||||||
| 	"code.gitea.io/gitea/services/attachment" | 	attachment_service "code.gitea.io/gitea/services/attachment" | ||||||
| 	"code.gitea.io/gitea/services/context" | 	"code.gitea.io/gitea/services/context" | ||||||
| 	"code.gitea.io/gitea/services/context/upload" | 	"code.gitea.io/gitea/services/context/upload" | ||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| @@ -181,7 +181,7 @@ func CreateIssueAttachment(ctx *context.APIContext) { | |||||||
| 		filename = query | 		filename = query | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	attachment, err := attachment.UploadAttachment(ctx, file, setting.Attachment.AllowedTypes, header.Size, &repo_model.Attachment{ | 	attachment, err := attachment_service.UploadAttachment(ctx, file, setting.Attachment.AllowedTypes, header.Size, &repo_model.Attachment{ | ||||||
| 		Name:       filename, | 		Name:       filename, | ||||||
| 		UploaderID: ctx.Doer.ID, | 		UploaderID: ctx.Doer.ID, | ||||||
| 		RepoID:     ctx.Repo.Repository.ID, | 		RepoID:     ctx.Repo.Repository.ID, | ||||||
| @@ -247,6 +247,8 @@ func EditIssueAttachment(ctx *context.APIContext) { | |||||||
| 	//     "$ref": "#/responses/Attachment" | 	//     "$ref": "#/responses/Attachment" | ||||||
| 	//   "404": | 	//   "404": | ||||||
| 	//     "$ref": "#/responses/error" | 	//     "$ref": "#/responses/error" | ||||||
|  | 	//   "422": | ||||||
|  | 	//     "$ref": "#/responses/validationError" | ||||||
| 	//   "423": | 	//   "423": | ||||||
| 	//     "$ref": "#/responses/repoArchivedError" | 	//     "$ref": "#/responses/repoArchivedError" | ||||||
|  |  | ||||||
| @@ -261,8 +263,13 @@ func EditIssueAttachment(ctx *context.APIContext) { | |||||||
| 		attachment.Name = form.Name | 		attachment.Name = form.Name | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := repo_model.UpdateAttachment(ctx, attachment); err != nil { | 	if err := attachment_service.UpdateAttachment(ctx, setting.Attachment.AllowedTypes, attachment); err != nil { | ||||||
|  | 		if upload.IsErrFileTypeForbidden(err) { | ||||||
|  | 			ctx.Error(http.StatusUnprocessableEntity, "", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", err) | 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", err) | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attachment)) | 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attachment)) | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/web" | 	"code.gitea.io/gitea/modules/web" | ||||||
| 	"code.gitea.io/gitea/services/attachment" | 	attachment_service "code.gitea.io/gitea/services/attachment" | ||||||
| 	"code.gitea.io/gitea/services/context" | 	"code.gitea.io/gitea/services/context" | ||||||
| 	"code.gitea.io/gitea/services/context/upload" | 	"code.gitea.io/gitea/services/context/upload" | ||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| @@ -189,7 +189,7 @@ func CreateIssueCommentAttachment(ctx *context.APIContext) { | |||||||
| 		filename = query | 		filename = query | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	attachment, err := attachment.UploadAttachment(ctx, file, setting.Attachment.AllowedTypes, header.Size, &repo_model.Attachment{ | 	attachment, err := attachment_service.UploadAttachment(ctx, file, setting.Attachment.AllowedTypes, header.Size, &repo_model.Attachment{ | ||||||
| 		Name:       filename, | 		Name:       filename, | ||||||
| 		UploaderID: ctx.Doer.ID, | 		UploaderID: ctx.Doer.ID, | ||||||
| 		RepoID:     ctx.Repo.Repository.ID, | 		RepoID:     ctx.Repo.Repository.ID, | ||||||
| @@ -263,6 +263,8 @@ func EditIssueCommentAttachment(ctx *context.APIContext) { | |||||||
| 	//     "$ref": "#/responses/Attachment" | 	//     "$ref": "#/responses/Attachment" | ||||||
| 	//   "404": | 	//   "404": | ||||||
| 	//     "$ref": "#/responses/error" | 	//     "$ref": "#/responses/error" | ||||||
|  | 	//   "422": | ||||||
|  | 	//     "$ref": "#/responses/validationError" | ||||||
| 	//   "423": | 	//   "423": | ||||||
| 	//     "$ref": "#/responses/repoArchivedError" | 	//     "$ref": "#/responses/repoArchivedError" | ||||||
| 	attach := getIssueCommentAttachmentSafeWrite(ctx) | 	attach := getIssueCommentAttachmentSafeWrite(ctx) | ||||||
| @@ -275,8 +277,13 @@ func EditIssueCommentAttachment(ctx *context.APIContext) { | |||||||
| 		attach.Name = form.Name | 		attach.Name = form.Name | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := repo_model.UpdateAttachment(ctx, attach); err != nil { | 	if err := attachment_service.UpdateAttachment(ctx, setting.Attachment.AllowedTypes, attach); err != nil { | ||||||
|  | 		if upload.IsErrFileTypeForbidden(err) { | ||||||
|  | 			ctx.Error(http.StatusUnprocessableEntity, "", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", attach) | 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", attach) | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
| 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attach)) | 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attach)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/web" | 	"code.gitea.io/gitea/modules/web" | ||||||
| 	"code.gitea.io/gitea/services/attachment" | 	attachment_service "code.gitea.io/gitea/services/attachment" | ||||||
| 	"code.gitea.io/gitea/services/context" | 	"code.gitea.io/gitea/services/context" | ||||||
| 	"code.gitea.io/gitea/services/context/upload" | 	"code.gitea.io/gitea/services/context/upload" | ||||||
| 	"code.gitea.io/gitea/services/convert" | 	"code.gitea.io/gitea/services/convert" | ||||||
| @@ -234,7 +234,7 @@ func CreateReleaseAttachment(ctx *context.APIContext) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Create a new attachment and save the file | 	// Create a new attachment and save the file | ||||||
| 	attach, err := attachment.UploadAttachment(ctx, content, setting.Repository.Release.AllowedTypes, size, &repo_model.Attachment{ | 	attach, err := attachment_service.UploadAttachment(ctx, content, setting.Repository.Release.AllowedTypes, size, &repo_model.Attachment{ | ||||||
| 		Name:       filename, | 		Name:       filename, | ||||||
| 		UploaderID: ctx.Doer.ID, | 		UploaderID: ctx.Doer.ID, | ||||||
| 		RepoID:     ctx.Repo.Repository.ID, | 		RepoID:     ctx.Repo.Repository.ID, | ||||||
| @@ -291,6 +291,8 @@ func EditReleaseAttachment(ctx *context.APIContext) { | |||||||
| 	// responses: | 	// responses: | ||||||
| 	//   "201": | 	//   "201": | ||||||
| 	//     "$ref": "#/responses/Attachment" | 	//     "$ref": "#/responses/Attachment" | ||||||
|  | 	//   "422": | ||||||
|  | 	//     "$ref": "#/responses/validationError" | ||||||
| 	//   "404": | 	//   "404": | ||||||
| 	//     "$ref": "#/responses/notFound" | 	//     "$ref": "#/responses/notFound" | ||||||
|  |  | ||||||
| @@ -322,8 +324,13 @@ func EditReleaseAttachment(ctx *context.APIContext) { | |||||||
| 		attach.Name = form.Name | 		attach.Name = form.Name | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if err := repo_model.UpdateAttachment(ctx, attach); err != nil { | 	if err := attachment_service.UpdateAttachment(ctx, setting.Repository.Release.AllowedTypes, attach); err != nil { | ||||||
|  | 		if upload.IsErrFileTypeForbidden(err) { | ||||||
|  | 			ctx.Error(http.StatusUnprocessableEntity, "", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", attach) | 		ctx.Error(http.StatusInternalServerError, "UpdateAttachment", attach) | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
| 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attach)) | 	ctx.JSON(http.StatusCreated, convert.ToAPIAttachment(ctx.Repo.Repository, attach)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -50,3 +50,12 @@ func UploadAttachment(ctx context.Context, file io.Reader, allowedTypes string, | |||||||
|  |  | ||||||
| 	return NewAttachment(ctx, attach, io.MultiReader(bytes.NewReader(buf), file), fileSize) | 	return NewAttachment(ctx, attach, io.MultiReader(bytes.NewReader(buf), file), fileSize) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // UpdateAttachment updates an attachment, verifying that its name is among the allowed types. | ||||||
|  | func UpdateAttachment(ctx context.Context, allowedTypes string, attach *repo_model.Attachment) error { | ||||||
|  | 	if err := upload.Verify(nil, attach.Name, allowedTypes); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return repo_model.UpdateAttachment(ctx, attach) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -28,12 +28,13 @@ func IsErrFileTypeForbidden(err error) bool { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (err ErrFileTypeForbidden) Error() string { | func (err ErrFileTypeForbidden) Error() string { | ||||||
| 	return "This file extension or type is not allowed to be uploaded." | 	return "This file cannot be uploaded or modified due to a forbidden file extension or type." | ||||||
| } | } | ||||||
|  |  | ||||||
| var wildcardTypeRe = regexp.MustCompile(`^[a-z]+/\*$`) | var wildcardTypeRe = regexp.MustCompile(`^[a-z]+/\*$`) | ||||||
|  |  | ||||||
| // Verify validates whether a file is allowed to be uploaded. | // Verify validates whether a file is allowed to be uploaded. If buf is empty, it will just check if the file | ||||||
|  | // has an allowed file extension. | ||||||
| func Verify(buf []byte, fileName, allowedTypesStr string) error { | func Verify(buf []byte, fileName, allowedTypesStr string) error { | ||||||
| 	allowedTypesStr = strings.ReplaceAll(allowedTypesStr, "|", ",") // compat for old config format | 	allowedTypesStr = strings.ReplaceAll(allowedTypesStr, "|", ",") // compat for old config format | ||||||
|  |  | ||||||
| @@ -56,21 +57,31 @@ func Verify(buf []byte, fileName, allowedTypesStr string) error { | |||||||
| 		return ErrFileTypeForbidden{Type: fullMimeType} | 		return ErrFileTypeForbidden{Type: fullMimeType} | ||||||
| 	} | 	} | ||||||
| 	extension := strings.ToLower(path.Ext(fileName)) | 	extension := strings.ToLower(path.Ext(fileName)) | ||||||
|  | 	isBufEmpty := len(buf) <= 1 | ||||||
|  |  | ||||||
| 	// https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#Unique_file_type_specifiers | 	// https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#Unique_file_type_specifiers | ||||||
| 	for _, allowEntry := range allowedTypes { | 	for _, allowEntry := range allowedTypes { | ||||||
| 		if allowEntry == "*/*" { | 		if allowEntry == "*/*" { | ||||||
| 			return nil // everything allowed | 			return nil // everything allowed | ||||||
| 		} else if strings.HasPrefix(allowEntry, ".") && allowEntry == extension { | 		} | ||||||
|  | 		if strings.HasPrefix(allowEntry, ".") && allowEntry == extension { | ||||||
| 			return nil // extension is allowed | 			return nil // extension is allowed | ||||||
| 		} else if mimeType == allowEntry { | 		} | ||||||
|  | 		if isBufEmpty { | ||||||
|  | 			continue // skip mime type checks if buffer is empty | ||||||
|  | 		} | ||||||
|  | 		if mimeType == allowEntry { | ||||||
| 			return nil // mime type is allowed | 			return nil // mime type is allowed | ||||||
| 		} else if wildcardTypeRe.MatchString(allowEntry) && strings.HasPrefix(mimeType, allowEntry[:len(allowEntry)-1]) { | 		} | ||||||
|  | 		if wildcardTypeRe.MatchString(allowEntry) && strings.HasPrefix(mimeType, allowEntry[:len(allowEntry)-1]) { | ||||||
| 			return nil // wildcard match, e.g. image/* | 			return nil // wildcard match, e.g. image/* | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	log.Info("Attachment with type %s blocked from upload", fullMimeType) | 	if !isBufEmpty { | ||||||
|  | 		log.Info("Attachment with type %s blocked from upload", fullMimeType) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return ErrFileTypeForbidden{Type: fullMimeType} | 	return ErrFileTypeForbidden{Type: fullMimeType} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								templates/swagger/v1_json.tmpl
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								templates/swagger/v1_json.tmpl
									
									
									
										generated
									
									
									
								
							| @@ -7706,6 +7706,9 @@ | |||||||
|           "404": { |           "404": { | ||||||
|             "$ref": "#/responses/error" |             "$ref": "#/responses/error" | ||||||
|           }, |           }, | ||||||
|  |           "422": { | ||||||
|  |             "$ref": "#/responses/validationError" | ||||||
|  |           }, | ||||||
|           "423": { |           "423": { | ||||||
|             "$ref": "#/responses/repoArchivedError" |             "$ref": "#/responses/repoArchivedError" | ||||||
|           } |           } | ||||||
| @@ -8328,6 +8331,9 @@ | |||||||
|           "404": { |           "404": { | ||||||
|             "$ref": "#/responses/error" |             "$ref": "#/responses/error" | ||||||
|           }, |           }, | ||||||
|  |           "422": { | ||||||
|  |             "$ref": "#/responses/validationError" | ||||||
|  |           }, | ||||||
|           "423": { |           "423": { | ||||||
|             "$ref": "#/responses/repoArchivedError" |             "$ref": "#/responses/repoArchivedError" | ||||||
|           } |           } | ||||||
| @@ -13474,6 +13480,9 @@ | |||||||
|           }, |           }, | ||||||
|           "404": { |           "404": { | ||||||
|             "$ref": "#/responses/notFound" |             "$ref": "#/responses/notFound" | ||||||
|  |           }, | ||||||
|  |           "422": { | ||||||
|  |             "$ref": "#/responses/validationError" | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -151,7 +151,7 @@ func TestAPICreateCommentAttachmentWithUnallowedFile(t *testing.T) { | |||||||
| func TestAPIEditCommentAttachment(t *testing.T) { | func TestAPIEditCommentAttachment(t *testing.T) { | ||||||
| 	defer tests.PrepareTestEnv(t)() | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
| 	const newAttachmentName = "newAttachmentName" | 	const newAttachmentName = "newAttachmentName.txt" | ||||||
|  |  | ||||||
| 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 6}) | 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 6}) | ||||||
| 	comment := unittest.AssertExistsAndLoadBean(t, &issues_model.Comment{ID: attachment.CommentID}) | 	comment := unittest.AssertExistsAndLoadBean(t, &issues_model.Comment{ID: attachment.CommentID}) | ||||||
| @@ -173,6 +173,27 @@ func TestAPIEditCommentAttachment(t *testing.T) { | |||||||
| 	unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: apiAttachment.ID, CommentID: comment.ID, Name: apiAttachment.Name}) | 	unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: apiAttachment.ID, CommentID: comment.ID, Name: apiAttachment.Name}) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestAPIEditCommentAttachmentWithUnallowedFile(t *testing.T) { | ||||||
|  | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
|  | 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 6}) | ||||||
|  | 	comment := unittest.AssertExistsAndLoadBean(t, &issues_model.Comment{ID: attachment.CommentID}) | ||||||
|  | 	issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: comment.IssueID}) | ||||||
|  | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: issue.RepoID}) | ||||||
|  | 	repoOwner := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: repo.OwnerID}) | ||||||
|  | 	session := loginUser(t, repoOwner.Name) | ||||||
|  | 	token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteIssue) | ||||||
|  |  | ||||||
|  | 	filename := "file.bad" | ||||||
|  | 	urlStr := fmt.Sprintf("/api/v1/repos/%s/%s/issues/comments/%d/assets/%d", | ||||||
|  | 		repoOwner.Name, repo.Name, comment.ID, attachment.ID) | ||||||
|  | 	req := NewRequestWithValues(t, "PATCH", urlStr, map[string]string{ | ||||||
|  | 		"name": filename, | ||||||
|  | 	}).AddTokenAuth(token) | ||||||
|  |  | ||||||
|  | 	session.MakeRequest(t, req, http.StatusUnprocessableEntity) | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestAPIDeleteCommentAttachment(t *testing.T) { | func TestAPIDeleteCommentAttachment(t *testing.T) { | ||||||
| 	defer tests.PrepareTestEnv(t)() | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -126,7 +126,7 @@ func TestAPICreateIssueAttachmentWithUnallowedFile(t *testing.T) { | |||||||
| func TestAPIEditIssueAttachment(t *testing.T) { | func TestAPIEditIssueAttachment(t *testing.T) { | ||||||
| 	defer tests.PrepareTestEnv(t)() | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
| 	const newAttachmentName = "newAttachmentName" | 	const newAttachmentName = "hello_world.txt" | ||||||
|  |  | ||||||
| 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 1}) | 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 1}) | ||||||
| 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: attachment.RepoID}) | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: attachment.RepoID}) | ||||||
| @@ -147,6 +147,26 @@ func TestAPIEditIssueAttachment(t *testing.T) { | |||||||
| 	unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: apiAttachment.ID, IssueID: issue.ID, Name: apiAttachment.Name}) | 	unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: apiAttachment.ID, IssueID: issue.ID, Name: apiAttachment.Name}) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestAPIEditIssueAttachmentWithUnallowedFile(t *testing.T) { | ||||||
|  | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
|  | 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 1}) | ||||||
|  | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: attachment.RepoID}) | ||||||
|  | 	issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: attachment.IssueID}) | ||||||
|  | 	repoOwner := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: repo.OwnerID}) | ||||||
|  |  | ||||||
|  | 	session := loginUser(t, repoOwner.Name) | ||||||
|  | 	token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteIssue) | ||||||
|  |  | ||||||
|  | 	filename := "file.bad" | ||||||
|  | 	urlStr := fmt.Sprintf("/api/v1/repos/%s/%s/issues/%d/assets/%d", repoOwner.Name, repo.Name, issue.Index, attachment.ID) | ||||||
|  | 	req := NewRequestWithValues(t, "PATCH", urlStr, map[string]string{ | ||||||
|  | 		"name": filename, | ||||||
|  | 	}).AddTokenAuth(token) | ||||||
|  |  | ||||||
|  | 	session.MakeRequest(t, req, http.StatusUnprocessableEntity) | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestAPIDeleteIssueAttachment(t *testing.T) { | func TestAPIDeleteIssueAttachment(t *testing.T) { | ||||||
| 	defer tests.PrepareTestEnv(t)() | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								tests/integration/api_releases_attachment_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								tests/integration/api_releases_attachment_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | // Copyright 2024 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  |  | ||||||
|  | package integration | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
|  | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
|  | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/modules/test" | ||||||
|  | 	"code.gitea.io/gitea/tests" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestAPIEditReleaseAttachmentWithUnallowedFile(t *testing.T) { | ||||||
|  | 	// Limit the allowed release types (since by default there is no restriction) | ||||||
|  | 	defer test.MockVariableValue(&setting.Repository.Release.AllowedTypes, ".exe")() | ||||||
|  | 	defer tests.PrepareTestEnv(t)() | ||||||
|  |  | ||||||
|  | 	attachment := unittest.AssertExistsAndLoadBean(t, &repo_model.Attachment{ID: 9}) | ||||||
|  | 	release := unittest.AssertExistsAndLoadBean(t, &repo_model.Release{ID: attachment.ReleaseID}) | ||||||
|  | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: attachment.RepoID}) | ||||||
|  | 	repoOwner := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: repo.OwnerID}) | ||||||
|  |  | ||||||
|  | 	session := loginUser(t, repoOwner.Name) | ||||||
|  | 	token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  |  | ||||||
|  | 	filename := "file.bad" | ||||||
|  | 	urlStr := fmt.Sprintf("/api/v1/repos/%s/%s/releases/%d/assets/%d", repoOwner.Name, repo.Name, release.ID, attachment.ID) | ||||||
|  | 	req := NewRequestWithValues(t, "PATCH", urlStr, map[string]string{ | ||||||
|  | 		"name": filename, | ||||||
|  | 	}).AddTokenAuth(token) | ||||||
|  |  | ||||||
|  | 	session.MakeRequest(t, req, http.StatusUnprocessableEntity) | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user