1 /* $OpenBSD: status.c,v 1.93 2010/04/19 13:03:10 millert Exp $ */ 2 /* 3 * Copyright (c) 2006 Joris Vink <joris@openbsd.org> 4 * Copyright (c) 2005-2008 Xavier Santolaria <xsa@openbsd.org> 5 * 6 * Permission to use, copy, modify, and distribute this software for any 7 * purpose with or without fee is hereby granted, provided that the above 8 * copyright notice and this permission notice appear in all copies. 9 * 10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 17 */ 18 19 #include <string.h> 20 #include <time.h> 21 #include <unistd.h> 22 23 #include "cvs.h" 24 #include "remote.h" 25 26 void cvs_status_local(struct cvs_file *); 27 28 static int show_sym = 0; 29 30 struct cvs_cmd cvs_cmd_status = { 31 CVS_OP_STATUS, CVS_USE_WDIR, "status", 32 { "st", "stat" }, 33 "Display status information on checked out files", 34 "[-lRv]", 35 "lRv", 36 NULL, 37 cvs_status 38 }; 39 40 #define CVS_STATUS_SEP \ 41 "===================================================================" 42 43 const char *status_tab[] = { 44 "Unknown", 45 "Locally Added", 46 "Locally Removed", 47 "Locally Modified", 48 "Up-to-date", 49 "Needs Checkout", 50 "Needs Checkout", 51 "Needs Merge", 52 "Needs Patch", 53 "Entry Invalid", 54 "Unresolved Conflict", 55 "Classifying error", 56 }; 57 58 int 59 cvs_status(int argc, char **argv) 60 { 61 int ch, flags; 62 char *arg = "."; 63 struct cvs_recursion cr; 64 65 flags = CR_RECURSE_DIRS; 66 67 while ((ch = getopt(argc, argv, cvs_cmd_status.cmd_opts)) != -1) { 68 switch (ch) { 69 case 'l': 70 flags &= ~CR_RECURSE_DIRS; 71 break; 72 case 'R': 73 flags |= CR_RECURSE_DIRS; 74 break; 75 case 'v': 76 show_sym = 1; 77 break; 78 default: 79 fatal("%s", cvs_cmd_status.cmd_synopsis); 80 } 81 } 82 83 argc -= optind; 84 argv += optind; 85 86 cr.enterdir = NULL; 87 cr.leavedir = NULL; 88 89 if (current_cvsroot->cr_method == CVS_METHOD_LOCAL) { 90 flags |= CR_REPO; 91 cr.fileproc = cvs_status_local; 92 } else { 93 cvs_client_connect_to_server(); 94 if (!(flags & CR_RECURSE_DIRS)) 95 cvs_client_send_request("Argument -l"); 96 if (show_sym) 97 cvs_client_send_request("Argument -v"); 98 cr.fileproc = cvs_client_sendfile; 99 } 100 101 cr.flags = flags; 102 103 if (argc > 0) 104 cvs_file_run(argc, argv, &cr); 105 else 106 cvs_file_run(1, &arg, &cr); 107 108 if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { 109 cvs_client_send_files(argv, argc); 110 cvs_client_senddir("."); 111 cvs_client_send_request("status"); 112 cvs_client_get_responses(); 113 } 114 115 return (0); 116 } 117 118 void 119 cvs_status_local(struct cvs_file *cf) 120 { 121 size_t len; 122 RCSNUM *head; 123 const char *status; 124 char buf[MAXPATHLEN + CVS_REV_BUFSZ + 128]; 125 char timebuf[CVS_TIME_BUFSZ], revbuf[CVS_REV_BUFSZ]; 126 struct rcs_sym *sym; 127 128 cvs_log(LP_TRACE, "cvs_status_local(%s)", cf->file_path); 129 130 cvs_file_classify(cf, cvs_directory_tag); 131 132 if (cf->file_type == CVS_DIR) { 133 if (verbosity > 1) 134 cvs_log(LP_NOTICE, "Examining %s", cf->file_path); 135 return; 136 } 137 138 if (cf->file_status == FILE_UPTODATE && 139 !(cf->file_flags & FILE_ON_DISK) && 140 !(cf->file_flags & FILE_USER_SUPPLIED)) 141 return; 142 143 if (cf->file_rcs != NULL) 144 head = cf->file_rcsrev; 145 else 146 head = NULL; 147 148 cvs_printf("%s\n", CVS_STATUS_SEP); 149 150 if (cf->file_rcs != NULL && head == NULL) 151 status = status_tab[FILE_UNKNOWN]; 152 else 153 status = status_tab[cf->file_status]; 154 155 if (cf->file_status == FILE_MODIFIED && 156 cf->file_ent->ce_conflict != NULL) 157 status = "File had conflicts on merge"; 158 159 if (!(cf->file_flags & FILE_ON_DISK)) { 160 (void)xsnprintf(buf, sizeof(buf), "no file %s\t", 161 cf->file_name); 162 } else 163 if (strlcpy(buf, cf->file_name, sizeof(buf)) >= sizeof(buf)) 164 fatal("cvs_status_local: overflow"); 165 166 cvs_printf("File: %-17s\tStatus: %s\n\n", buf, status); 167 168 if (cf->file_ent == NULL) { 169 (void)xsnprintf(buf, sizeof(buf), 170 "No entry for %s", cf->file_name); 171 } else if (cf->file_ent->ce_status == CVS_ENT_ADDED) { 172 len = strlcpy(buf, "New file!", sizeof(buf)); 173 if (len >= sizeof(buf)) 174 fatal("cvs_status_local: truncation"); 175 } else { 176 rcsnum_tostr(cf->file_ent->ce_rev, revbuf, sizeof(revbuf)); 177 178 if (cf->file_ent->ce_conflict == NULL) { 179 ctime_r(&(cf->file_ent->ce_mtime), timebuf); 180 if (timebuf[strlen(timebuf) - 1] == '\n') 181 timebuf[strlen(timebuf) - 1] = '\0'; 182 } else { 183 len = strlcpy(timebuf, cf->file_ent->ce_conflict, 184 sizeof(timebuf)); 185 if (len >= sizeof(timebuf)) 186 fatal("cvs_status_local: truncation"); 187 } 188 189 (void)strlcpy(buf, revbuf, sizeof(buf)); 190 if (cvs_server_active == 0) { 191 (void)strlcat(buf, "\t", sizeof(buf)); 192 (void)strlcat(buf, timebuf, sizeof(buf)); 193 } 194 } 195 196 cvs_printf(" Working revision:\t%s\n", buf); 197 198 buf[0] = '\0'; 199 if (cf->file_rcs == NULL) { 200 len = strlcat(buf, "No revision control file", sizeof(buf)); 201 if (len >= sizeof(buf)) 202 fatal("cvs_status_local: truncation"); 203 } else if (head == NULL) { 204 len = strlcat(buf, "No head revision", sizeof(buf)); 205 if (len >= sizeof(buf)) 206 fatal("cvs_status_local: truncation"); 207 } else { 208 rcsnum_tostr(head, revbuf, sizeof(revbuf)); 209 (void)xsnprintf(buf, sizeof(buf), "%s\t%s", revbuf, 210 cf->file_rpath); 211 } 212 213 cvs_printf(" Repository revision:\t%s\n", buf); 214 215 if (cf->file_ent != NULL) { 216 if (cf->file_ent->ce_tag != NULL) 217 cvs_printf(" Sticky Tag:\t\t%s\n", 218 cf->file_ent->ce_tag); 219 else if (verbosity > 0) 220 cvs_printf(" Sticky Tag:\t\t(none)\n"); 221 222 if (cf->file_ent->ce_date != -1) { 223 struct tm datetm; 224 char datetmp[CVS_TIME_BUFSZ]; 225 226 gmtime_r(&(cf->file_ent->ce_date), &datetm); 227 (void)strftime(datetmp, sizeof(datetmp), 228 CVS_DATE_FMT, &datetm); 229 230 cvs_printf(" Sticky Date:\t\t%s\n", datetmp); 231 } else if (verbosity > 0) 232 cvs_printf(" Sticky Date:\t\t(none)\n"); 233 234 if (cf->file_ent->ce_opts != NULL) 235 cvs_printf(" Sticky Options:\t%s\n", 236 cf->file_ent->ce_opts); 237 else if (verbosity > 0) 238 cvs_printf(" Sticky Options:\t(none)\n"); 239 } 240 241 if (cf->file_rcs != NULL && show_sym == 1) { 242 cvs_printf("\n"); 243 cvs_printf(" Existing Tags:\n"); 244 245 if (!TAILQ_EMPTY(&(cf->file_rcs->rf_symbols))) { 246 TAILQ_FOREACH(sym, 247 &(cf->file_rcs->rf_symbols), rs_list) { 248 (void)rcsnum_tostr(sym->rs_num, revbuf, 249 sizeof(revbuf)); 250 251 cvs_printf("\t%-25s\t(%s: %s)\n", sym->rs_name, 252 RCSNUM_ISBRANCH(sym->rs_num) ? "branch" : 253 "revision", revbuf); 254 } 255 } else 256 cvs_printf("\tNo Tags Exist\n"); 257 } 258 259 cvs_printf("\n"); 260 } 261