mirror of
				https://github.com/go-gitea/gitea
				synced 2025-09-28 03:28:13 +00:00 
			
		
		
		
	To make it more flexible and support SSH signature. The existing tests are not changed, there are also tests covering `parseTagRef` which also calls `parsePayloadSignature` now. Add some new tests to `Test_parseTagData`
		
			
				
	
	
		
			121 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 The Gogs Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package git
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"sort"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| )
 | |
| 
 | |
| // Tag represents a Git tag.
 | |
| type Tag struct {
 | |
| 	Name      string
 | |
| 	ID        ObjectID
 | |
| 	Object    ObjectID // The id of this commit object
 | |
| 	Type      string
 | |
| 	Tagger    *Signature
 | |
| 	Message   string
 | |
| 	Signature *CommitSignature
 | |
| }
 | |
| 
 | |
| // Commit return the commit of the tag reference
 | |
| func (tag *Tag) Commit(gitRepo *Repository) (*Commit, error) {
 | |
| 	return gitRepo.getCommit(tag.Object)
 | |
| }
 | |
| 
 | |
| func parsePayloadSignature(data []byte, messageStart int) (payload, msg, sign string) {
 | |
| 	pos := messageStart
 | |
| 	signStart, signEnd := -1, -1
 | |
| 	for {
 | |
| 		eol := bytes.IndexByte(data[pos:], '\n')
 | |
| 		if eol < 0 {
 | |
| 			break
 | |
| 		}
 | |
| 		line := data[pos : pos+eol]
 | |
| 		signType, hasPrefix := bytes.CutPrefix(line, []byte("-----BEGIN "))
 | |
| 		signType, hasSuffix := bytes.CutSuffix(signType, []byte(" SIGNATURE-----"))
 | |
| 		if hasPrefix && hasSuffix {
 | |
| 			signEndBytes := append([]byte("\n-----END "), signType...)
 | |
| 			signEndBytes = append(signEndBytes, []byte(" SIGNATURE-----")...)
 | |
| 			signEnd = bytes.Index(data[pos:], signEndBytes)
 | |
| 			if signEnd != -1 {
 | |
| 				signStart = pos
 | |
| 				signEnd = pos + signEnd + len(signEndBytes)
 | |
| 			}
 | |
| 		}
 | |
| 		pos += eol + 1
 | |
| 	}
 | |
| 
 | |
| 	if signStart != -1 && signEnd != -1 {
 | |
| 		msgEnd := max(messageStart, signStart-1)
 | |
| 		return string(data[:msgEnd]), string(data[messageStart:msgEnd]), string(data[signStart:signEnd])
 | |
| 	}
 | |
| 	return string(data), string(data[messageStart:]), ""
 | |
| }
 | |
| 
 | |
| // Parse commit information from the (uncompressed) raw
 | |
| // data from the commit object.
 | |
| // \n\n separate headers from message
 | |
| func parseTagData(objectFormat ObjectFormat, data []byte) (*Tag, error) {
 | |
| 	tag := new(Tag)
 | |
| 	tag.ID = objectFormat.EmptyObjectID()
 | |
| 	tag.Object = objectFormat.EmptyObjectID()
 | |
| 	tag.Tagger = &Signature{}
 | |
| 
 | |
| 	pos := 0
 | |
| 	for {
 | |
| 		eol := bytes.IndexByte(data[pos:], '\n')
 | |
| 		if eol == -1 {
 | |
| 			break // shouldn't happen, but could just tolerate it
 | |
| 		}
 | |
| 		if eol == 0 {
 | |
| 			pos++
 | |
| 			break // end of headers
 | |
| 		}
 | |
| 		line := data[pos : pos+eol]
 | |
| 		key, val, _ := bytes.Cut(line, []byte(" "))
 | |
| 		switch string(key) {
 | |
| 		case "object":
 | |
| 			id, err := NewIDFromString(string(val))
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			tag.Object = id
 | |
| 		case "type":
 | |
| 			tag.Type = string(val) // A commit can have one or more parents
 | |
| 		case "tagger":
 | |
| 			tag.Tagger = parseSignatureFromCommitLine(util.UnsafeBytesToString(val))
 | |
| 		}
 | |
| 		pos += eol + 1
 | |
| 	}
 | |
| 	payload, msg, sign := parsePayloadSignature(data, pos)
 | |
| 	tag.Message = msg
 | |
| 	if len(sign) > 0 {
 | |
| 		tag.Signature = &CommitSignature{Signature: sign, Payload: payload}
 | |
| 	}
 | |
| 	return tag, nil
 | |
| }
 | |
| 
 | |
| type tagSorter []*Tag
 | |
| 
 | |
| func (ts tagSorter) Len() int {
 | |
| 	return len([]*Tag(ts))
 | |
| }
 | |
| 
 | |
| func (ts tagSorter) Less(i, j int) bool {
 | |
| 	return []*Tag(ts)[i].Tagger.When.After([]*Tag(ts)[j].Tagger.When)
 | |
| }
 | |
| 
 | |
| func (ts tagSorter) Swap(i, j int) {
 | |
| 	[]*Tag(ts)[i], []*Tag(ts)[j] = []*Tag(ts)[j], []*Tag(ts)[i]
 | |
| }
 | |
| 
 | |
| // sortTagsByTime
 | |
| func sortTagsByTime(tags []*Tag) {
 | |
| 	sorter := tagSorter(tags)
 | |
| 	sort.Sort(sorter)
 | |
| }
 |