| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | // Copyright 2023 The Gitea Authors. All rights reserved. | 
					
						
							|  |  |  | // SPDX-License-Identifier: MIT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package rpm | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-27 19:40:14 +00:00
										 |  |  | 	"forgejo.org/modules/timeutil" | 
					
						
							|  |  |  | 	"forgejo.org/modules/validation" | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-25 12:30:04 +02:00
										 |  |  | 	"code.forgejo.org/forgejo/go-rpmutils" | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2024-01-19 12:37:10 +01:00
										 |  |  | 	PropertyMetadata     = "rpm.metadata" | 
					
						
							|  |  |  | 	PropertyGroup        = "rpm.group" | 
					
						
							|  |  |  | 	PropertyArchitecture = "rpm.architecture" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 	SettingKeyPrivate = "rpm.key.private" | 
					
						
							|  |  |  | 	SettingKeyPublic  = "rpm.key.public" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	RepositoryPackage = "_rpm" | 
					
						
							|  |  |  | 	RepositoryVersion = "_repository" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	// Can't use the syscall constants because they are not available for windows build. | 
					
						
							|  |  |  | 	sIFMT  = 0xf000 | 
					
						
							|  |  |  | 	sIFDIR = 0x4000 | 
					
						
							|  |  |  | 	sIXUSR = 0x40 | 
					
						
							|  |  |  | 	sIXGRP = 0x8 | 
					
						
							|  |  |  | 	sIXOTH = 0x1 | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://rpm-software-management.github.io/rpm/manual/spec.html | 
					
						
							|  |  |  | // https://refspecs.linuxbase.org/LSB_3.1.0/LSB-Core-generic/LSB-Core-generic/pkgformat.html | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Package struct { | 
					
						
							|  |  |  | 	Name            string | 
					
						
							|  |  |  | 	Version         string | 
					
						
							|  |  |  | 	VersionMetadata *VersionMetadata | 
					
						
							|  |  |  | 	FileMetadata    *FileMetadata | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type VersionMetadata struct { | 
					
						
							|  |  |  | 	License     string `json:"license,omitempty"` | 
					
						
							|  |  |  | 	ProjectURL  string `json:"project_url,omitempty"` | 
					
						
							|  |  |  | 	Summary     string `json:"summary,omitempty"` | 
					
						
							|  |  |  | 	Description string `json:"description,omitempty"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type FileMetadata struct { | 
					
						
							|  |  |  | 	Architecture  string `json:"architecture,omitempty"` | 
					
						
							|  |  |  | 	Epoch         string `json:"epoch,omitempty"` | 
					
						
							|  |  |  | 	Version       string `json:"version,omitempty"` | 
					
						
							|  |  |  | 	Release       string `json:"release,omitempty"` | 
					
						
							|  |  |  | 	Vendor        string `json:"vendor,omitempty"` | 
					
						
							|  |  |  | 	Group         string `json:"group,omitempty"` | 
					
						
							|  |  |  | 	Packager      string `json:"packager,omitempty"` | 
					
						
							|  |  |  | 	SourceRpm     string `json:"source_rpm,omitempty"` | 
					
						
							|  |  |  | 	BuildHost     string `json:"build_host,omitempty"` | 
					
						
							|  |  |  | 	BuildTime     uint64 `json:"build_time,omitempty"` | 
					
						
							|  |  |  | 	FileTime      uint64 `json:"file_time,omitempty"` | 
					
						
							|  |  |  | 	InstalledSize uint64 `json:"installed_size,omitempty"` | 
					
						
							|  |  |  | 	ArchiveSize   uint64 `json:"archive_size,omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Provides  []*Entry `json:"provide,omitempty"` | 
					
						
							|  |  |  | 	Requires  []*Entry `json:"require,omitempty"` | 
					
						
							|  |  |  | 	Conflicts []*Entry `json:"conflict,omitempty"` | 
					
						
							|  |  |  | 	Obsoletes []*Entry `json:"obsolete,omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Files []*File `json:"files,omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Changelogs []*Changelog `json:"changelogs,omitempty"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Entry struct { | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 	Name     string `json:"name" xml:"name,attr"` | 
					
						
							|  |  |  | 	Flags    string `json:"flags,omitempty" xml:"flags,attr,omitempty"` | 
					
						
							|  |  |  | 	AltFlags uint32 `json:"alt_flags,omitempty" xml:"alt_flags,attr,omitempty"` | 
					
						
							|  |  |  | 	Version  string `json:"version,omitempty" xml:"ver,attr,omitempty"` | 
					
						
							|  |  |  | 	Epoch    string `json:"epoch,omitempty" xml:"epoch,attr,omitempty"` | 
					
						
							|  |  |  | 	Release  string `json:"release,omitempty" xml:"rel,attr,omitempty"` | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type File struct { | 
					
						
							|  |  |  | 	Path         string `json:"path" xml:",chardata"` | 
					
						
							|  |  |  | 	Type         string `json:"type,omitempty" xml:"type,attr,omitempty"` | 
					
						
							|  |  |  | 	IsExecutable bool   `json:"is_executable" xml:"-"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Changelog struct { | 
					
						
							|  |  |  | 	Author string             `json:"author,omitempty" xml:"author,attr"` | 
					
						
							|  |  |  | 	Date   timeutil.TimeStamp `json:"date,omitempty" xml:"date,attr"` | 
					
						
							|  |  |  | 	Text   string             `json:"text,omitempty" xml:",chardata"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ParsePackage parses the RPM package file | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | func ParsePackage(r io.Reader, repoType string) (*Package, error) { | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 	rpm, err := rpmutils.ReadRpm(r) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nevra, err := rpm.Header.GetNEVRA() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	version := fmt.Sprintf("%s-%s", nevra.Version, nevra.Release) | 
					
						
							|  |  |  | 	if nevra.Epoch != "" && nevra.Epoch != "0" { | 
					
						
							|  |  |  | 		version = fmt.Sprintf("%s-%s", nevra.Epoch, version) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p := &Package{ | 
					
						
							|  |  |  | 		Name:    nevra.Name, | 
					
						
							|  |  |  | 		Version: version, | 
					
						
							|  |  |  | 		VersionMetadata: &VersionMetadata{ | 
					
						
							|  |  |  | 			Summary:     getString(rpm.Header, rpmutils.SUMMARY), | 
					
						
							|  |  |  | 			Description: getString(rpm.Header, rpmutils.DESCRIPTION), | 
					
						
							|  |  |  | 			License:     getString(rpm.Header, rpmutils.LICENSE), | 
					
						
							|  |  |  | 			ProjectURL:  getString(rpm.Header, rpmutils.URL), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		FileMetadata: &FileMetadata{ | 
					
						
							|  |  |  | 			Architecture:  nevra.Arch, | 
					
						
							|  |  |  | 			Epoch:         nevra.Epoch, | 
					
						
							|  |  |  | 			Version:       nevra.Version, | 
					
						
							|  |  |  | 			Release:       nevra.Release, | 
					
						
							|  |  |  | 			Vendor:        getString(rpm.Header, rpmutils.VENDOR), | 
					
						
							|  |  |  | 			Group:         getString(rpm.Header, rpmutils.GROUP), | 
					
						
							|  |  |  | 			Packager:      getString(rpm.Header, rpmutils.PACKAGER), | 
					
						
							|  |  |  | 			SourceRpm:     getString(rpm.Header, rpmutils.SOURCERPM), | 
					
						
							|  |  |  | 			BuildHost:     getString(rpm.Header, rpmutils.BUILDHOST), | 
					
						
							|  |  |  | 			BuildTime:     getUInt64(rpm.Header, rpmutils.BUILDTIME), | 
					
						
							|  |  |  | 			FileTime:      getUInt64(rpm.Header, rpmutils.FILEMTIMES), | 
					
						
							|  |  |  | 			InstalledSize: getUInt64(rpm.Header, rpmutils.SIZE), | 
					
						
							|  |  |  | 			ArchiveSize:   getUInt64(rpm.Header, rpmutils.SIG_PAYLOADSIZE), | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 			Provides:   getEntries(rpm.Header, rpmutils.PROVIDENAME, rpmutils.PROVIDEVERSION, rpmutils.PROVIDEFLAGS, repoType), | 
					
						
							|  |  |  | 			Requires:   getEntries(rpm.Header, rpmutils.REQUIRENAME, rpmutils.REQUIREVERSION, rpmutils.REQUIREFLAGS, repoType), | 
					
						
							|  |  |  | 			Conflicts:  getEntries(rpm.Header, rpmutils.CONFLICTNAME, rpmutils.CONFLICTVERSION, rpmutils.CONFLICTFLAGS, repoType), | 
					
						
							|  |  |  | 			Obsoletes:  getEntries(rpm.Header, rpmutils.OBSOLETENAME, rpmutils.OBSOLETEVERSION, rpmutils.OBSOLETEFLAGS, repoType), | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 			Files:      getFiles(rpm.Header), | 
					
						
							|  |  |  | 			Changelogs: getChangelogs(rpm.Header), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !validation.IsValidURL(p.VersionMetadata.ProjectURL) { | 
					
						
							|  |  |  | 		p.VersionMetadata.ProjectURL = "" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return p, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func getString(h *rpmutils.RpmHeader, tag int) string { | 
					
						
							|  |  |  | 	values, err := h.GetStrings(tag) | 
					
						
							|  |  |  | 	if err != nil || len(values) < 1 { | 
					
						
							|  |  |  | 		return "" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return values[0] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func getUInt64(h *rpmutils.RpmHeader, tag int) uint64 { | 
					
						
							|  |  |  | 	values, err := h.GetUint64s(tag) | 
					
						
							|  |  |  | 	if err != nil || len(values) < 1 { | 
					
						
							|  |  |  | 		return 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return values[0] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | func getEntries(h *rpmutils.RpmHeader, namesTag, versionsTag, flagsTag int, repoType string) []*Entry { | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 	names, err := h.GetStrings(namesTag) | 
					
						
							|  |  |  | 	if err != nil || len(names) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	flags, err := h.GetUint64s(flagsTag) | 
					
						
							|  |  |  | 	if err != nil || len(flags) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	versions, err := h.GetStrings(versionsTag) | 
					
						
							|  |  |  | 	if err != nil || len(versions) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(names) != len(flags) || len(names) != len(versions) { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	entries := make([]*Entry, 0, len(names)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 	switch repoType { | 
					
						
							|  |  |  | 	case "rpm": | 
					
						
							|  |  |  | 		for i := range names { | 
					
						
							|  |  |  | 			e := &Entry{ | 
					
						
							|  |  |  | 				Name: names[i], | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 			flags := flags[i] | 
					
						
							|  |  |  | 			if (flags&rpmutils.RPMSENSE_GREATER) != 0 && (flags&rpmutils.RPMSENSE_EQUAL) != 0 { | 
					
						
							|  |  |  | 				e.Flags = "GE" | 
					
						
							|  |  |  | 			} else if (flags&rpmutils.RPMSENSE_LESS) != 0 && (flags&rpmutils.RPMSENSE_EQUAL) != 0 { | 
					
						
							|  |  |  | 				e.Flags = "LE" | 
					
						
							|  |  |  | 			} else if (flags & rpmutils.RPMSENSE_GREATER) != 0 { | 
					
						
							|  |  |  | 				e.Flags = "GT" | 
					
						
							|  |  |  | 			} else if (flags & rpmutils.RPMSENSE_LESS) != 0 { | 
					
						
							|  |  |  | 				e.Flags = "LT" | 
					
						
							|  |  |  | 			} else if (flags & rpmutils.RPMSENSE_EQUAL) != 0 { | 
					
						
							|  |  |  | 				e.Flags = "EQ" | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 			version := versions[i] | 
					
						
							|  |  |  | 			if version != "" { | 
					
						
							|  |  |  | 				parts := strings.Split(version, "-") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				versionParts := strings.Split(parts[0], ":") | 
					
						
							|  |  |  | 				if len(versionParts) == 2 { | 
					
						
							|  |  |  | 					e.Version = versionParts[1] | 
					
						
							|  |  |  | 					e.Epoch = versionParts[0] | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					e.Version = versionParts[0] | 
					
						
							|  |  |  | 					e.Epoch = "0" | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if len(parts) > 1 { | 
					
						
							|  |  |  | 					e.Release = parts[1] | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 			entries = append(entries, e) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	case "alt": | 
					
						
							|  |  |  | 		for i := range names { | 
					
						
							|  |  |  | 			e := &Entry{ | 
					
						
							| 
									
										
											  
											
												several fixes of ALT Package registry (#8475)
closes #7946
- The `rpmsRepoPattern` regex has been fixed to handle releases with dots correctly. For example, the version `0.9.0-alt1.git.17.g2ba905d` is valid, just like `0.1.0-1.n1` mentioned in the issue (https://codeberg.org/forgejo/forgejo/issues/7946#issue-1628991)
- getEntries now returns entry names. In the integration tests, there were lines like:
```go
assert.Equal(t, []string{"", ""}, result.ProvideNames)
```
and it’s unclear how such test logic could have ever worked correctly (fixes problems with deps https://codeberg.org/forgejo/forgejo/issues/7946#issuecomment-5109795)
- ALT is an acronym for ALT Linux Team, so `Alt` was replaced with `ALT`. Strictly speaking, it should probably be `ALT Linux`, but since we use `Arch` instead of `Arch Linux`, this seems fine. Also, Distrowatch shows `Arch`/`ALT` in its dropdown, so it’s consistent.
- The strings `"Alt Linux Team"` and `"Sisyphus"` in the `Origin` and `Suite` fields have been replaced with `setting.AppName` and `"Unknown"`. `Unknown` is a valid value and is set by default, so this won’t cause any issues.
- The documentation link has been fixed: (404 docs.gitea.com/usage/packages/alt/ -> 200 forgejo.org/docs/latest/user/packages/alt/)
---
## Checklist
The [contributor guide](https://forgejo.org/docs/next/contributor/) contains information that will be helpful to first time contributors. There also are a few [conditions for merging Pull Requests in Forgejo repositories](https://codeberg.org/forgejo/governance/src/branch/main/PullRequestsAgreement.md). You are also welcome to join the [Forgejo development chatroom](https://matrix.to/#/#forgejo-development:matrix.org).
### Tests
- I added test coverage for Go changes...
  - [ ] in their respective `*_test.go` for unit tests.
  - [x] in the `tests/integration` directory if it involves interactions with a live Forgejo server.
- I added test coverage for JavaScript changes...
  - [ ] in `web_src/js/*.test.js` if it can be unit tested.
  - [ ] in `tests/e2e/*.test.e2e.js` if it requires interactions with a live Forgejo server (see also the [developer guide for JavaScript testing](https://codeberg.org/forgejo/forgejo/src/branch/forgejo/tests/e2e/README.md#end-to-end-tests)).
### Documentation
- [ ] I created a pull request [to the documentation](https://codeberg.org/forgejo/docs) to explain to Forgejo users how to use this change.
- [x] I did not document these changes and I do not expect someone else to do it.
### Release notes
- [ ] I do not want this change to show in the release notes.
- [x] I want the title to show in the release notes with a link to this pull request.
- [ ] I want the content of the `release-notes/<pull request number>.md` to be be used for the release notes instead of the title.
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/8475
Reviewed-by: Gusted <gusted@noreply.codeberg.org>
Co-authored-by: Maxim Slipenko <maks1ms@altlinux.org>
Co-committed-by: Maxim Slipenko <maks1ms@altlinux.org>
											
										 
											2025-07-10 17:12:07 +02:00
										 |  |  | 				Name:     names[i], | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 				AltFlags: uint32(flags[i]), | 
					
						
							| 
									
										
											  
											
												several fixes of ALT Package registry (#8475)
closes #7946
- The `rpmsRepoPattern` regex has been fixed to handle releases with dots correctly. For example, the version `0.9.0-alt1.git.17.g2ba905d` is valid, just like `0.1.0-1.n1` mentioned in the issue (https://codeberg.org/forgejo/forgejo/issues/7946#issue-1628991)
- getEntries now returns entry names. In the integration tests, there were lines like:
```go
assert.Equal(t, []string{"", ""}, result.ProvideNames)
```
and it’s unclear how such test logic could have ever worked correctly (fixes problems with deps https://codeberg.org/forgejo/forgejo/issues/7946#issuecomment-5109795)
- ALT is an acronym for ALT Linux Team, so `Alt` was replaced with `ALT`. Strictly speaking, it should probably be `ALT Linux`, but since we use `Arch` instead of `Arch Linux`, this seems fine. Also, Distrowatch shows `Arch`/`ALT` in its dropdown, so it’s consistent.
- The strings `"Alt Linux Team"` and `"Sisyphus"` in the `Origin` and `Suite` fields have been replaced with `setting.AppName` and `"Unknown"`. `Unknown` is a valid value and is set by default, so this won’t cause any issues.
- The documentation link has been fixed: (404 docs.gitea.com/usage/packages/alt/ -> 200 forgejo.org/docs/latest/user/packages/alt/)
---
## Checklist
The [contributor guide](https://forgejo.org/docs/next/contributor/) contains information that will be helpful to first time contributors. There also are a few [conditions for merging Pull Requests in Forgejo repositories](https://codeberg.org/forgejo/governance/src/branch/main/PullRequestsAgreement.md). You are also welcome to join the [Forgejo development chatroom](https://matrix.to/#/#forgejo-development:matrix.org).
### Tests
- I added test coverage for Go changes...
  - [ ] in their respective `*_test.go` for unit tests.
  - [x] in the `tests/integration` directory if it involves interactions with a live Forgejo server.
- I added test coverage for JavaScript changes...
  - [ ] in `web_src/js/*.test.js` if it can be unit tested.
  - [ ] in `tests/e2e/*.test.e2e.js` if it requires interactions with a live Forgejo server (see also the [developer guide for JavaScript testing](https://codeberg.org/forgejo/forgejo/src/branch/forgejo/tests/e2e/README.md#end-to-end-tests)).
### Documentation
- [ ] I created a pull request [to the documentation](https://codeberg.org/forgejo/docs) to explain to Forgejo users how to use this change.
- [x] I did not document these changes and I do not expect someone else to do it.
### Release notes
- [ ] I do not want this change to show in the release notes.
- [x] I want the title to show in the release notes with a link to this pull request.
- [ ] I want the content of the `release-notes/<pull request number>.md` to be be used for the release notes instead of the title.
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/8475
Reviewed-by: Gusted <gusted@noreply.codeberg.org>
Co-authored-by: Maxim Slipenko <maks1ms@altlinux.org>
Co-committed-by: Maxim Slipenko <maks1ms@altlinux.org>
											
										 
											2025-07-10 17:12:07 +02:00
										 |  |  | 				Version:  versions[i], | 
					
						
							| 
									
										
										
										
											2025-01-22 14:01:49 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			entries = append(entries, e) | 
					
						
							| 
									
										
										
										
											2023-05-05 22:33:37 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return entries | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func getFiles(h *rpmutils.RpmHeader) []*File { | 
					
						
							|  |  |  | 	baseNames, _ := h.GetStrings(rpmutils.BASENAMES) | 
					
						
							|  |  |  | 	dirNames, _ := h.GetStrings(rpmutils.DIRNAMES) | 
					
						
							|  |  |  | 	dirIndexes, _ := h.GetUint32s(rpmutils.DIRINDEXES) | 
					
						
							|  |  |  | 	fileFlags, _ := h.GetUint32s(rpmutils.FILEFLAGS) | 
					
						
							|  |  |  | 	fileModes, _ := h.GetUint32s(rpmutils.FILEMODES) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	files := make([]*File, 0, len(baseNames)) | 
					
						
							|  |  |  | 	for i := range baseNames { | 
					
						
							|  |  |  | 		if len(dirIndexes) <= i { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		dirIndex := dirIndexes[i] | 
					
						
							|  |  |  | 		if len(dirNames) <= int(dirIndex) { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var fileType string | 
					
						
							|  |  |  | 		var isExecutable bool | 
					
						
							|  |  |  | 		if i < len(fileFlags) && (fileFlags[i]&rpmutils.RPMFILE_GHOST) != 0 { | 
					
						
							|  |  |  | 			fileType = "ghost" | 
					
						
							|  |  |  | 		} else if i < len(fileModes) { | 
					
						
							|  |  |  | 			if (fileModes[i] & sIFMT) == sIFDIR { | 
					
						
							|  |  |  | 				fileType = "dir" | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				mode := fileModes[i] & ^uint32(sIFMT) | 
					
						
							|  |  |  | 				isExecutable = (mode&sIXUSR) != 0 || (mode&sIXGRP) != 0 || (mode&sIXOTH) != 0 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		files = append(files, &File{ | 
					
						
							|  |  |  | 			Path:         dirNames[dirIndex] + baseNames[i], | 
					
						
							|  |  |  | 			Type:         fileType, | 
					
						
							|  |  |  | 			IsExecutable: isExecutable, | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return files | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func getChangelogs(h *rpmutils.RpmHeader) []*Changelog { | 
					
						
							|  |  |  | 	texts, err := h.GetStrings(rpmutils.CHANGELOGTEXT) | 
					
						
							|  |  |  | 	if err != nil || len(texts) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	authors, err := h.GetStrings(rpmutils.CHANGELOGNAME) | 
					
						
							|  |  |  | 	if err != nil || len(authors) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	times, err := h.GetUint32s(rpmutils.CHANGELOGTIME) | 
					
						
							|  |  |  | 	if err != nil || len(times) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(texts) != len(authors) || len(texts) != len(times) { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	changelogs := make([]*Changelog, 0, len(texts)) | 
					
						
							|  |  |  | 	for i := range texts { | 
					
						
							|  |  |  | 		changelogs = append(changelogs, &Changelog{ | 
					
						
							|  |  |  | 			Author: authors[i], | 
					
						
							|  |  |  | 			Date:   timeutil.TimeStamp(times[i]), | 
					
						
							|  |  |  | 			Text:   texts[i], | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return changelogs | 
					
						
							|  |  |  | } |