home *** CD-ROM | disk | FTP | other *** search
/ Usenet 1994 October / usenetsourcesnewsgroupsinfomagicoctober1994disk2.iso / misc / volume10 / weekday < prev    next >
Text File  |  1990-02-26  |  30KB  |  762 lines

  1. Newsgroups: comp.sources.misc
  2. organization: Delft University of Technology, Dep. of Electrotechnical engineering.
  3. subject: v10i084: new version of weekday
  4. From: marcel@duteca.tudelft.nl (Marcel Mol)
  5. Sender: allbery@uunet.UU.NET (Brandon S. Allbery - comp.sources.misc)
  6.  
  7. Posting-number: Volume 10, Issue 84
  8. Submitted-by: marcel@duteca.tudelft.nl (Marcel Mol)
  9. Archive-name: weekday
  10.  
  11. [Shar, please!!!  ++bsa]
  12.  
  13. This version of weekday repairs some bugs and adds the ISO date standard as
  14. date input format. (The diffs were almost as large as the source so I send out
  15. the soure).
  16.  
  17. -Marcel
  18.  
  19. ------------------------- Cut here ---------------------------------------
  20. #! /bin/sh
  21. # This file was wrapped with "dummyshar".  "sh" this file to extract.
  22. # Contents:  weekday.c
  23. echo extracting 'weekday.c'
  24. if test -f 'weekday.c' -a -z "$1"; then echo Not overwriting 'weekday.c'; else
  25. sed 's/^X//' << \EOF > 'weekday.c'
  26. X/* @(#) weekday.c       3.40 02/06/90  */
  27. X/*************************************************************************
  28. X **                                                                     **
  29. X **  Name    :  weekday                                                 **
  30. X **  Author  :  Marcel J.E. Mol                                         **
  31. X **  Date    :  12/28/89              (first release)                   **
  32. X **  Version :  3.40                                                    **
  33. X **  Files   :  weekday.c       Main source file                        **
  34. X **                                                                     **
  35. X **  ------------------------- Revision List -------------------------  **
  36. X **  Ver   Date       Name                   Remarks                    **
  37. X **  3.40  02/06/90   Marcel Mol    Added ISO date standard YYYY-MM-DD  **
  38. X **                              Suggested by mathew smm12@cl.cam.ac.uk **
  39. X **  3.33  02/04/90   Marcel Mol    Fixed bug when entering years in    **
  40. X **                                 0 <= year <= 99 with leading '0'    **
  41. X **  3.32  02/01/90   Marcel Mol    Fixed typos in January February     **
  42. X **                                 October                             **
  43. X **  3.31  01/09/90   Marcel Mol    Removed unnecessary code.           **
  44. X **                                 Added comments.                     **
  45. X **  3.30  01/08/90   Marcel Mol    Improved option handling.           **
  46. X **                                 verbose works on more options.      **
  47. X **                                 use printf field length option to   **
  48. X **                                 get short day and month names.      **
  49. X **  3.26  01/07/90   Marcel Mol    Filled in usage function.           **
  50. X **  3.25  01/04/90   Marcel Mol    Use 'today' when no date is given.  **
  51. X **  3.20  01/03/90   Marcel Mol    Allow bsd `date` as input.          **
  52. X **  3.11  01/03/90   Marcel Mol    Documentation, typos, checkdate().  **
  53. X **                                 Removed redundant checks.           **
  54. X **  3.10  01/02/90   Marcel Mol    Added -s and -v options.            **
  55. X **                                 allow yy.ddd input and implement    **
  56. X **                                 julian date conversion.             **
  57. X **  3.00  12/30/89   Marcel Mol    Set up user interface.              **
  58. X **                                 Allow mm/dd/yy and dd-mm-yy input.  **
  59. X **                                 Options to print different parts.   **
  60. X **  2.10  12/28/89   Marcel Mol    Added Julianday.                    **
  61. X **  2.00  12/27/89   Marcel Mol    Made weekday and weeknr everlasting.**
  62. X **  1.00  12/26/89   Marcel Mol    First release.                      **
  63. X **                                 Set up algorithms weekday weeknr.   **
  64. X **  =================================================================  **
  65. X **                                                                     **
  66. X **  Compile as follows: cc weekday.c -Oso weekday                      **
  67. X **                                                                     **
  68. X **  Usage: weekday [-svdDmMywjnah] [<date>]                            **
  69. X **                                                                     **
  70. X **         Weekday outputs line on which it informs you about several  **
  71. X **         thing of the given <date>.                                  **
  72. X **         The <date> input may be in several formats:                 **
  73. X **                                                                     **
  74. X **             mm/dd/yy      e.g.   10/30/89    or 1/2/510             **
  75. X **             dd-mm-yy      e.g.   30-10-89    or 2-1-510             **
  76. X **             yyyy-mm-dd    e.g. 1989-10-30    or 0510-1-2            **
  77. X **             yy.ddd        e.g.   89.303      or 510.2               **
  78. X **             `date` output e.g.   Mon Oct 30 99:99:99 1989           **
  79. X **                                                                     **
  80. X **         If no date argument is given, weekday uses the system date. **
  81. X **         The yyyy-mm-dd date format must have 4 digits for the yyyy  **
  82. X **         part, 2 for the mm part and 2 for the dd part; it's just    **
  83. X **         the ISO date standard. (undocumented feature: it also works **
  84. X **         if the mm and/or dd part is only one digit long.)           **
  85. X **         So the following trick does not apply to this date format:  **
  86. X **         When the year number is less than 100 and the string        **
  87. X **         forming the year does not start with a 0, the year is       **
  88. X **         converted to be in the current century. So 10/30/89 is      **
  89. X **         actually 10/30/1989. If the real year 89 is wanted specify  **
  90. X **         it as 10/30/089. (the current century is a constant in the  **
  91. X **         source code. If this programs survives 1999 this constant   **
  92. X **         must be changed, and the program recompiled.)               **
  93. X **         Note: make sure that `date` output is passed as one         **
  94. X **         argument, so enclose it with "s.                            **
  95. X **                                                                     **
  96. X **         The output, when no options are given, is something like    **
  97. X **         this:                                                       **
  98. X **                                                                     **
  99. X **         30 October 1989 is on a Monday on julian day 303 in week 44 **
  100. X **                                                                     **
  101. X **         or:                                                         **
  102. X **                                                                     **
  103. X **         2 January 510 is on a Friday on julian day 2 in week 1      **
  104. X **                                                                     **
  105. X **         Depending on the options the output changes:                **
  106. X **                                                                     **
  107. X **           -s toggle short form output: Monday -> Mon, June -> Jun   **
  108. X **              and vice versa.                                        **
  109. X **           -v toggle verbose output; removes 'is on a' etc.          **
  110. X **              and vice versa.                                        **
  111. X **           -d prints day of week number: 1 is Monday, 7 is Sunday    **
  112. X **           -D gives day of week name, i.e Monday or Mon              **
  113. X **           -m prints the month number                                **
  114. X **           -M outputs the month name, i.e. January or Jan            **
  115. X **           -y outputs the year number                                **
  116. X **           -w gives the week number                                  **
  117. X **           -j outputs the julian day number                          **
  118. X **           -n gives day of month                                     **
  119. X **           -a gives everything it knows. Without the -s and -v       **
  120. X **              options this outputs the same as the default case.     **
  121. X **              In fact -a is just short for -nmyDjw.                  ** 
  122. X **           -h give a usage message                                   **
  123. X **                                                                     **
  124. X **         With the verbose option on, each printing option speaks to  **
  125. X **         you in a more or less english way.                          **
  126. X **         All options that print something also print a space         **
  127. X **         character (except -a).                                      **
  128. X **         The options may come in any order and as many times as      **
  129. X **         you like. The -s and -v options only have an effect on the  **
  130. X **         options that follow them.                                   **
  131. X **         When now option is given (except for the -v and -s option)  **
  132. X **         the -a option is default.                                   **
  133. X **         The output is ended with a newline                          **
  134. X **         Examples:                                                   **
  135. X **                                                                     **
  136. X **           % weekday  -yjMdD 12/31/89                                **
  137. X **           89 365 December 7 Sunday                                  **
  138. X **                                                                     **
  139. X **           % weekday  -yjMsdD 12/31/89                               **
  140. X **           89 365 December 7 Sun                                     **
  141. X **                                                                     **
  142. X **           % weekday  -yjsMsdD 12/31/89                              **
  143. X **           89 365 Dec 7 Sunday                                       **
  144. X **                                                                     **
  145. X **           % weekday  -v 30/30/89                                    **
  146. X **           30 Oct 1989 is on a Mon on julian day 303 in week 44      **
  147. X **                                                                     **
  148. X **   BUGS: the program assumes there have always been leap years       **
  149. X **         so year 4 is a leap year as far as the program can tell...  **
  150. X **                                                                     **
  151. X **   This software is copyrighted (c) 1989 1990 by M.J.E. Mol.         **
  152. X **   This code or part of this code may be used by freeware,           **
  153. X **   public domain or any other non-commercial product. I only ask     **
  154. X **   you to give me credit for the used code in the documentation      **
  155. X **   and in your source code.                                          **
  156. X **                                                                     **
  157. X **   If you want to use this code or part of this code in your         **
  158. X **   shareware or any other commercial product, I ask you to contact   **
  159. X **   me so we can setup an arrangement for the use of the code or part **
  160. X **   of the code.                                                      **
  161. X **                                                                     **
  162. X **   In any case, I can not be held responsible for any damage this    **
  163. X **   code or part of this code may cause you or anyone or anything     **
  164. X **   else.                                                             **
  165. X **                                                                     **
  166. X **   Mail bugs to: email: marcel@duteca.tudelft.nl                     **
  167. X **                 pmail: Marcel J.E. Mol                              **
  168. X **                        Karel Doormanlaan 58                         **
  169. X **                        2283AT Rijswijk                              **
  170. X **                        The Netherlands (Holland)                    **
  171. X **                 Phone: (+31) 070-3932390                            **
  172. X **                 Bankaccount: 36.89.35.922 (In the Netherlands)      **
  173. X **                                                                     **
  174. X ************************************************************************/
  175. X
  176. Xchar * copyright = "@(#) weekday.c  3.40 02/06/90  (c) M.J.E. Mol";
  177. X
  178. X#include <stdio.h>
  179. X#include <string.h>
  180. X#include <time.h>
  181. X
  182. X
  183. X#define CENTURY     1900                             /* prefix for years */
  184. X#define LEAP(year)  ((year%4 == 0) && (year%100 != 0) || (year%400 == 0)) 
  185. X
  186. X/*
  187. X * Global variables
  188. X */
  189. X
  190. Xchar *weekdays[8]   = {"", "Monday", "Tuesday", "Wednesday", "Thursday",
  191. X                           "Friday", "Saterday", "Sunday"};
  192. Xchar  days[13]      = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
  193. Xint   juldays[13]   = {0, 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
  194. Xchar *monthname[13] = {"", "January", "February", "March", "April",
  195. X                           "May", "June", "July", "August",
  196. X                           "September", "October", "November", "December"};
  197. X
  198. Xint shortflag = 0;                  /* short output of day and month name */
  199. Xint verbose   = 1;                  /* output in a 'sentence' */
  200. Xchar * progname;
  201. X
  202. X
  203. X/*
  204. X * function declarations
  205. X */
  206. Xvoid main();
  207. Xvoid scanmmddyy();
  208. Xvoid scanyyyymmdd();
  209. Xvoid scanddmmyy();
  210. Xint  scanjul();
  211. Xint  scandate();
  212. Xint  julianday();
  213. Xvoid julian2day();
  214. Xint  weekday();
  215. Xint  weeknr();
  216. Xvoid checkdate();
  217. Xvoid usage();
  218. X
  219. X
  220. Xvoid main(argc, argv)
  221. Xint argc;
  222. Xchar **argv;
  223. X{
  224. X    int year, month, day, week;
  225. X    int wday = -1;
  226. X    int jday = -1;
  227. X    int opt, haveopt;
  228. X    char *date;
  229. X    extern int optind;                  /* For getopt */
  230. X    extern char * optarg;               /* For getopt */
  231. X
  232. X    progname = *argv;
  233. X    date = argv[argc-1];
  234. X
  235. X    /*
  236. X     * pick up the date argument and scan it
  237. X     */
  238. X    if (*date == '-' || argc == 1)                   /* no date, take current */
  239. X        wday = scandate(NULL, &year, &month, &day);
  240. X    else if (strchr(date, '/') != NULL)              /* mm/dd/yy format date */
  241. X        scanmmddyy(date, &year, &month, &day);
  242. X    else if (strchr(date, '-') == date+4)            /* yyyy-mm-dd format    */
  243. X        scanyyyymmdd(date, &year, &month, &day);
  244. X    else if (strchr(date, '-') != NULL)              /* dd-mm-yy format date */
  245. X        scanddmmyy(date, &year, &month, &day);
  246. X    else if (strchr(date, '.') != NULL)              /* yyyy.ddd format date */
  247. X        jday = scanjulian(date, &year, &month, &day);
  248. X    else                                           /* unix date output format */
  249. X        wday = scandate(date, &year, &month, &day);
  250. X
  251. X    /*
  252. X     * get missing data
  253. X     */
  254. X    if (jday == -1)
  255. X        jday = julianday(year, month, day);
  256. X    if (wday == -1)
  257. X        wday = weekday(year, month, day);
  258. X    week = weeknr(year, month, day);
  259. X
  260. X
  261. X    /*
  262. X     * print the wanted stuff ...
  263. X     */
  264. X    while ((opt = getopt(argc, argv, "svdDmMywjnah")) != EOF) {
  265. X        switch (opt) {
  266. X            case 's' : shortflag ^=  1;
  267. X                       break;
  268. X            case 'v' : verbose ^= 1;
  269. X                       break;
  270. X            case 'D' : printf("%s%.*s ",
  271. X                                   verbose ? "is on a " : "",
  272. X                                   shortflag ? 3 : 9 , weekdays[wday]);
  273. X                       haveopt = 1;
  274. X                       break;
  275. X            case 'd' : printf("%s%d ", verbose ? "is weekday " : "", wday);
  276. X                       haveopt = 1;
  277. X                       break;
  278. X            case 'm' : printf("%d ", month);
  279. X                       haveopt = 1;
  280. X                       break;
  281. X            case 'M' : printf("%.*s ", shortflag ? 3 : 9, monthname[month]);
  282. X                       haveopt = 1;
  283. X                       break;
  284. X            case 'y' : printf("%d ", year);
  285. X                       haveopt = 1;
  286. X                       break;
  287. X            case 'w' : printf("%s%d ", verbose ? "in week " : "", week);
  288. X                       haveopt = 1;
  289. X                       break;
  290. X            case 'j' : printf("%s%d ", verbose ? "on julian day " : "", jday);
  291. X                       haveopt = 1;
  292. X                       break;
  293. X            case 'n' : printf("%d ", day);
  294. X                       haveopt = 1;
  295. X                       break;
  296. X            case 'a' : if (verbose) {
  297. X                           printf("%d %.*s %d is on a %.*s ",
  298. X                                  day,  shortflag ? 3 : 9, monthname[month],
  299. X                                  year, shortflag ? 3 : 9, weekdays[wday]);
  300. X                           printf("on julian day %d in week %d", jday, week);
  301. X                       }
  302. X                       else
  303. X                           printf("%d %.*s %d %.*s %d %d",
  304. X                                  day,  shortflag ? 3 : 9, monthname[month],
  305. X                                  year, shortflag ? 3 : 9, weekdays[wday],
  306. X                                  jday, week);
  307. X                       haveopt = 1;
  308. X                       break;
  309. X            case 'h' : usage();
  310. X                       break;
  311. X            default  : fprintf(stderr,
  312. X                               "%s: use -h option for help\n", progname);
  313. X                       exit(1);
  314. X                       break;
  315. X        }
  316. X    }
  317. X
  318. X
  319. X    if (!haveopt) {
  320. X        if (verbose) {
  321. X            printf("%d %.*s %d is on a %.*s ",
  322. X                   day,  shortflag ? 3 : 9, monthname[month],
  323. X                   year, shortflag ? 3 : 9, weekdays[wday]);
  324. X            printf("on julian day %d in week %d", jday, week);
  325. X        }
  326. X        else
  327. X            printf("%d %.*s %d %.*s %d %d",
  328. X                   day,  shortflag ? 3 : 9, monthname[month],
  329. X                   year, shortflag ? 3 : 9, weekdays[wday],
  330. X                   jday, week);
  331. X    }
  332. X
  333. X    /*
  334. X     * finish with a newline
  335. X     */
  336. X    putchar('\n');
  337. X
  338. X    exit(0);
  339. X
  340. X} /* main */
  341. X /*************************************************************************
  342. X **                                                                     **
  343. X **                    DATE SCANNING ROUTINES                           **
  344. X **                                                                     **
  345. X *************************************************************************/
  346. X
  347. X/*
  348. X * scanmmddyy --- scan the date variable in the form MM/DD/YY.
  349. X *                MM, DD and YY may be variable sized numbers.
  350. X *                output is in output variables year, month and day.
  351. X */
  352. Xvoid scanmmddyy(date, year, month, day)
  353. Xchar *date;
  354. Xint *year, *month, *day;
  355. X{
  356. X    char *i;
  357. X
  358. X    if (sscanf(date, "%d/%d/%d", month, day, year) != 3 ||
  359. X        *month <= 0 || *day <= 0 || *year < 0) {
  360. X        fprintf(stderr, "%s: error in mm/dd/yy format date %s\n",
  361. X                        progname, date);
  362. X        exit(4);
  363. X    }
  364. X
  365. X    if (*year < 100) {
  366. X        i = strrchr(date, '/') + 1;
  367. X        if (*i != '0')
  368. X           *year += CENTURY;
  369. X    }
  370. X
  371. X    return;
  372. X
  373. X} /* scanmmddyy */
  374. X
  375. X
  376. X
  377. X/*
  378. X * scanyyyymmdd --- scan the date variable in the form YYYY-MM-DD.
  379. X *                YYYY, MM and DD must be 4, 2 and 2 digits wide.
  380. X *                output is in output variables year, month and day.
  381. X */
  382. Xvoid scanyyyymmdd(date, year, month, day)
  383. Xchar *date;
  384. Xint *year, *month, *day;
  385. X{
  386. X
  387. X    if (sscanf(date, "%4d-%2d-%2d", year, month, day) != 3 ||
  388. X        *month <= 0 || *day <= 0 || *year < 0) {
  389. X        fprintf(stderr, "%s: error in yyyy-mm-dd format date %s\n",
  390. X                        progname, date);
  391. X        exit(4);
  392. X    }
  393. X
  394. X    return;
  395. X
  396. X} /* scanyyyymmdd */
  397. X
  398. X
  399. X/*
  400. X * scanddmmyy --- scan the date variable in the form DD-MM-YY.
  401. X *                Also YYYY-MM-DD is recognized if YYYY > 31 and DD <= 31.
  402. X *                DD, MM and YY may be variable sized numbers.
  403. X *                output is in output variables year, month and day.
  404. X */
  405. Xvoid scanddmmyy(date, year, month, day)
  406. Xchar *date;
  407. Xint *year, *month, *day;
  408. X{
  409. X    char *i;
  410. X
  411. X    if (sscanf(date, "%d-%d-%d", day, month, year) != 3 ||
  412. X        *month <= 0 || *day <= 0 || *year < 0) {
  413. X        fprintf(stderr, "%s: error in dd-mm-yy format date %s\n",
  414. X                        progname, date);
  415. X        exit(4);
  416. X    }
  417. X
  418. X    if (*year < 100) {
  419. X        i = strrchr(date, '-') + 1;
  420. X        if (*i != '0')
  421. X           *year += CENTURY;
  422. X    }
  423. X
  424. X    return;
  425. X
  426. X} /* scanddmmyy */
  427. X
  428. X
  429. X
  430. X/*
  431. X * scanjulian --- scan the date variable in the form YY.DDD.
  432. X *                YY and DDD may be variable sized numbers.
  433. X *                output is in output variables year, month and day.
  434. X *                the return value is tha actual julian day DDD.
  435. X */
  436. Xint scanjulian(date, year, month, day)
  437. Xchar *date;
  438. Xint *year, *month, *day;
  439. X{
  440. X    int jday;
  441. X
  442. X    if (sscanf(date, "%d.%d", year, &jday) != 2 || *year < 0 || jday <= 0) {
  443. X        fprintf(stderr, "%s: error in julian date %s\n", progname, date);
  444. X        exit(4);
  445. X    }
  446. X
  447. X    if ((*year < 100) && (*date != '0'))
  448. X        *year += CENTURY;
  449. X    julian2day(*year, jday, month, day);
  450. X
  451. X    return jday;
  452. X
  453. X} /* scanjulian */
  454. X
  455. X
  456. X
  457. X/*
  458. X * scandate --- scan the date variable in the Unix `date` form.
  459. X *              This can look like: "Mon Jan 8 20:10:10 MET 1990"
  460. X *              or:                 "Mon Jan 8 20:10:10 1990"
  461. X *              the number of spaces between the fields is not fixed
  462. X *              (although date output is always in  the same with, see ctime(3).
  463. X *              output is in output variables year, month and day.
  464. X *              the return value is the day of the week number.
  465. X *              If the date input is the NULL pointer, the xurrent system date
  466. X *              is taken.
  467. X */
  468. Xint scandate(date, year, month, day)
  469. Xchar *date;
  470. Xint *year, *month, *day;
  471. X{
  472. X    char *i;
  473. X    char dayname[4];
  474. X    char monname[10];
  475. X    int j, k;
  476. X    long tim;
  477. X
  478. X    if (date == NULL) {
  479. X        tim = time((long *) 0);
  480. X        date= ctime(&tim);
  481. X    }
  482. X
  483. X    if (sscanf(date, "%3s %3s %d %*s %d",      /* date without time zone */
  484. X                     dayname, monname, day, year) != 4 &&
  485. X        sscanf(date, "%3s %3s %d %*s %*s %d", /* date with time zone */
  486. X                     dayname, monname, day, year) != 4) {
  487. X        fprintf(stderr, "%s: error in unix `date` format date %s\n",
  488. X                        progname, date);
  489. X        exit(4);
  490. X    }
  491. X
  492. X    for(j=1; j<12; j++) {
  493. X        if (strncmp(monname, monthname[j], 3) == NULL)
  494. X            break;
  495. X    }
  496. X    if (strncmp(monname, monthname[j], 3) != NULL) {
  497. X        fprintf(stderr, "%s: error in unix `date` format date %s\n",
  498. X                        progname, date);
  499. X        exit(4);
  500. X    }
  501. X    *month = j;
  502. X    for(j=1; j<7; j++) {
  503. X        if (strncmp(dayname, weekdays[j], 3) == NULL)
  504. X            break;
  505. X    }
  506. X    if (strncmp(dayname, weekdays[j], 3) != NULL) {
  507. X        fprintf(stderr, "%s: error in unix `date` format date %s\n",
  508. X                        progname, date);
  509. X        exit(4);
  510. X    }
  511. X
  512. X    return j;
  513. X
  514. X} /* scandate */
  515. X /*************************************************************************
  516. X **                                                                     **
  517. X **                    DATE CONVERSION ROUTINES                         **
  518. X **                                                                     **
  519. X *************************************************************************/
  520. X
  521. X
  522. X
  523. X/* 
  524. X * julian2day -- converts year and julian day value to the corresponding 
  525. X *               month and day of the month values for that year.
  526. X *               these values are returned in the ouput variables month and day.
  527. X */
  528. Xvoid julian2day(year, jday, month, day)
  529. Xint year, jday;
  530. Xint *month, *day;
  531. X{
  532. X    int leap;
  533. X
  534. X    leap = LEAP(year) ? 1 : 0;
  535. X    if (jday > 365+leap || jday <= 0) {
  536. X        fprintf(stderr, "%s: error in julian date %d.%d\n",
  537. X                        progname, year, jday);
  538. X       exit(8);
  539. X    }
  540. X
  541. X    if (jday <= 31) {
  542. X        *month = 1;
  543. X        *day = jday;
  544. X        return;
  545. X    }
  546. X
  547. X    jday -= leap;
  548. X    if (jday <= 59) {
  549. X        *month = 2;
  550. X        *day = jday - 31 + leap;
  551. X    }
  552. X    else if (jday <= 90) {
  553. X        *month = 3;
  554. X        *day = jday - 59;
  555. X    }
  556. X    else if (jday <= 120) {
  557. X        *month = 4;
  558. X        *day = jday - 90;
  559. X    }
  560. X    else if (jday <= 151) {
  561. X        *month = 5;
  562. X        *day = jday - 120;
  563. X    }
  564. X    else if (jday <= 181) {
  565. X        *month = 6;
  566. X        *day = jday - 151;
  567. X    }
  568. X    else if (jday <= 212) {
  569. X        *month = 7;
  570. X        *day = jday - 181;
  571. X    }
  572. X    else if (jday <= 243) {
  573. X        *month = 8;
  574. X        *day = jday - 212;
  575. X    }
  576. X    else if (jday <= 273) {
  577. X        *month = 9;
  578. X        *day = jday - 243;
  579. X    }
  580. X    else if (jday <= 304) {
  581. X        *month = 10;
  582. X        *day = jday - 273;
  583. X    }
  584. X    else if (jday <= 334) {
  585. X        *month = 11;
  586. X        *day = jday - 304;
  587. X    }
  588. X    else {
  589. X        *month = 12;
  590. X        *day = jday - 334;
  591. X    }
  592. X
  593. X    return;
  594. X
  595. X} /* julian2day */
  596. X
  597. X
  598. X
  599. X/* 
  600. X * julianday -- converts a year, mont and day value the corresponding julian
  601. X *              day number of that year.
  602. X *              the julian day is passed as a return value.
  603. X */
  604. Xjulianday(year, month, day)
  605. Xint year, month, day;
  606. X{
  607. X    register int doy;
  608. X
  609. X    checkdate(year, month, day);           /* exits if date in error */
  610. X    doy = juldays[month];
  611. X    if ((month > 2) && LEAP(year)) 
  612. X        doy++;
  613. X    doy += day;
  614. X
  615. X    return doy;
  616. X
  617. X} /* julianday */
  618. X
  619. X
  620. X
  621. X/*
  622. X * weekday -- returns the day of the week number based on the given
  623. X *            year, month and day of the month value.
  624. X */
  625. Xweekday(year, month, day)
  626. Xint year, month, day;
  627. X{
  628. X    register int dow;
  629. X
  630. X    checkdate(year, month, day);           /* exits if date in error */
  631. X    dow = 6 +                             /* the magic number */
  632. X          year +                          /* shift a day each year */
  633. X          year/4 - year/100 + year/400 +  /* shift a day each leap year */
  634. X          juldays[month] + day;           /* day of year */
  635. X
  636. X    if ((month < 3) && LEAP(year))        /* first two months of leap don't */
  637. X        dow--;                            /* have the extra leap shift      */
  638. X
  639. X    dow %= 7;
  640. X    return (dow == 0) ? 7 : dow;          /* convert 0 to 7 */
  641. X
  642. X} /* weekday */
  643. X
  644. X
  645. X/*
  646. X * weeknr -- returns the weeknumber of the given date (year, month and day of
  647. X *           the month) for that year.
  648. X */
  649. Xweeknr(year, month, day)
  650. Xint year, month, day;
  651. X{
  652. X    register int fday, jul, weeknum;
  653. X
  654. X    checkdate(year, month, day);           /* exits if date in error */
  655. X
  656. X    fday = weekday(year, 1, 1);
  657. X
  658. X    jul = julianday(year, month, day);
  659. X    weeknum = (jul + fday - 2)/7;
  660. X    if (fday <= 4)
  661. X        weeknum++;
  662. X
  663. X    if (weeknum == 0)                      /* go into previous year           */
  664. X        if ((fday == 5) ||                 /* week 1 has 4 or 5 days and week */
  665. X           ((fday == 6) && LEAP(year-1)) ) /*    53 has 4 or 5 days in year-1 */
  666. X            return 53;
  667. X        else
  668. X            return 52;
  669. X
  670. X    if (weeknum == 53) {                  /* only a week 53 if we have > 4   */
  671. X        fday = weekday(year, 12, 31);     /*     days in the week            */
  672. X        if (fday < 4)
  673. X           weeknum = 1;
  674. X    }
  675. X
  676. X    return weeknum;
  677. X
  678. X} /* weeknr */
  679. X /*************************************************************************
  680. X **                                                                     **
  681. X **                         OTHER  ROUTINES                             **
  682. X **                                                                     **
  683. X *************************************************************************/
  684. X
  685. X
  686. X
  687. X/*
  688. X * checkdate -- checks if the given date is a valid one.
  689. X *              if not, it just exits.
  690. X */
  691. Xvoid checkdate(year, month, day)
  692. Xint year, month, day;
  693. X{
  694. X    int dim;
  695. X
  696. X    if (year < 0) {
  697. X        fprintf(stderr, "%s: %d is an invalid year\n", progname, year);
  698. X        exit(2);
  699. X    }
  700. X    if ((month < 1) || (month > 12)) {
  701. X        fprintf(stderr, "%s: %d is an invalid month\n", progname, month);
  702. X        exit(2);
  703. X    }
  704. X    dim = days[month];
  705. X    if ((month == 2) && LEAP(year)) 
  706. X        dim++;
  707. X    if ((day > dim) || (day < 1)) {
  708. X        fprintf(stderr, "%s: %d is an invalid day in %s %d\n", progname,
  709. X                         day, monthname[month], year);
  710. X        exit(2);
  711. X    }
  712. X
  713. X   return;
  714. X
  715. X} /* checkdate */
  716. X
  717. X
  718. X
  719. X/*
  720. X * usage -- give usage message.
  721. X */
  722. Xvoid usage()
  723. X{
  724. X
  725. X    fprintf(stderr,"Usage: weekday [-svdDmMywjnah] [<date>]\n"); 
  726. X    fprintf(stderr,"   <date>: mm/dd/yy  |  dd-mm-yy  |  yyyy-mm-dd\n  |");
  727. X    fprintf(stderr,"           yy.ddd  |  Unix date\n");
  728. X    fprintf(stderr,"   default is today\n");
  729. X    fprintf(stderr,"   options: -s toggle short form output\n");
  730. X    fprintf(stderr,"            -v toggle verbose output\n");
  731. X    fprintf(stderr,"            -d prints day of week number\n");
  732. X    fprintf(stderr,"            -D gives day of week name\n");
  733. X    fprintf(stderr,"            -m prints the month number\n");
  734. X    fprintf(stderr,"            -M outputs the month name\n");
  735. X    fprintf(stderr,"            -y outputs the year number\n");
  736. X    fprintf(stderr,"            -w gives the week number\n");
  737. X    fprintf(stderr,"            -j outputs the julian day number\n");
  738. X    fprintf(stderr,"            -n gives day of month\n");
  739. X    fprintf(stderr,"            -a gives everything it knows (default)\n");
  740. X    fprintf(stderr,"            -h give a usage message\n");
  741. X
  742. X    return;
  743. X
  744. X} /* usage */
  745. EOF
  746. chars=`wc -c < 'weekday.c'`
  747. if test $chars !=    27998; then echo 'weekday.c' is $chars characters, should be    27998 characters!; fi
  748. fi
  749. exit 0
  750. -- 
  751. #########################################
  752. # Marcel J.E. Mol                       ######################################
  753. # Delft University of Technology          Pink Elephant Management Services  #
  754. # The Netherlands                         Voorburg                           #
  755. # UUCP: marcel@duteca.tudelft.nl          Tel: 070-694231                    #
  756. #                                          ######################################
  757. #########################################
  758.