2  * GIT - The information manager from hell
 
   4  * Copyright (C) Linus Torvalds, 2005
 
  11 #include "parse-options.h"
 
  16 static void pprint_tag(const unsigned char *sha1, const char *buf, unsigned long size)
 
  18         /* the parser in tag.c is useless here. */
 
  19         const char *endp = buf + size;
 
  26                 if (7 <= endp - cp && !memcmp("tagger ", cp, 7)) {
 
  27                         const char *tagger = cp;
 
  29                         /* Found the tagger line.  Copy out the contents
 
  30                          * of the buffer so far.
 
  32                         write_or_die(1, buf, cp - buf);
 
  35                          * Do something intelligent, like pretty-printing
 
  40                                         /* tagger to cp is a line
 
  41                                          * that has ident and time.
 
  43                                         const char *sp = tagger;
 
  47                                         while (sp < cp && *sp != '>')
 
  51                                                 write_or_die(1, tagger,
 
  56                                                !('0' <= *sp && *sp <= '9'))
 
  58                                         write_or_die(1, tagger, sp - tagger);
 
  59                                         date = strtoul(sp, &ep, 10);
 
  60                                         tz = strtol(ep, NULL, 10);
 
  61                                         sp = show_date(date, tz, 0);
 
  62                                         write_or_die(1, sp, strlen(sp));
 
  69                 if (cp < endp && *cp == '\n')
 
  73         /* At this point, we have copied out the header up to the end of
 
  74          * the tagger line and cp points at one past \n.  It could be the
 
  75          * next header line after the tagger line, or it could be another
 
  76          * \n that marks the end of the headers.  We need to copy out the
 
  80                 write_or_die(1, cp, endp - cp);
 
  83 static int cat_one_file(int opt, const char *exp_type, const char *obj_name)
 
  85         unsigned char sha1[20];
 
  86         enum object_type type;
 
  90         if (get_sha1(obj_name, sha1))
 
  91                 die("Not a valid object name %s", obj_name);
 
  96                 type = sha1_object_info(sha1, NULL);
 
  98                         printf("%s\n", typename(type));
 
 104                 type = sha1_object_info(sha1, &size);
 
 106                         printf("%lu\n", size);
 
 112                 return !has_sha1_file(sha1);
 
 115                 type = sha1_object_info(sha1, NULL);
 
 117                         die("Not a valid object name %s", obj_name);
 
 119                 /* custom pretty-print here */
 
 120                 if (type == OBJ_TREE) {
 
 121                         const char *ls_args[3] = {"ls-tree", obj_name, NULL};
 
 122                         return cmd_ls_tree(2, ls_args, NULL);
 
 125                 buf = read_sha1_file(sha1, &type, &size);
 
 127                         die("Cannot read object %s", obj_name);
 
 128                 if (type == OBJ_TAG) {
 
 129                         pprint_tag(sha1, buf, size);
 
 133                 /* otherwise just spit out the data */
 
 136                 buf = read_object_with_reference(sha1, exp_type, &size, NULL);
 
 140                 die("git cat-file: unknown option: %s", exp_type);
 
 144                 die("git cat-file %s: bad file", obj_name);
 
 146         write_or_die(1, buf, size);
 
 150 static int batch_one_object(const char *obj_name, int print_contents)
 
 152         unsigned char sha1[20];
 
 153         enum object_type type = 0;
 
 155         void *contents = contents;
 
 160         if (get_sha1(obj_name, sha1)) {
 
 161                 printf("%s missing\n", obj_name);
 
 166         if (print_contents == BATCH)
 
 167                 contents = read_sha1_file(sha1, &type, &size);
 
 169                 type = sha1_object_info(sha1, &size);
 
 172                 printf("%s missing\n", obj_name);
 
 177         printf("%s %s %lu\n", sha1_to_hex(sha1), typename(type), size);
 
 180         if (print_contents == BATCH) {
 
 181                 write_or_die(1, contents, size);
 
 190 static int batch_objects(int print_contents)
 
 192         struct strbuf buf = STRBUF_INIT;
 
 194         while (strbuf_getline(&buf, stdin, '\n') != EOF) {
 
 195                 int error = batch_one_object(buf.buf, print_contents);
 
 203 static const char * const cat_file_usage[] = {
 
 204         "git cat-file (-t|-s|-e|-p|<type>) <object>",
 
 205         "git cat-file (--batch|--batch-check) < <list_of_objects>",
 
 209 int cmd_cat_file(int argc, const char **argv, const char *prefix)
 
 211         int opt = 0, batch = 0;
 
 212         const char *exp_type = NULL, *obj_name = NULL;
 
 214         const struct option options[] = {
 
 215                 OPT_GROUP("<type> can be one of: blob, tree, commit, tag"),
 
 216                 OPT_SET_INT('t', NULL, &opt, "show object type", 't'),
 
 217                 OPT_SET_INT('s', NULL, &opt, "show object size", 's'),
 
 218                 OPT_SET_INT('e', NULL, &opt,
 
 219                             "exit with zero when there's no error", 'e'),
 
 220                 OPT_SET_INT('p', NULL, &opt, "pretty-print object's content", 'p'),
 
 221                 OPT_SET_INT(0, "batch", &batch,
 
 222                             "show info and content of objects feeded on stdin", BATCH),
 
 223                 OPT_SET_INT(0, "batch-check", &batch,
 
 224                             "show info about objects feeded on stdin",
 
 229         git_config(git_default_config, NULL);
 
 231         if (argc != 3 && argc != 2)
 
 232                 usage_with_options(cat_file_usage, options);
 
 234         argc = parse_options(argc, argv, prefix, options, cat_file_usage, 0);
 
 240                         usage_with_options(cat_file_usage, options);
 
 242         if (!opt && !batch) {
 
 247                         usage_with_options(cat_file_usage, options);
 
 249         if (batch && (opt || argc)) {
 
 250                 usage_with_options(cat_file_usage, options);
 
 254                 return batch_objects(batch);
 
 256         return cat_one_file(opt, exp_type, obj_name);