2022-05-11 22:56:25 -03:00
|
|
|
// Package archivefiles can evaluate a list of config.Files into their final form.
|
|
|
|
package archivefiles
|
|
|
|
|
|
|
|
import (
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
"errors"
|
2022-05-11 22:56:25 -03:00
|
|
|
"fmt"
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
"io/fs"
|
|
|
|
"os"
|
2022-05-11 22:56:25 -03:00
|
|
|
"path/filepath"
|
|
|
|
"sort"
|
2022-12-14 12:16:43 -03:00
|
|
|
"time"
|
2022-05-11 22:56:25 -03:00
|
|
|
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2022-05-11 22:56:25 -03:00
|
|
|
"github.com/goreleaser/fileglob"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
|
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Eval evaluates the given list of files to their final form.
|
2023-06-06 13:54:43 -03:00
|
|
|
func Eval(template *tmpl.Template, files []config.File) ([]config.File, error) {
|
2022-05-11 22:56:25 -03:00
|
|
|
var result []config.File
|
|
|
|
for _, f := range files {
|
2023-05-06 23:08:36 -03:00
|
|
|
glob, err := template.Apply(f.Source)
|
2022-05-11 22:56:25 -03:00
|
|
|
if err != nil {
|
|
|
|
return result, fmt.Errorf("failed to apply template %s: %w", f.Source, err)
|
|
|
|
}
|
|
|
|
|
2023-05-06 23:08:36 -03:00
|
|
|
files, err := fileglob.Glob(glob)
|
2022-05-11 22:56:25 -03:00
|
|
|
if err != nil {
|
2023-05-06 23:08:36 -03:00
|
|
|
return result, fmt.Errorf("globbing failed for pattern %s: %w", glob, err)
|
2022-05-11 22:56:25 -03:00
|
|
|
}
|
|
|
|
|
2023-05-06 23:08:36 -03:00
|
|
|
if len(files) == 0 {
|
2023-05-16 09:22:22 -03:00
|
|
|
if !f.Default {
|
|
|
|
// only log if its not a default glob, as those are usually
|
|
|
|
// very generic and are not really warnings for the user.
|
|
|
|
log.WithField("glob", f.Source).Warn("no files matched")
|
|
|
|
}
|
2023-05-06 23:08:36 -03:00
|
|
|
continue
|
2022-12-14 12:16:43 -03:00
|
|
|
}
|
2023-05-06 23:08:36 -03:00
|
|
|
|
|
|
|
if err := tmplInfo(template, &f.Info); err != nil {
|
|
|
|
return result, err
|
2022-12-14 12:16:43 -03:00
|
|
|
}
|
|
|
|
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
// the prefix may not be a complete path or may use glob patterns, in that case use the parent directory
|
2023-05-06 23:08:36 -03:00
|
|
|
prefix := glob
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
if _, err := os.Stat(prefix); errors.Is(err, fs.ErrNotExist) || fileglob.ContainsMatchers(prefix) {
|
|
|
|
prefix = filepath.Dir(longestCommonPrefix(files))
|
|
|
|
}
|
|
|
|
|
2022-05-11 22:56:25 -03:00
|
|
|
for _, file := range files {
|
2023-06-06 13:54:43 -03:00
|
|
|
dst, err := destinationFor(f, prefix, file)
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-05-11 22:56:25 -03:00
|
|
|
result = append(result, config.File{
|
2023-06-15 23:43:29 -03:00
|
|
|
Source: filepath.ToSlash(file),
|
|
|
|
Destination: filepath.ToSlash(dst),
|
2022-05-11 22:56:25 -03:00
|
|
|
Info: f.Info,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(result, func(i, j int) bool {
|
|
|
|
return result[i].Destination < result[j].Destination
|
|
|
|
})
|
|
|
|
|
|
|
|
return unique(result), nil
|
|
|
|
}
|
|
|
|
|
2023-05-06 23:08:36 -03:00
|
|
|
func tmplInfo(template *tmpl.Template, info *config.FileInfo) error {
|
2023-06-30 19:46:53 +02:00
|
|
|
if err := template.ApplyAll(
|
|
|
|
&info.Owner,
|
|
|
|
&info.Group,
|
|
|
|
&info.MTime,
|
|
|
|
); err != nil {
|
|
|
|
return err
|
2023-05-06 23:08:36 -03:00
|
|
|
}
|
|
|
|
if info.MTime != "" {
|
2023-06-30 19:46:53 +02:00
|
|
|
var err error
|
2023-05-06 23:08:36 -03:00
|
|
|
info.ParsedMTime, err = time.Parse(time.RFC3339Nano, info.MTime)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to parse %s: %w", info.MTime, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-05-11 22:56:25 -03:00
|
|
|
// remove duplicates
|
|
|
|
func unique(in []config.File) []config.File {
|
|
|
|
var result []config.File
|
|
|
|
exist := map[string]string{}
|
|
|
|
for _, f := range in {
|
|
|
|
if current := exist[f.Destination]; current != "" {
|
|
|
|
log.Warnf(
|
|
|
|
"file '%s' already exists in archive as '%s' - '%s' will be ignored",
|
|
|
|
f.Destination,
|
|
|
|
current,
|
|
|
|
f.Source,
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
exist[f.Destination] = f.Source
|
|
|
|
result = append(result, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2023-06-06 13:54:43 -03:00
|
|
|
func destinationFor(f config.File, prefix, path string) (string, error) {
|
2022-05-11 22:56:25 -03:00
|
|
|
if f.StripParent {
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
return filepath.Join(f.Destination, filepath.Base(path)), nil
|
|
|
|
}
|
|
|
|
|
2023-06-06 13:54:43 -03:00
|
|
|
if f.Destination != "" {
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
relpath, err := filepath.Rel(prefix, path)
|
|
|
|
if err != nil {
|
|
|
|
// since prefix is a prefix of src a relative path should always be found
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return filepath.ToSlash(filepath.Join(f.Destination, relpath)), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return filepath.Join(f.Destination, path), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// longestCommonPrefix returns the longest prefix of all strings the argument
|
|
|
|
// slice. If the slice is empty the empty string is returned.
|
|
|
|
// copied from nfpm
|
|
|
|
func longestCommonPrefix(strs []string) string {
|
|
|
|
if len(strs) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
lcp := strs[0]
|
|
|
|
for _, str := range strs {
|
|
|
|
lcp = strlcp(lcp, str)
|
|
|
|
}
|
|
|
|
return lcp
|
|
|
|
}
|
|
|
|
|
|
|
|
// copied from nfpm
|
|
|
|
func strlcp(a, b string) string {
|
|
|
|
var min int
|
|
|
|
if len(a) > len(b) {
|
|
|
|
min = len(b)
|
|
|
|
} else {
|
|
|
|
min = len(a)
|
|
|
|
}
|
|
|
|
for i := 0; i < min; i++ {
|
|
|
|
if a[i] != b[i] {
|
|
|
|
return a[0:i]
|
|
|
|
}
|
2022-05-11 22:56:25 -03:00
|
|
|
}
|
feat: better archives relative paths (#3656)
with this patch, a config like:
```yaml
archives:
- format: tar.gz
# this name template makes the OS and Arch compatible with the results of uname.
name_template: >-
{{ .ProjectName }}_
{{- title .Os }}_
{{- if eq .Arch "amd64" }}x86_64
{{- else if eq .Arch "386" }}i386
{{- else }}{{ .Arch }}{{ end }}
{{- if .Arm }}v{{ .Arm }}{{ end }}
rlcp: true
files:
- src: "build/**/*"
dst: .
nfpms:
- package_name: foo
contents:
- src: "build/**/*"
dst: usr/share/foo
formats:
- apk
```
will eval this:
<img width="1384" alt="CleanShot 2022-12-21 at 22 21 00@2x"
src="https://user-images.githubusercontent.com/245435/209034244-7c31b5f7-cfcd-4825-bb2f-7dd463c5286a.png">
as much as I would like to make this the default, it would be a breaking
change, so we really can't do it.
If `dst` is empty, it'll have the same behavior as before (no rlcp), and
if `strip_parent` is set, it will also still have the same behavior.
Finally, if the format is binary, `rlcp` is ignored too (as it doesn't
make sense).
So, this only changes if:
- your format is not binary; and
- you have files with `src` and `dst` set
Then, goreleaser will warn you to set `rlcp: true`.
## todo
- [x] docs
- [x] more tests probably
- [x] any ideas for a better name for the new config option?
fixes #3655
Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
2022-12-27 17:42:55 -03:00
|
|
|
return a[0:min]
|
2022-05-11 22:56:25 -03:00
|
|
|
}
|