#include <fcntl.h>
#include <stdbool.h>
#include "talloc/talloc.h"
-#include "string/string.h"
+#include "str/str.h"
+#include "str_talloc/str_talloc.h"
+#include "grab_file/grab_file.h"
+static char **grab_doc(const char *fname)
+{
+ char *file;
+ char **lines, **ret;
+ unsigned int i, num;
+ bool printing = false, printed = false;
+
+ file = grab_file(NULL, fname, NULL);
+ if (!file)
+ err(1, "Reading file %s", fname);
+ lines = strsplit(file, file, "\n", &num);
+ ret = talloc_array(NULL, char *, num+1);
+
+ num = 0;
+ for (i = 0; lines[i]; i++) {
+ if (streq(lines[i], "/**")) {
+ printing = true;
+ if (printed++)
+ talloc_append_string(ret[num], "\n");
+ } else if (streq(lines[i], " */"))
+ printing = false;
+ else if (printing) {
+ if (strstarts(lines[i], " * "))
+ ret[num++] = talloc_strdup(ret, lines[i]+3);
+ else if (strstarts(lines[i], " *"))
+ ret[num++] = talloc_strdup(ret, lines[i]+2);
+ else
+ errx(1, "Malformed line %s:%u", fname, i);
+ }
+ }
+ ret[num] = NULL;
+ talloc_free(file);
+ return ret;
+}
+
+static bool is_blank(const char *line)
+{
+ return line && line[strspn(line, " \t\n")] == '\0';
+}
+
+static bool is_section(const char *line, bool maybe_one_liner)
+{
+ unsigned int len;
+
+ len = strcspn(line, " \t\n:");
+ if (len == 0)
+ return false;
+
+ if (line[len] != ':')
+ return false;
+
+ /* If it can be a one-liner, a space is sufficient.*/
+ if (maybe_one_liner && (line[len+1] == ' ' || line[len+1] == '\t'))
+ return true;
+
+ return line[len] == ':' && is_blank(line+len+1);
+}
+
+
+static bool end_section(const char *line)
+{
+ return !line || is_section(line, true);
+}
+
+static unsigned int find_section(char **lines, const char *name,
+ bool maybe_one_liner)
+{
+ unsigned int i;
+
+ for (i = 0; lines[i]; i++) {
+ if (!is_section(lines[i], maybe_one_liner))
+ continue;
+ if (strncasecmp(lines[i], name, strlen(name)) != 0)
+ continue;
+ if (lines[i][strlen(name)] == ':')
+ break;
+ }
+ return i;
+}
int main(int argc, char *argv[])
{
- unsigned int i, j;
-
- for (i = 1; i < argc; i++) {
- char *file;
- char **lines;
- bool printing = false, printed = false;
-
- file = grab_file(NULL, argv[i]);
- if (!file)
- err(1, "Reading file %s", argv[i]);
- lines = strsplit(file, file, "\n", NULL);
-
- for (j = 0; lines[j]; j++) {
- if (streq(lines[j], "/**")) {
- printing = true;
- if (printed++)
- puts("\n");
- } else if (streq(lines[j], " */"))
- printing = false;
- else if (printing) {
- if (strstarts(lines[j], " * "))
- puts(lines[j] + 3);
- else if (strstarts(lines[j], " *"))
- puts(lines[j] + 2);
- else
- errx(1, "Malformed line %s:%u",
- argv[i], j);
+ unsigned int i;
+ const char *type;
+
+ if (argc < 3)
+ errx(1, "Usage: doc_extract TYPE <file>...\n"
+ "Where TYPE is author|licence|maintainer|summary|description|example|all");
+
+ type = argv[1];
+ for (i = 2; i < argc; i++) {
+ unsigned int line;
+ char **lines = grab_doc(argv[i]);
+
+ if (!lines[0])
+ errx(1, "No documentation in file");
+
+ /* Simple one-line fields. */
+ if (streq(type, "author")
+ || streq(type, "maintainer")
+ || streq(type, "licence")) {
+ line = find_section(lines, type, true);
+ if (lines[line]) {
+ const char *p = strchr(lines[line], ':') + 1;
+ p += strspn(p, " \t\n");
+ if (p[0] == '\0') {
+ /* Must be on next line. */
+ if (end_section(lines[line+1]))
+ errx(1, "Malformed %s", type);
+ puts(lines[line+1]);
+ } else
+ puts(p);
}
- }
- talloc_free(file);
+ } else if (streq(type, "summary")) {
+ /* Summary comes after - on first line. */
+ char *dash;
+
+ dash = strchr(lines[0], '-');
+ if (!dash)
+ errx(1, "Malformed first line: no -");
+ dash += strspn(dash, "- ");
+ puts(dash);
+ } else if (streq(type, "description")) {
+ line = 1;
+ while (is_blank(lines[line]))
+ line++;
+
+ while (!end_section(lines[line]))
+ puts(lines[line++]);
+ } else if (streq(type, "example")) {
+ line = find_section(lines, type, false);
+ if (lines[line]) {
+ unsigned int strip;
+ line++;
+
+ while (is_blank(lines[line]))
+ line++;
+
+ /* Examples can be indented. Take cue
+ * from first non-blank line. */
+ if (lines[line])
+ strip = strspn(lines[line], " \t");
+
+ while (!end_section(lines[line])) {
+ if (strspn(lines[line], " \t") >= strip)
+ puts(lines[line] + strip);
+ else
+ puts(lines[line]);
+ line++;
+ }
+ }
+ } else if (streq(type, "all")) {
+ for (line = 0; lines[line]; line++)
+ puts(lines[line]);
+ } else
+ errx(1, "Unknown type '%s'", type);
+
+ talloc_free(lines);
}
return 0;
}