/* * file.c - Open and read a file * * Written 2016 by Werner Almesberger * Copyright 2016 by Werner Almesberger * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. */ #include #include #include #include #include "util.h" #include "main.h" #include "git-file.h" #include "file.h" bool file_cat(const struct file *file, void *user, const char *line) { printf("%s\n", line); return 1; } static bool try_related(struct file *file) { const char *related; const char *slash; char *tmp; unsigned len; if (!file->related) return 0; if (*file->name == '/') return 0; related = file->related->name; slash = strrchr(related, '/'); if (!slash) return 0; len = slash + 1 - related; tmp = alloc_size(len + strlen(file->name) + 1); memcpy(tmp, related, len); strcpy(tmp + len, file->name); file->file = fopen(tmp, "r"); if (!file->file) { free(tmp); return 0; } if (verbose) fprintf(stderr, "reading %s\n", tmp); free((char *) file->name); file->name = tmp; return 1; } /* * @@@ logic isn't quite complete yet. It should go something like this: * * - if there is no related item, * - try file, * - if there is a colon, try VCS, * - give up * - if there is a related item, * - if it is a VCS, * - try the revision matching or predating (if different repo) that of the * related repo, ( - fall through and try as if it was a file * - try opening as file. If this fails, * - if the name of the file to open is absolute, give up * - try `dirname related`/file_ope_open * - give up * * @@@ should we see if non-VCS file is in a VCS ? E.g., * /home/my-libs/foo.lib 1234:/home/my-project/foo.sch * or maybe use : as indictor for VCS, i.e., * :/home/my-libs/foo.lib ... * * @@@ explicit revision should always win over related. */ void file_open(struct file *file, const char *name, const struct file *related) { char *colon, *tmp; file->name = stralloc(name); file->lineno = 0; file->related = related; file->vcs = NULL; file->file = fopen(name, "r"); if (file->file) { if (verbose) fprintf(stderr, "reading %s\n", name); return; } if (try_related(file)) return; if (verbose) perror(name); colon = strchr(name, ':'); if (!colon) { if (!verbose) perror(name); exit(1); } tmp = stralloc(name); tmp[colon - name] = 0; file->vcs = vcs_git_open(tmp, colon + 1); if (!file->vcs) { fprintf(stderr, "could not open %s:%s\n", tmp, colon + 1); exit(1); } free(tmp); } void file_read(struct file *file, bool (*parse)(const struct file *file, void *user, const char *line), void *user) { char buf[1000]; char *nl; if (file->vcs) { vcs_read(file->vcs, file, parse, user); return; } while (fgets(buf, sizeof(buf), file->file)) { nl = strchr(buf, '\n'); if (nl) *nl = 0; file->lineno++; if (!parse(file, user, buf)) break; } } void file_close(struct file *file) { if (file->file) fclose(file->file); if (file->vcs) vcs_close(file->vcs); free((char *) file->name); }