summaryrefslogtreecommitdiff
path: root/pkgtools
diff options
context:
space:
mode:
authorrillig <rillig@pkgsrc.org>2020-04-30 21:15:03 +0000
committerrillig <rillig@pkgsrc.org>2020-04-30 21:15:03 +0000
commit16ead704053adf3f7b8af5b1e0d64370ba78d471 (patch)
treeb86361e5ddf29794fc37d9f3b713313eb6f90473 /pkgtools
parent152bee3addcb376bb4fd8301da2686a477994810 (diff)
downloadpkgsrc-16ead704053adf3f7b8af5b1e0d64370ba78d471.tar.gz
pkgtools/pkglint: update to 20.1.3
Changes since 20.1.2: Stricter check for Python version numbers. Before, 25 and 26 had not been marked as wrong. In assignments like PKGNAME=${DISTNAME:S,from,to,}, modifiers that don't have any effect generate a note. Most of these modifiers are redundant or outdated. Patches must not add well-known absolute paths like /usr/pkg, /var and /etc since these must be overridable by the pkgsrc user. Other absolute paths continue to be allowed.
Diffstat (limited to 'pkgtools')
-rw-r--r--pkgtools/pkglint/Makefile4
-rw-r--r--pkgtools/pkglint/files/homepage_test.go3
-rw-r--r--pkgtools/pkglint/files/mkassignchecker_test.go20
-rw-r--r--pkgtools/pkglint/files/mklinechecker.go2
-rw-r--r--pkgtools/pkglint/files/mklinechecker_test.go2
-rw-r--r--pkgtools/pkglint/files/mktypes.go10
-rw-r--r--pkgtools/pkglint/files/mktypes_test.go34
-rw-r--r--pkgtools/pkglint/files/mkvarusechecker_test.go4
-rw-r--r--pkgtools/pkglint/files/package.go11
-rw-r--r--pkgtools/pkglint/files/package_test.go45
-rw-r--r--pkgtools/pkglint/files/patches.go87
-rw-r--r--pkgtools/pkglint/files/patches_test.go162
-rw-r--r--pkgtools/pkglint/files/pkglint.go6
-rw-r--r--pkgtools/pkglint/files/pkgver/vercmp.go23
-rw-r--r--pkgtools/pkglint/files/plist.go7
-rw-r--r--pkgtools/pkglint/files/regex/regex.go14
-rw-r--r--pkgtools/pkglint/files/util.go10
-rw-r--r--pkgtools/pkglint/files/vardefs.go12
-rw-r--r--pkgtools/pkglint/files/vartypecheck.go4
19 files changed, 366 insertions, 94 deletions
diff --git a/pkgtools/pkglint/Makefile b/pkgtools/pkglint/Makefile
index b80898a1cca..3bea1707805 100644
--- a/pkgtools/pkglint/Makefile
+++ b/pkgtools/pkglint/Makefile
@@ -1,6 +1,6 @@
-# $NetBSD: Makefile,v 1.641 2020/04/13 19:46:44 rillig Exp $
+# $NetBSD: Makefile,v 1.642 2020/04/30 21:15:03 rillig Exp $
-PKGNAME= pkglint-20.1.2
+PKGNAME= pkglint-20.1.3
CATEGORIES= pkgtools
DISTNAME= tools
MASTER_SITES= ${MASTER_SITE_GITHUB:=golang/}
diff --git a/pkgtools/pkglint/files/homepage_test.go b/pkgtools/pkglint/files/homepage_test.go
index 49a6d90e491..ca7ae4f9efc 100644
--- a/pkgtools/pkglint/files/homepage_test.go
+++ b/pkgtools/pkglint/files/homepage_test.go
@@ -393,7 +393,8 @@ func (s *Suite) Test_HomepageChecker_checkReachable(c *check.C) {
"https://no-such-name.example.org/",
// The "unknown network error" is for compatibility with Go < 1.13.
`^WARN: filename\.mk:1: Homepage "https://no-such-name.example.org/" `+
- `cannot be checked: (name not found|timeout|unknown network error:.*)$`)
+ `cannot be checked: `+
+ `(name not found|timeout|connection refused|unknown network error:.*)$`)
// Syntactically invalid URLs are silently skipped since VartypeCheck.URL
// already warns about them.
diff --git a/pkgtools/pkglint/files/mkassignchecker_test.go b/pkgtools/pkglint/files/mkassignchecker_test.go
index cff62ac1175..7164f4ae625 100644
--- a/pkgtools/pkglint/files/mkassignchecker_test.go
+++ b/pkgtools/pkglint/files/mkassignchecker_test.go
@@ -995,23 +995,25 @@ func (s *Suite) Test_MkAssignChecker_checkVarassignDecreasingVersions(c *check.C
mklines.Check()
- // Half of these warnings are from VartypeCheck.Version, the
- // other half are from checkVarassignDecreasingVersions.
- // Strictly speaking some of them are redundant, but that would
- // mean to reject only variable references in checkVarassignDecreasingVersions.
- // This is probably ok.
- // TODO: Fix this.
+ // Half of these warnings are from VartypeCheck.Enum,
+ // the other half are from checkVarassignDecreasingVersions.
+ // Strictly speaking some of them are redundant, but that's ok.
+ // They all need to be fixed in the end.
t.CheckOutputLines(
- "WARN: Makefile:2: Invalid version number \"__future__\".",
+ "WARN: Makefile:2: \"__future__\" is not valid for PYTHON_VERSIONS_ACCEPTED. "+
+ "Use one of { 27 36 37 38 } instead.",
"ERROR: Makefile:2: Value \"__future__\" for "+
"PYTHON_VERSIONS_ACCEPTED must be a positive integer.",
- "WARN: Makefile:3: Invalid version number \"-13\".",
+ "WARN: Makefile:3: \"-13\" is not valid for PYTHON_VERSIONS_ACCEPTED. "+
+ "Use one of { 27 36 37 38 } instead.",
"ERROR: Makefile:3: Value \"-13\" for "+
"PYTHON_VERSIONS_ACCEPTED must be a positive integer.",
"ERROR: Makefile:4: Value \"${PKGVERSION_NOREV}\" for "+
"PYTHON_VERSIONS_ACCEPTED must be a positive integer.",
"WARN: Makefile:5: The values for PYTHON_VERSIONS_ACCEPTED "+
- "should be in decreasing order (37 before 36).")
+ "should be in decreasing order (37 before 36).",
+ "WARN: Makefile:6: \"25\" is not valid for PYTHON_VERSIONS_ACCEPTED. "+
+ "Use one of { 27 36 37 38 } instead.")
}
func (s *Suite) Test_MkAssignChecker_checkVarassignMiscRedundantInstallationDirs__AUTO_MKDIRS_yes(c *check.C) {
diff --git a/pkgtools/pkglint/files/mklinechecker.go b/pkgtools/pkglint/files/mklinechecker.go
index 7e4818c0f17..5f74fcb5064 100644
--- a/pkgtools/pkglint/files/mklinechecker.go
+++ b/pkgtools/pkglint/files/mklinechecker.go
@@ -98,7 +98,7 @@ func (ck MkLineChecker) checkTextWrksrcDotDot(text string) {
"",
"Example:",
"",
- "\tWRKSRC=\t${WRKDIR}",
+ "\tWRKSRC=\t\t${WRKDIR}",
"\tCONFIGURE_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src",
"\tBUILD_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src ${WRKSRC}/cmd",
"",
diff --git a/pkgtools/pkglint/files/mklinechecker_test.go b/pkgtools/pkglint/files/mklinechecker_test.go
index c397163629b..2c6126c1908 100644
--- a/pkgtools/pkglint/files/mklinechecker_test.go
+++ b/pkgtools/pkglint/files/mklinechecker_test.go
@@ -161,7 +161,7 @@ func (s *Suite) Test_MkLineChecker_checkText__WRKSRC(c *check.C) {
"",
"\tExample:",
"",
- "\t\tWRKSRC=\t${WRKDIR}",
+ "\t\tWRKSRC=\t\t${WRKDIR}",
"\t\tCONFIGURE_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src",
"\t\tBUILD_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src ${WRKSRC}/cmd",
"",
diff --git a/pkgtools/pkglint/files/mktypes.go b/pkgtools/pkglint/files/mktypes.go
index b862346c075..9ad08e0ae9d 100644
--- a/pkgtools/pkglint/files/mktypes.go
+++ b/pkgtools/pkglint/files/mktypes.go
@@ -40,7 +40,7 @@ func NewMkVarUse(varname string, modifiers ...MkVarUseModifier) *MkVarUse {
func (vu *MkVarUse) String() string { return sprintf("${%s%s}", vu.varname, vu.Mod()) }
type MkVarUseModifier struct {
- Text string
+ Text string // The text of the modifier, without the initial colon.
}
func (m MkVarUseModifier) IsQ() bool { return m.Text == "Q" }
@@ -59,13 +59,13 @@ func (m MkVarUseModifier) MatchSubst() (ok bool, regex bool, from string, to str
//
// Example:
// MkVarUseModifier{"S,name,file,g"}.Subst("distname-1.0") => "distfile-1.0"
-func (m MkVarUseModifier) Subst(str string) (string, bool) {
+func (m MkVarUseModifier) Subst(str string) (bool, string) {
// XXX: The call to MatchSubst is usually redundant because MatchSubst
// is typically called directly before calling Subst.
// This comes from a time when there was no boolean return value.
ok, isRegex, from, to, options := m.MatchSubst()
if !ok {
- return "", false
+ return false, ""
}
leftAnchor := hasPrefix(from, "^")
@@ -86,14 +86,14 @@ func (m MkVarUseModifier) Subst(str string) (string, bool) {
if isRegex {
// XXX: Maybe implement regular expression substitutions later.
- return "", false
+ return false, ""
}
ok, result := m.EvalSubst(str, leftAnchor, from, rightAnchor, to, options)
if trace.Tracing && ok && result != str {
trace.Stepf("Subst: %q %q => %q", str, m.Text, result)
}
- return result, ok
+ return ok, result
}
// mkopSubst evaluates make(1)'s :S substitution operator.
diff --git a/pkgtools/pkglint/files/mktypes_test.go b/pkgtools/pkglint/files/mktypes_test.go
index d9911462585..b927bd91e5d 100644
--- a/pkgtools/pkglint/files/mktypes_test.go
+++ b/pkgtools/pkglint/files/mktypes_test.go
@@ -90,60 +90,60 @@ func (s *Suite) Test_MkVarUseModifier_MatchSubst__backslash_as_separator(c *chec
func (s *Suite) Test_MkVarUseModifier_Subst(c *check.C) {
t := s.Init(c)
- test := func(mod, str, result string, ok bool) {
+ test := func(mod, str string, ok bool, result string) {
m := MkVarUseModifier{mod}
- actualResult, actualOk := m.Subst(str)
+ actualOk, actualResult := m.Subst(str)
t.CheckDeepEquals(
- []interface{}{actualResult, actualOk},
- []interface{}{result, ok})
+ []interface{}{actualOk, actualResult},
+ []interface{}{ok, result})
}
- test("???", "anything", "", false)
+ test("???", "anything", false, "")
- test("S,from,to,", "from", "to", true)
+ test("S,from,to,", "from", true, "to")
- test("C,from,to,", "from", "to", true)
+ test("C,from,to,", "from", true, "to")
- test("C,syntax error", "anything", "", false)
+ test("C,syntax error", "anything", false, "")
// The substitution modifier does not match, therefore
// the value is returned unmodified, but successful.
- test("C,no_match,replacement,", "value", "value", true)
+ test("C,no_match,replacement,", "value", true, "value")
// As of December 2019, pkglint doesn't know how to handle
// complicated :C modifiers.
- test("C,.*,,", "anything", "", false)
+ test("C,.*,,", "anything", false, "")
// When given a modifier that is not actually a :S or :C, Subst
// doesn't do anything.
- test("Mpattern", "anything", "", false)
+ test("Mpattern", "anything", false, "")
- test("S,from,to,", "from a to b", "to a to b", true)
+ test("S,from,to,", "from a to b", true, "to a to b")
// Since the replacement text is not a simple string, the :C modifier
// cannot be treated like the :S modifier. The variable could contain
// one of the special characters that would need to be escaped in the
// replacement text.
- test("C,from,${VAR},", "from a to b", "", false)
+ test("C,from,${VAR},", "from a to b", false, "")
// As of December 2019, nothing is substituted. If pkglint should ever
// handle variables in the modifier, this test would need to provide a
// context in which to resolve the variables. If that happens, the
// .TARGET variable needs to be set to "target".
- test("S/$@/replaced/", "The target", "The target", true)
- test("S,${PREFIX},/prefix,", "${PREFIX}/dir", "", false)
+ test("S/$@/replaced/", "The target", true, "The target")
+ test("S,${PREFIX},/prefix,", "${PREFIX}/dir", false, "")
// Just for code coverage.
t.DisableTracing()
- test("S,long,long long,g", "A long story", "A long long story", true)
+ test("S,long,long long,g", "A long story", true, "A long long story")
t.EnableTracing()
// And now again with full tracing, to investigate cases where
// pkglint produces results that are not easily understandable.
t.EnableTracingToLog()
- test("S,long,long long,g", "A long story", "A long long story", true)
+ test("S,long,long long,g", "A long story", true, "A long long story")
t.EnableTracing()
t.CheckOutputLines(
"TRACE: Subst: \"A long story\" " +
diff --git a/pkgtools/pkglint/files/mkvarusechecker_test.go b/pkgtools/pkglint/files/mkvarusechecker_test.go
index 67b50f050d6..ac063a982d9 100644
--- a/pkgtools/pkglint/files/mkvarusechecker_test.go
+++ b/pkgtools/pkglint/files/mkvarusechecker_test.go
@@ -592,13 +592,13 @@ func (s *Suite) Test_MkVarUseChecker_checkPermissions__indirectly(c *check.C) {
t.SetUpVartypes()
mklines := t.NewMkLines("file.mk",
MkCvsID,
- "IGNORE_PKG.package=\t${ONLY_FOR_UNPRIVILEGED}")
+ "IGNORE_PKG.package=\t${NOT_FOR_UNPRIVILEGED}")
mklines.Check()
t.CheckOutputLines(
"WARN: file.mk:2: IGNORE_PKG.package should be set to YES or yes.",
- "WARN: file.mk:2: ONLY_FOR_UNPRIVILEGED should not be used indirectly at load time (via IGNORE_PKG.package).")
+ "WARN: file.mk:2: NOT_FOR_UNPRIVILEGED should not be used indirectly at load time (via IGNORE_PKG.package).")
}
// This test is only here for branch coverage.
diff --git a/pkgtools/pkglint/files/package.go b/pkgtools/pkglint/files/package.go
index 9b0e881c922..cbd18d941b9 100644
--- a/pkgtools/pkglint/files/package.go
+++ b/pkgtools/pkglint/files/package.go
@@ -275,7 +275,6 @@ func (pkg *Package) loadPackageMakefile() (*MkLines, *MkLines) {
return mainLines, allLines
}
-// TODO: What is allLines used for, is it still necessary? Would it be better as a field in Package?
func (pkg *Package) parse(mklines *MkLines, allLines *MkLines, includingFileForUsedCheck CurrPath, main bool) bool {
if trace.Tracing {
defer trace.Call(mklines.lines.Filename)()
@@ -891,7 +890,6 @@ func (pkg *Package) CheckVarorder(mklines *MkLines) {
{"NOT_FOR_COMPILER", many},
{"ONLY_FOR_COMPILER", many},
{"NOT_FOR_UNPRIVILEGED", optional},
- {"ONLY_FOR_UNPRIVILEGED", optional},
emptyLine,
{"BUILD_DEPENDS", many},
{"TOOL_DEPENDS", many},
@@ -1147,7 +1145,7 @@ func (pkg *Package) determineEffectivePkgVars() {
effname := pkgname
if distname != "" && effname != "" {
- merged, ok := pkg.pkgnameFromDistname(effname, distname)
+ merged, ok := pkg.pkgnameFromDistname(effname, distname, pkgnameLine)
if ok {
effname = merged
}
@@ -1209,7 +1207,7 @@ func (pkg *Package) nbPart() string {
return ""
}
-func (pkg *Package) pkgnameFromDistname(pkgname, distname string) (string, bool) {
+func (pkg *Package) pkgnameFromDistname(pkgname, distname string, diag Diagnoser) (string, bool) {
tokens, rest := NewMkLexer(pkgname, nil).MkTokens()
if rest != "" {
return "", false
@@ -1228,7 +1226,10 @@ func (pkg *Package) pkgnameFromDistname(pkgname, distname string) (string, bool)
for _, mod := range token.Varuse.modifiers {
if mod.IsToLower() {
newDistname = strings.ToLower(newDistname)
- } else if subst, ok := mod.Subst(newDistname); ok {
+ } else if ok, subst := mod.Subst(newDistname); ok {
+ if subst == newDistname && !containsVarUse(subst) {
+ diag.Notef("The modifier :%s does not have an effect.", mod.Text)
+ }
newDistname = subst
} else {
return "", false
diff --git a/pkgtools/pkglint/files/package_test.go b/pkgtools/pkglint/files/package_test.go
index f9bdf9df121..b772c1049d8 100644
--- a/pkgtools/pkglint/files/package_test.go
+++ b/pkgtools/pkglint/files/package_test.go
@@ -2609,6 +2609,43 @@ func (s *Suite) Test_Package_determineEffectivePkgVars__ineffective_C_modifier(c
pkg.check(files, mklines, allLines)
t.CheckEquals(pkg.EffectivePkgname, "distname-1.0")
+ t.CheckOutputLines(
+ "NOTE: ~/category/package/Makefile:4: " +
+ "The modifier :C:does_not_match:replacement: does not have an effect.")
+}
+
+func (s *Suite) Test_Package_determineEffectivePkgVars__ineffective_S_modifier_with_variable(c *check.C) {
+ t := s.Init(c)
+
+ t.SetUpPackage("category/package",
+ "VERSION=\t1.008",
+ "DISTNAME=\tdistname-v${VERSION}",
+ "PKGNAME=\t${DISTNAME:S/v1/1/}")
+ t.FinishSetUp()
+ pkg := NewPackage(t.File("category/package"))
+ files, mklines, allLines := pkg.load()
+
+ pkg.check(files, mklines, allLines)
+
+ // TODO: Expand ${VERSION}, that's pretty simple.
+ t.CheckEquals(pkg.EffectivePkgname, "") // Because of the unexpanded VERSION.
+ t.CheckOutputEmpty()
+}
+
+func (s *Suite) Test_Package_determineEffectivePkgVars__effective_S_modifier_with_variable(c *check.C) {
+ t := s.Init(c)
+
+ t.SetUpPackage("category/package",
+ "MINOR_VERSION=\t1.008",
+ "DISTNAME=\tdistname-v1.${MINOR_VERSION}",
+ "PKGNAME=\t${DISTNAME:S/v1/1/}")
+ t.FinishSetUp()
+ pkg := NewPackage(t.File("category/package"))
+ files, mklines, allLines := pkg.load()
+
+ pkg.check(files, mklines, allLines)
+
+ t.CheckEquals(pkg.EffectivePkgname, "") // because of MINOR_VERSION
t.CheckOutputEmpty()
}
@@ -2724,9 +2761,11 @@ func (s *Suite) Test_Package_pkgnameFromDistname(c *check.C) {
// the package version. Therefore it is discarded completely.
test("${DISTNAME:S|^lib||}", "libncurses", "")
- // The substitution succeeds, but the substituted value is missing
- // the package version. Therefore it is discarded completely.
- test("${DISTNAME:S|^lib||}", "mylib", "")
+ // The substitution does not have an effect.
+ // The substituted value is missing the package version.
+ // Therefore it is discarded completely.
+ test("${DISTNAME:S|^lib||}", "mylib", "",
+ "NOTE: ~/category/package/Makefile:4: The modifier :S|^lib|| does not have an effect.")
test("${DISTNAME:tl:S/-/./g:S/he/-/1}", "SaxonHE9-5-0-1J", "saxon-9.5.0.1j")
diff --git a/pkgtools/pkglint/files/patches.go b/pkgtools/pkglint/files/patches.go
index 31c71a3d57a..34147872aaa 100644
--- a/pkgtools/pkglint/files/patches.go
+++ b/pkgtools/pkglint/files/patches.go
@@ -224,15 +224,90 @@ func (ck *PatchChecker) checkConfigure(addedText string, isConfigure bool) {
}
func (ck *PatchChecker) checkAddedLine(addedText string) {
- if !matches(addedText, `/usr/pkg\b`) {
- return
+ dirs := regcomp(`(?:^|[^.@)}])(/usr/pkg|/var|/etc)([^\w-]|$)`)
+ for _, m := range dirs.FindAllStringSubmatchIndex(addedText, -1) {
+ before := addedText[:m[2]]
+ dir := NewPath(addedText[m[2]:m[3]])
+ ck.checkAddedAbsPath(before, dir, addedText[m[4]:])
}
+}
+func (ck *PatchChecker) checkAddedAbsPath(before string, dir Path, after string) {
line := ck.llex.PreviousLine()
- line.Errorf("Patches must not hard-code the pkgsrc PREFIX.")
- line.Explain(
- "Instead of hard-coding /usr/pkg, packages should use the PREFIX variable.",
- "The usual way of doing this is to use the SUBST framework in mk/subst.mk.")
+
+ // Remove the #define from C and C++ macros.
+ before = replaceAll(before, `^[ \t]*#[ \t]*define[ \t]*\w+[ \t]*(.+)[ \t]*$`, "$1")
+
+ // Remove the "set(VAR" from CMakeLists.txt.
+ before = replaceAll(before, `^[ \t]*set\(\w+[ \t]*`, "")
+
+ // Ignore comments in shell programs.
+ if m, first := match1(before, `^[ \t]*#[ \t]*(\w*)`); m && first != "define" {
+ return
+ }
+
+ // Ignore paths inside C-style comments.
+ if contains(before, "/*") && contains(after, "*/") {
+ return
+ }
+
+ // Ignore composed C string literals such as PREFIX "/etc".
+ if matches(before, `\w+[ \t]*"$`) {
+ return
+ }
+
+ // Ignore shell literals such as $PREFIX/etc.
+ // But keep compiler options like -I/usr/pkg even though they look
+ // like a relative pathname.
+ if matches(before, `\w$`) && !matches(before, `(^|[ \t])-(I|L|R|rpath|Wl,-R)$`) {
+ return
+ }
+
+ switch dir {
+ case "/usr/pkg":
+
+ line.Errorf("Patches must not hard-code the pkgsrc PREFIX.")
+ line.Explain(
+ "Not every pkgsrc installation uses /usr/pkg as its PREFIX.",
+ "To keep the PREFIX configurable, the patch files should contain",
+ "the placeholder @PREFIX@ instead.",
+ "",
+ "In the pre-configure stage, this placeholder should then be",
+ "replaced with the actual configuration directory",
+ "using a SUBST block containing SUBST_VARS.dirs=PREFIX.",
+ "See mk/subst.mk for details.")
+
+ case "/var":
+ afterPath := NewPath(after)
+ if afterPath.HasPrefixPath("/tmp") || afterPath.HasPrefixPath("/shm") {
+ break
+ }
+
+ line.Errorf("Patches must not hard-code the pkgsrc VARBASE.")
+ line.Explain(
+ "Not every pkgsrc installation uses /var as its directory",
+ "for writable files.",
+ "To keep the VARBASE configurable, the patch files should",
+ "contain the placeholder @VARBASE@ instead.",
+ "",
+ "In the pre-configure stage, this placeholder should then be",
+ "replaced with the actual configuration directory",
+ "using a SUBST block containing SUBST_VARS.dirs=VARBASE.",
+ "See mk/subst.mk for details.")
+
+ default:
+ line.Errorf("Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+ line.Explain(
+ "Not every pkgsrc installation uses /etc as its directory",
+ "for configuration files.",
+ "To keep the PKG_SYSCONFDIR configurable, the patch files should",
+ "contain the placeholder @PKG_SYSCONFDIR@ instead.",
+ "",
+ "In the pre-configure stage, this placeholder should then be",
+ "replaced with the actual configuration directory",
+ "using a SUBST block containing SUBST_VARS.dirs=PKG_SYSCONFDIR.",
+ "See mk/subst.mk for details.")
+ }
}
func (ck *PatchChecker) checktextUniHunkCr() {
diff --git a/pkgtools/pkglint/files/patches_test.go b/pkgtools/pkglint/files/patches_test.go
index 23977c9e97f..542338e73a4 100644
--- a/pkgtools/pkglint/files/patches_test.go
+++ b/pkgtools/pkglint/files/patches_test.go
@@ -754,6 +754,168 @@ func (s *Suite) Test_PatchChecker_checkConfigure__configure_ac(c *check.C) {
"ERROR: ~/patch-aa:9: This code must not be included in patches.")
}
+func (s *Suite) Test_PatchChecker_checkAddedAbsPath(c *check.C) {
+ t := s.Init(c)
+
+ test := func(addedLine string, diagnostics ...string) {
+ lines := t.NewLines("patch-file",
+ CvsID,
+ "",
+ "Demonstrates absolute paths.",
+ "",
+ "--- before",
+ "+++ after",
+ "@@ -1,0 +1,1 @@",
+ "+"+addedLine)
+
+ CheckLinesPatch(lines, nil)
+
+ t.CheckOutput(diagnostics)
+ }
+
+ test(
+ "/usr/pkg",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ test(
+ "/usr/pkgsrc",
+ nil...)
+
+ test(
+ "/usr/pkg/bin",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ test(
+ "/usr/local:/usr/pkg:/opt",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ test(
+ "/var",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.")
+
+ test(
+ "/var/db",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.")
+
+ test(
+ "/var/run",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.")
+
+ // A well-known path that is not specific to pkgsrc.
+ test(
+ "/var/shm",
+ nil...)
+
+ // A well-known path that is not specific to pkgsrc.
+ test(
+ "/var/tmp",
+ nil...)
+
+ test(
+ "/etc",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+
+ // BSD-style Makefile
+ test(
+ "${PREFIX}/etc",
+ nil...)
+
+ // GNU automake-style Makefile
+ test(
+ "$(prefix)/etc",
+ nil...)
+
+ // C source code.
+ // Instead of PREFIX/etc, this should rather be PKG_SYSCONFDIR.
+ // This is a relative path because of the PREFIX.
+ test(
+ "const char *conf_dir = PREFIX \"/etc\"",
+ nil...)
+
+ // CMakeLists.txt
+ test(
+ "set(ETC_DIR \"/etc\")",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+
+ test(
+ "/etc/mk.conf",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+
+ test(
+ "/etc/rc.d/daemon",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+
+ test(
+ "/usr/pkg and /var and /etc",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.")
+
+ // From the --help text of a GNU configure script.
+ test(
+ "[PREFIX/etc]",
+ nil...)
+
+ // Shell program, default value for a variable.
+ test(
+ "DIR=${DIR-/var/bytebench}",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.")
+
+ // Shell program or Makefile.
+ // The placeholder will make this a relative path.
+ test(
+ "dir=@prefix@/etc",
+ nil...)
+
+ // Makefile with flags for the C compiler.
+ test(
+ "CFLAGS+= -I/usr/pkg/include",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ // Makefile with flags for the linker.
+ test(
+ "LDFLAGS+= -L/usr/pkg/lib",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ // Makefile with flags for the linker.
+ // There should be an additional warning for using COMPILER_RPATH_FLAG.
+ test(
+ "LDFLAGS+= -rpath/usr/pkg/lib",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ // Makefile with flags for the linker.
+ // There should be an additional warning for using COMPILER_RPATH_FLAG.
+ test(
+ "LDFLAGS+= -Wl,-R/usr/pkg/lib",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc PREFIX.")
+
+ // The dot before the "/etc" makes it a relative pathname.
+ test(
+ "cp ./etc/hostname /tmp")
+
+ // +> +# from /etc/inittab (SYSV systems)
+ // +ERROR: devel/tet3/patches/patch-ac:51: Patches must not hard-code the pkgsrc PKG_SYSCONFDIR.
+
+ test(
+ "# SysV /etc/install, /usr/sbin/install")
+
+ // C or C++ program, macro definition.
+ // This is an absolute path since the PID_FILE is the macro name,
+ // and not part of the macro body containing the path.
+ test(
+ "#define PID_FILE \"/var/run/daemon.pid\"",
+ "ERROR: patch-file:8: Patches must not hard-code the pkgsrc VARBASE.")
+
+ // This is a relative path because of the PREFIX before it.
+ test(
+ "#define PID_FILE PREFIX \"/etc/conf\"",
+ nil...)
+
+ test(
+ "#define L 150 /* Length of a line in /etc/passwd */",
+ nil...)
+}
+
func (s *Suite) Test_PatchChecker_checktextCvsID(c *check.C) {
t := s.Init(c)
diff --git a/pkgtools/pkglint/files/pkglint.go b/pkgtools/pkglint/files/pkglint.go
index 7732369a113..0d38983a4cd 100644
--- a/pkgtools/pkglint/files/pkglint.go
+++ b/pkgtools/pkglint/files/pkglint.go
@@ -823,6 +823,12 @@ func (ip *InterPackage) Enable() {
make(map[string]*Hash),
make(map[string]struct{}),
make(map[string]Location)}
+
+ // This is the only license that is added by an infrastructure file,
+ // mk/djbware.mk. The correct way to handle this situation would be
+ // to scan Package.check.allLines for LICENSE lines, but that would
+ // be too much just to cover this special case.
+ ip.UseLicense("djb-unlicense")
}
func (ip *InterPackage) Enabled() bool { return ip.hashes != nil }
diff --git a/pkgtools/pkglint/files/pkgver/vercmp.go b/pkgtools/pkglint/files/pkgver/vercmp.go
index ae5843b5309..e9f3f237875 100644
--- a/pkgtools/pkglint/files/pkgver/vercmp.go
+++ b/pkgtools/pkglint/files/pkgver/vercmp.go
@@ -31,7 +31,7 @@ func Compare(left, right string) int {
m := imax(len(lv.v), len(rv.v))
for i := 0; i < m; i++ {
- if c := icmp(lv.Field(i), rv.Field(i)); c != 0 {
+ if c := icmp(lv.field(i), rv.field(i)); c != 0 {
return c
}
}
@@ -52,24 +52,24 @@ func newVersion(vstr string) *version {
case lex.TestByteSet(textproc.Digit):
num := lex.NextBytesSet(textproc.Digit)
n, _ := strconv.Atoi(num)
- v.Add(n)
+ v.add(n)
case lex.SkipByte('_') || lex.SkipByte('.'):
- v.Add(0)
+ v.add(0)
case lex.SkipString("alpha"):
- v.Add(-3)
+ v.add(-3)
case lex.SkipString("beta"):
- v.Add(-2)
+ v.add(-2)
case lex.SkipString("pre"):
- v.Add(-1)
+ v.add(-1)
case lex.SkipString("rc"):
- v.Add(-1)
+ v.add(-1)
case lex.SkipString("pl"):
- v.Add(0)
+ v.add(0)
case lex.SkipString("nb"):
num := lex.NextBytesSet(textproc.Digit)
v.nb, _ = strconv.Atoi(num)
case lex.TestByteSet(textproc.Lower):
- v.Add(int(lex.Rest()[0] - 'a' + 1))
+ v.add(int(lex.Rest()[0] - 'a' + 1))
lex.Skip(1)
default:
lex.Skip(1)
@@ -78,11 +78,12 @@ func newVersion(vstr string) *version {
return v
}
-func (v *version) Add(i int) {
+//go:noinline
+func (v *version) add(i int) {
v.v = append(v.v, i)
}
-func (v *version) Field(i int) int {
+func (v *version) field(i int) int {
if i < len(v.v) {
return v.v[i]
}
diff --git a/pkgtools/pkglint/files/plist.go b/pkgtools/pkglint/files/plist.go
index 6a097ebb905..a31507806bb 100644
--- a/pkgtools/pkglint/files/plist.go
+++ b/pkgtools/pkglint/files/plist.go
@@ -370,7 +370,12 @@ func (ck *PlistChecker) checkPathLib(pline *PlistLine, rel RelPath) {
}
func (ck *PlistChecker) checkPathMan(pline *PlistLine) {
- m, catOrMan, section, manpage, ext, gz := match5(pline.text, `^man/(cat|man)(\w+)/(.*?)\.(\w+)(\.gz)?$`)
+ m, catOrMan, section, base := match3(pline.text, `^man/(cat|man)(\w+)/(.*)$`)
+ if !m {
+ // maybe: line.Warnf("Invalid filename %q for manual page.", text)
+ return
+ }
+ m, manpage, ext, gz := match3(base, `^(.*?)\.(\w+)(\.gz)?$`)
if !m {
// maybe: line.Warnf("Invalid filename %q for manual page.", text)
return
diff --git a/pkgtools/pkglint/files/regex/regex.go b/pkgtools/pkglint/files/regex/regex.go
index 97f40acf6ac..77cf6246e64 100644
--- a/pkgtools/pkglint/files/regex/regex.go
+++ b/pkgtools/pkglint/files/regex/regex.go
@@ -103,20 +103,6 @@ func (r *Registry) Match3(s string, re Pattern) (matched bool, m1, m2, m3 string
return
}
-func (r *Registry) Match4(s string, re Pattern) (matched bool, m1, m2, m3, m4 string) {
- if m := r.matchn(s, re, 4); m != nil {
- return true, m[1], m[2], m[3], m[4]
- }
- return
-}
-
-func (r *Registry) Match5(s string, re Pattern) (matched bool, m1, m2, m3, m4, m5 string) {
- if m := r.matchn(s, re, 5); m != nil {
- return true, m[1], m[2], m[3], m[4], m[5]
- }
- return
-}
-
func (r *Registry) ReplaceFirst(s string, re Pattern, replacement string) ([]string, string) {
if m := r.Compile(re).FindStringSubmatchIndex(s); m != nil {
replaced := s[:m[0]] + replacement + s[m[1]:]
diff --git a/pkgtools/pkglint/files/util.go b/pkgtools/pkglint/files/util.go
index a10f8328c8e..8c96eef38fb 100644
--- a/pkgtools/pkglint/files/util.go
+++ b/pkgtools/pkglint/files/util.go
@@ -58,12 +58,6 @@ func match2(s string, re regex.Pattern) (matched bool, m1, m2 string) {
func match3(s string, re regex.Pattern) (matched bool, m1, m2, m3 string) {
return G.res.Match3(s, re)
}
-func match4(s string, re regex.Pattern) (matched bool, m1, m2, m3, m4 string) {
- return G.res.Match4(s, re)
-}
-func match5(s string, re regex.Pattern) (matched bool, m1, m2, m3, m4, m5 string) {
- return G.res.Match5(s, re)
-}
func replaceAll(s string, re regex.Pattern, repl string) string {
return G.res.Compile(re).ReplaceAllString(s, repl)
}
@@ -558,7 +552,7 @@ func (o *Once) FirstTime(what string) bool {
key := o.keyString(what)
firstTime := o.check(key)
if firstTime && o.Trace {
- G.Logger.out.WriteLine(sprintf("FirstTime: %s", what))
+ G.Logger.out.WriteLine("FirstTime: " + what)
}
return firstTime
}
@@ -567,7 +561,7 @@ func (o *Once) FirstTimeSlice(whats ...string) bool {
key := o.keyStrings(whats)
firstTime := o.check(key)
if firstTime && o.Trace {
- G.Logger.out.WriteLine(sprintf("FirstTime: %s", strings.Join(whats, ", ")))
+ G.Logger.out.WriteLine("FirstTime: " + strings.Join(whats, ", "))
}
return firstTime
}
diff --git a/pkgtools/pkglint/files/vardefs.go b/pkgtools/pkglint/files/vardefs.go
index 0415a9ae8b2..2423e7939a9 100644
--- a/pkgtools/pkglint/files/vardefs.go
+++ b/pkgtools/pkglint/files/vardefs.go
@@ -1102,7 +1102,7 @@ func (reg *VarTypeRegistry) Init(src *Pkgsrc) {
reg.pkg("DJB_RESTRICTED", BtYesNo)
reg.pkg("DJB_SLASHPACKAGE", BtYesNo)
reg.pkg("DLOPEN_REQUIRE_PTHREADS", BtYesNo)
- reg.pkg("DL_AUTO_VARS", BtYes)
+ reg.pkg("DL_AUTO_VARS", BtYesNo)
reg.acllist("DL_LIBS", BtLdFlag,
PackageSettable,
"*: append, use")
@@ -1357,7 +1357,6 @@ func (reg *VarTypeRegistry) Init(src *Pkgsrc) {
reg.sysload("OBJECT_FMT", enum("COFF ECOFF ELF SOM XCOFF Mach-O PE a.out"))
reg.pkglistrat("ONLY_FOR_COMPILER", compilers)
reg.pkglistrat("ONLY_FOR_PLATFORM", BtMachinePlatformPattern)
- reg.pkgrat("ONLY_FOR_UNPRIVILEGED", BtYesNo)
reg.sysload("OPSYS", operatingSystems, DefinedIfInScope|NonemptyIfDefined)
reg.pkglistbl3("OPSYSVARS", BtVariableName)
reg.pkg("OSVERSION_SPECIFIC", BtYes)
@@ -1569,11 +1568,12 @@ func (reg *VarTypeRegistry) Init(src *Pkgsrc) {
"special:pyversion.mk: set",
"*: use, use-loadtime")
// See lang/python/pyversion.mk
+ py := reg.enumFromDirs(src, "lang", `^python(\d+)$`, "$1", "27 36 37 38")
reg.pkg("PYTHON_FOR_BUILD_ONLY", enum("yes no test tool YES"))
- reg.pkglistrat("PYTHON_VERSIONS_ACCEPTED", BtVersion)
- reg.pkglistrat("PYTHON_VERSIONS_INCOMPATIBLE", BtVersion)
- reg.usr("PYTHON_VERSION_DEFAULT", BtVersion)
- reg.usr("PYTHON_VERSION_REQD", BtVersion)
+ reg.pkglistrat("PYTHON_VERSIONS_ACCEPTED", py)
+ reg.pkglistrat("PYTHON_VERSIONS_INCOMPATIBLE", py)
+ reg.usr("PYTHON_VERSION_DEFAULT", py)
+ reg.sys("PYTHON_VERSION_REQD", py)
reg.pkglist("PYTHON_VERSIONED_DEPENDENCIES", BtPythonDependency)
reg.sys("RANLIB", BtShellCommand)
reg.pkglist("RCD_SCRIPTS", BtFilename)
diff --git a/pkgtools/pkglint/files/vartypecheck.go b/pkgtools/pkglint/files/vartypecheck.go
index 528f9e89b35..d0009d11a17 100644
--- a/pkgtools/pkglint/files/vartypecheck.go
+++ b/pkgtools/pkglint/files/vartypecheck.go
@@ -1387,7 +1387,7 @@ func (cv *VartypeCheck) URL() {
} else if containsVarUse(value) {
// No further checks
- } else if m, _, host, _, _ := match4(value, `^(https?|ftp|gopher)://([-0-9A-Za-z.]+)(?::(\d+))?/([-%&+,./0-9:;=?@A-Z_a-z~]|#)*$`); m {
+ } else if m, host := match1(value, `^(?:https?|ftp|gopher)://([-0-9A-Za-z.]+)(?::\d+)?/[-#%&+,./0-9:;=?@A-Z_a-z~]*$`); m {
if matches(host, `(?i)\.NetBSD\.org$`) && !matches(host, `\.NetBSD\.org$`) {
prefix := host[:len(host)-len(".NetBSD.org")]
fix := cv.Autofix()
@@ -1566,7 +1566,7 @@ func (cv *VartypeCheck) WrksrcSubdirectory() {
"",
"Example:",
"",
- "\tWRKSRC=\t${WRKDIR}",
+ "\tWRKSRC=\t\t${WRKDIR}",
"\tCONFIGURE_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src",
"\tBUILD_DIRS=\t${WRKSRC}/lib ${WRKSRC}/src ${WRKSRC}/cmd",
"",