1 /* $OpenBSD: status.c,v 1.96 2015/04/04 14:20:11 stsp 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 struct rcs_delta *rdp; 125 char buf[PATH_MAX + CVS_REV_BUFSZ + 128]; 126 char timebuf[CVS_TIME_BUFSZ], revbuf[CVS_REV_BUFSZ]; 127 struct rcs_sym *sym; 128 129 cvs_log(LP_TRACE, "cvs_status_local(%s)", cf->file_path); 130 131 cvs_file_classify(cf, cvs_directory_tag); 132 133 if (cf->file_type == CVS_DIR) { 134 if (verbosity > 1) 135 cvs_log(LP_NOTICE, "Examining %s", cf->file_path); 136 return; 137 } 138 139 if (cf->file_status == FILE_UPTODATE && 140 !(cf->file_flags & FILE_ON_DISK) && 141 !(cf->file_flags & FILE_USER_SUPPLIED)) 142 return; 143 144 if (cf->file_rcs != NULL) 145 head = cf->file_rcsrev; 146 else 147 head = NULL; 148 149 cvs_printf("%s\n", CVS_STATUS_SEP); 150 151 if (cf->file_rcs != NULL && head == NULL) 152 status = status_tab[FILE_UNKNOWN]; 153 else 154 status = status_tab[cf->file_status]; 155 156 if (cf->file_status == FILE_MODIFIED && 157 cf->file_ent->ce_conflict != NULL) 158 status = "File had conflicts on merge"; 159 160 if (!(cf->file_flags & FILE_ON_DISK)) { 161 (void)xsnprintf(buf, sizeof(buf), "no file %s\t", 162 cf->file_name); 163 } else 164 if (strlcpy(buf, cf->file_name, sizeof(buf)) >= sizeof(buf)) 165 fatal("cvs_status_local: overflow"); 166 167 cvs_printf("File: %-17s\tStatus: %s\n\n", buf, status); 168 169 if (cf->file_ent == NULL) { 170 (void)xsnprintf(buf, sizeof(buf), 171 "No entry for %s", cf->file_name); 172 } else if (cf->file_ent->ce_status == CVS_ENT_ADDED) { 173 len = strlcpy(buf, "New file!", sizeof(buf)); 174 if (len >= sizeof(buf)) 175 fatal("cvs_status_local: truncation"); 176 } else { 177 rcsnum_tostr(cf->file_ent->ce_rev, revbuf, sizeof(revbuf)); 178 179 if (cf->file_ent->ce_conflict == NULL) { 180 (void)strlcpy(timebuf, cf->file_ent->ce_time, 181 sizeof(timebuf)); 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_rcs != NULL && head != NULL) { 216 rdp = rcs_findrev(cf->file_rcs, head); 217 if (rdp == NULL) { 218 fatal("cvs_status_local: No head revision delta"); 219 } 220 221 cvs_printf(" Commit Identifier:\t%s\n", 222 (rdp->rd_commitid != NULL) ? rdp->rd_commitid : "(none)"); 223 } 224 225 if (cf->file_ent != NULL) { 226 if (cf->file_ent->ce_tag != NULL) 227 cvs_printf(" Sticky Tag:\t\t%s\n", 228 cf->file_ent->ce_tag); 229 else if (verbosity > 0) 230 cvs_printf(" Sticky Tag:\t\t(none)\n"); 231 232 if (cf->file_ent->ce_date != -1) { 233 struct tm datetm; 234 char datetmp[CVS_TIME_BUFSZ]; 235 236 gmtime_r(&(cf->file_ent->ce_date), &datetm); 237 (void)strftime(datetmp, sizeof(datetmp), 238 CVS_DATE_FMT, &datetm); 239 240 cvs_printf(" Sticky Date:\t\t%s\n", datetmp); 241 } else if (verbosity > 0) 242 cvs_printf(" Sticky Date:\t\t(none)\n"); 243 244 if (cf->file_ent->ce_opts != NULL) 245 cvs_printf(" Sticky Options:\t%s\n", 246 cf->file_ent->ce_opts); 247 else if (verbosity > 0) 248 cvs_printf(" Sticky Options:\t(none)\n"); 249 } 250 251 if (cf->file_rcs != NULL && show_sym == 1) { 252 cvs_printf("\n"); 253 cvs_printf(" Existing Tags:\n"); 254 255 if (!TAILQ_EMPTY(&(cf->file_rcs->rf_symbols))) { 256 TAILQ_FOREACH(sym, 257 &(cf->file_rcs->rf_symbols), rs_list) { 258 (void)rcsnum_tostr(sym->rs_num, revbuf, 259 sizeof(revbuf)); 260 261 cvs_printf("\t%-25s\t(%s: %s)\n", sym->rs_name, 262 RCSNUM_ISBRANCH(sym->rs_num) ? "branch" : 263 "revision", revbuf); 264 } 265 } else 266 cvs_printf("\tNo Tags Exist\n"); 267 } 268 269 cvs_printf("\n"); 270 } 271