2016-08-02 17:00:08 +03:00
|
|
|
/*
|
|
|
|
* 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 <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
#include "main.h"
|
|
|
|
#include "git-file.h"
|
|
|
|
#include "file.h"
|
|
|
|
|
|
|
|
|
2016-08-02 21:36:18 +03:00
|
|
|
bool file_cat(const struct file *file, void *user, const char *line)
|
2016-08-02 17:00:08 +03:00
|
|
|
{
|
|
|
|
printf("%s\n", line);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-02 22:27:08 +03:00
|
|
|
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)
|
2016-08-02 17:00:08 +03:00
|
|
|
{
|
|
|
|
char *colon, *tmp;
|
|
|
|
|
2016-08-02 21:36:18 +03:00
|
|
|
file->name = stralloc(name);
|
|
|
|
file->lineno = 0;
|
|
|
|
file->related = related;
|
|
|
|
file->vcs = NULL;
|
|
|
|
|
|
|
|
file->file = fopen(name, "r");
|
|
|
|
if (file->file) {
|
2016-08-02 17:00:08 +03:00
|
|
|
if (verbose)
|
|
|
|
fprintf(stderr, "reading %s\n", name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-02 22:27:08 +03:00
|
|
|
if (try_related(file))
|
|
|
|
return;
|
|
|
|
|
2016-08-02 17:00:08 +03:00
|
|
|
if (verbose)
|
|
|
|
perror(name);
|
|
|
|
|
|
|
|
colon = strchr(name, ':');
|
|
|
|
if (!colon) {
|
|
|
|
if (!verbose)
|
|
|
|
perror(name);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = stralloc(name);
|
|
|
|
tmp[colon - name] = 0;
|
2016-08-02 21:36:18 +03:00
|
|
|
file->vcs = vcs_git_open(tmp, colon + 1);
|
|
|
|
if (!file->vcs) {
|
|
|
|
fprintf(stderr, "could not open %s:%s\n", tmp, colon + 1);
|
|
|
|
exit(1);
|
|
|
|
}
|
2016-08-02 17:00:08 +03:00
|
|
|
free(tmp);
|
|
|
|
}
|
2016-08-02 21:36:18 +03:00
|
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|