summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRusty Russell <rusty@rustcorp.com.au>2011-07-19 17:30:49 +0930
committerRusty Russell <rusty@rustcorp.com.au>2011-07-19 17:30:49 +0930
commit8de1e8a6faef77773cbf8ed237eb4e330143d2f2 (patch)
tree0d736b893803e9f5564f2a8f8d859e9830f7c78f
parentda8558a174da778aeb9dd95e62e023d39349b948 (diff)
ccanlint: make a license enum, and parse the license string to set it.
This improves on the current ad-hoc methods, and also fixes a bug where we mapped "GPLv2" to the GPLv3 symlink.
-rw-r--r--tools/ccanlint/ccanlint.h18
-rw-r--r--tools/ccanlint/tests/license_exists.c105
2 files changed, 91 insertions, 32 deletions
diff --git a/tools/ccanlint/ccanlint.h b/tools/ccanlint/ccanlint.h
index 23b6492d..b9965f75 100644
--- a/tools/ccanlint/ccanlint.h
+++ b/tools/ccanlint/ccanlint.h
@@ -16,6 +16,21 @@
4 == Describe every action. */
extern int verbose;
+enum license {
+ LICENSE_LGPLv2_PLUS,
+ LICENSE_LGPLv2,
+ LICENSE_LGPLv3,
+ LICENSE_LGPL,
+ LICENSE_GPLv2_PLUS,
+ LICENSE_GPLv2,
+ LICENSE_GPLv3,
+ LICENSE_GPL,
+ LICENSE_BSD,
+ LICENSE_MIT,
+ LICENSE_PUBLIC_DOMAIN,
+ LICENSE_UNKNOWN
+};
+
struct manifest {
char *dir;
/* The module name, ie. final element of dir name */
@@ -43,6 +58,9 @@ struct manifest {
/* From tests/check_depends_exist.c */
struct list_head deps;
+
+ /* From tests/license_exists.c */
+ enum license license;
};
/* Get the manifest for a given directory. */
diff --git a/tools/ccanlint/tests/license_exists.c b/tools/ccanlint/tests/license_exists.c
index a841d665..9cf86083 100644
--- a/tools/ccanlint/tests/license_exists.c
+++ b/tools/ccanlint/tests/license_exists.c
@@ -10,8 +10,9 @@
#include <err.h>
#include <ccan/talloc/talloc.h>
#include <ccan/str/str.h>
+#include <ccan/str_talloc/str_talloc.h>
-static struct doc_section *find_license(const struct manifest *m)
+static struct doc_section *find_license_tag(const struct manifest *m)
{
struct doc_section *d;
@@ -24,44 +25,76 @@ static struct doc_section *find_license(const struct manifest *m)
return NULL;
}
-static const char *expected_link(const struct manifest *m,
- struct doc_section *d)
+static enum license which_license(struct doc_section *d)
{
- if (streq(d->lines[0], "GPL")
- || streq(d->lines[0], "GPLv3")
- || streq(d->lines[0], "GPLv3 or later")
- || streq(d->lines[0], "GPLv3 (or later)")
- || streq(d->lines[0], "GPL (3 or any later version)"))
- return "../../licenses/GPL-3";
- if (streq(d->lines[0], "GPLv2")
- || streq(d->lines[0], "GPLv2 or later")
- || streq(d->lines[0], "GPLv2 (or later)")
- || streq(d->lines[0], "GPL (2 or any later version)"))
- return "../../licenses/GPL-3";
- if (streq(d->lines[0], "LGPL")
- || streq(d->lines[0], "LGPLv3")
- || streq(d->lines[0], "LGPLv3 or later")
- || streq(d->lines[0], "LGPLv3 (or later)")
- || streq(d->lines[0], "LGPL (3 or any later version)"))
- return "../../licenses/LGPL-3";
- if (streq(d->lines[0], "LGPLv2")
- || streq(d->lines[0], "LGPLv2 or later")
- || streq(d->lines[0], "LGPLv2 (or later)")
- || streq(d->lines[0], "LGPL (2 or any later version)"))
- return "../../licenses/LGPL-2.1";
+ if (strstarts(d->lines[0], "GPL")) {
+ if (strchr(d->lines[0], '3'))
+ return LICENSE_GPLv3;
+ else if (strchr(d->lines[0], '2')) {
+ if (strreg(NULL, d->lines[0], "or (any )?later", NULL))
+ return LICENSE_GPLv2_PLUS;
+ else
+ return LICENSE_GPLv2;
+ }
+ return LICENSE_GPL;
+ }
+
+ if (strstarts(d->lines[0], "LGPL")) {
+ if (strchr(d->lines[0], '3'))
+ return LICENSE_LGPLv3;
+ else if (strchr(d->lines[0], '2')) {
+ if (strreg(NULL, d->lines[0], "or (any )?later", NULL))
+ return LICENSE_LGPLv2_PLUS;
+ else
+ return LICENSE_LGPLv2;
+ }
+ return LICENSE_LGPL;
+ }
if (streq(d->lines[0], "BSD-MIT")
|| streq(d->lines[0], "MIT"))
- return "../../licenses/BSD-MIT";
+ return LICENSE_MIT;
if (streq(d->lines[0], "BSD (3 clause)"))
+ return LICENSE_BSD;
+ if (strreg(NULL, d->lines[0], "[Pp]ublic [Dd]omain"))
+ return LICENSE_PUBLIC_DOMAIN;
+
+ return LICENSE_UNKNOWN;
+}
+
+static const char *expected_link(enum license license)
+{
+ switch (license) {
+ case LICENSE_LGPLv2_PLUS:
+ case LICENSE_LGPLv2:
+ return "../../licenses/LGPL-2.1";
+ case LICENSE_LGPLv3:
+ case LICENSE_LGPL:
+ return "../../licenses/LGPL-3";
+
+ case LICENSE_GPLv2_PLUS:
+ case LICENSE_GPLv2:
+ return "../../licenses/GPL-2";
+
+ case LICENSE_GPLv3:
+ case LICENSE_GPL:
+ return "../../licenses/GPL-3";
+
+ case LICENSE_BSD:
return "../../licenses/BSD-3CLAUSE";
- return NULL;
+
+ case LICENSE_MIT:
+ return "../../licenses/BSD-MIT";
+
+ default:
+ return NULL;
+ }
}
static void handle_license_link(struct manifest *m, struct score *score)
{
+ struct doc_section *d = find_license_tag(m);
const char *link = talloc_asprintf(m, "%s/LICENSE", m->dir);
- struct doc_section *d = find_license(m);
- const char *ldest = expected_link(m, d);
+ const char *ldest = expected_link(m->license);
char *q;
printf(
@@ -89,22 +122,25 @@ static void check_has_license(struct manifest *m,
const char *expected;
struct doc_section *d;
- d = find_license(m);
+ d = find_license_tag(m);
if (!d) {
score->error = talloc_strdup(score, "No License: tag in _info");
return;
}
+
+ m->license = which_license(d);
+
/* If they have a license tag at all, we pass. */
score->pass = true;
- expected = expected_link(m, d);
+ expected = expected_link(m->license);
len = readlink(license, buf, sizeof(buf));
if (len < 0) {
/* Could be a real file... OK if not a standard license. */
if (errno == EINVAL) {
if (!expected) {
- score->pass = true;
+ score->score = score->total;
return;
}
score->error
@@ -115,6 +151,11 @@ static void check_has_license(struct manifest *m,
return;
}
if (errno == ENOENT) {
+ /* Public domain doesn't really need a file. */
+ if (m->license == LICENSE_PUBLIC_DOMAIN) {
+ score->score = score->total;
+ return;
+ }
score->error = talloc_strdup(score,
"LICENSE does not exist");
if (expected)