Fix git-stripspace to process correctly long lines and spaces.

Now the implementation gets more memory to store completely
each line before removing trailing spaces, and does it right
when the last line of the file ends with spaces and no newline
at the end.

Function stripspace needs again to be non-static in order to call
it from "builtin-tag.c" and the upcoming "builtin-commit.c".
A new parameter skip_comments was also added to the stripspace
function to optionally strips every shell #comment from the input,
needed for doing this task on those programs.

Signed-off-by: Carlos Rica <jasampler@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Carlos Rica 2007-06-25 21:28:01 +02:00 committed by Junio C Hamano
parent 384f122b7c
commit 9690c118fa
2 changed files with 46 additions and 24 deletions

View File

@ -1,58 +1,79 @@
#include "builtin.h" #include "builtin.h"
#include "cache.h"
/* /*
* Remove empty lines from the beginning and end. * Remove trailing spaces from a line.
* *
* Turn multiple consecutive empty lines into just one * If the line ends with newline, it will be removed too.
* empty line. Return true if it is an incomplete line. * Returns the new length of the string.
*/ */
static int cleanup(char *line) static int cleanup(char *line, int len)
{ {
int len = strlen(line); if (len) {
if (line[len - 1] == '\n')
len--;
if (len && line[len-1] == '\n') { while (len) {
if (len == 1) unsigned char c = line[len - 1];
return 0;
do {
unsigned char c = line[len-2];
if (!isspace(c)) if (!isspace(c))
break; break;
line[len-2] = '\n';
len--; len--;
line[len] = 0;
} while (len > 1);
return 0;
} }
return 1; line[len] = 0;
}
return len;
} }
static void stripspace(FILE *in, FILE *out) /*
* Remove empty lines from the beginning and end
* and also trailing spaces from every line.
*
* Turn multiple consecutive empty lines between paragraphs
* into just one empty line.
*
* If the input has only empty lines and spaces,
* no output will be produced.
*
* Enable skip_comments to skip every line starting with "#".
*/
void stripspace(FILE *in, FILE *out, int skip_comments)
{ {
int empties = -1; int empties = -1;
int incomplete = 0; int alloc = 1024;
char line[1024]; char *line = xmalloc(alloc);
while (fgets(line, sizeof(line), in)) { while (fgets(line, alloc, in)) {
incomplete = cleanup(line); int len = strlen(line);
while (len == alloc - 1 && line[len - 1] != '\n') {
alloc = alloc_nr(alloc);
line = xrealloc(line, alloc);
fgets(line + len, alloc - len, in);
len += strlen(line + len);
}
if (skip_comments && line[0] == '#')
continue;
len = cleanup(line, len);
/* Not just an empty line? */ /* Not just an empty line? */
if (line[0] != '\n') { if (len) {
if (empties > 0) if (empties > 0)
fputc('\n', out); fputc('\n', out);
empties = 0; empties = 0;
fputs(line, out); fputs(line, out);
fputc('\n', out);
continue; continue;
} }
if (empties < 0) if (empties < 0)
continue; continue;
empties++; empties++;
} }
if (incomplete) free(line);
fputc('\n', out);
} }
int cmd_stripspace(int argc, const char **argv, const char *prefix) int cmd_stripspace(int argc, const char **argv, const char *prefix)
{ {
stripspace(stdin, stdout); stripspace(stdin, stdout, 0);
return 0; return 0;
} }

View File

@ -7,6 +7,7 @@ extern const char git_version_string[];
extern const char git_usage_string[]; extern const char git_usage_string[];
extern void help_unknown_cmd(const char *cmd); extern void help_unknown_cmd(const char *cmd);
extern void stripspace(FILE *in, FILE *out, int skip_comments);
extern int write_tree(unsigned char *sha1, int missing_ok, const char *prefix); extern int write_tree(unsigned char *sha1, int missing_ok, const char *prefix);
extern void prune_packed_objects(int); extern void prune_packed_objects(int);