[news.software.anu-news] arbitron V1.1 Part 1 of 2

glassmann@ccavax.camb.com (01/13/90)

$! ------------------ CUT HERE -----------------------
$!
$! This archive created by VMS_SHARE Version 7.1-001  26-JUN-1989
$!   On 13-JAN-1990 09:29:26.09   By user GLASSMANN 
$!
$! This VMS_SHARE Written by:
$!    Andy Harper, Kings College London UK
$!
$! Acknowledgements to:
$!    James Gray       - Original VMS_SHARE
$!    Michael Bednarek - Original Concept and implementation
$!
$!+ THIS PACKAGE DISTRIBUTED IN 2 PARTS, TO KEEP EACH PART
$!  BELOW 30 BLOCKS
$!
$! TO UNPACK THIS SHARE FILE, CONCATENATE ALL PARTS IN ORDER
$! AND EXECUTE AS A COMMAND PROCEDURE  (  @name  )
$!
$! THE FOLLOWING FILE(S) WILL BE CREATED AFTER UNPACKING:
$!       1. AAAREADME.TXT;1
$!       2. ARBBUILD.COM;2
$!       3. ARBITRON.C;22
$!       4. ARBITRON.COM;1
$!       5. ARBTEST.COM;2
$!       6. NEWSRC.PATCH;1
$!       7. UAF.H;1
$!
$f=f$parse("SHARE_TEMP","SYS$SCRATCH:.TMP_"+f$getjpi("","PID"))
$e="write sys$error  ""%UNPACK"", "
$w="write sys$output ""%UNPACK"", "
$ if f$trnlnm("SHARE_LOG") then $ w = "!"
$ if f$getsyi("version") .ges. "4.4" then $ goto START
$ e "-E-OLDVER, Must run at least VMS 4.4"
$ exit 44
$UNPACK: SUBROUTINE ! P1=filename, P2=checksum
$ if f$search(P1) .eqs. "" then $ goto file_absent
$ e "-W-EXISTS, File ''P1' exists. Skipped."
$ delete/nolog 'f'*
$ exit
$file_absent:
$ if f$parse(P1) .nes. "" then $ goto dirok
$ dn=f$parse(P1,,,"DIRECTORY")
$ w "-I-CREDIR, Creating directory ''dn'."
$ create/dir 'dn'
$ if $status then $ goto dirok
$ e "-E-CREDIRFAIL, Unable to create ''dn'. File skipped."
$ delete/nolog 'f'*
$ exit
$dirok:
$ w "-I-PROCESS, Processing file ''P1'."
$ define/user sys$output nl:
$ EDIT/TPU/NOSEC/NODIS/COM=SYS$INPUT 'f'/OUT='P1'
PROCEDURE Unpacker ON_ERROR ENDON_ERROR;SET(FACILITY_NAME,"UNPACK");SET(
SUCCESS,OFF);SET(INFORMATIONAL,OFF);f:=GET_INFO(COMMAND_LINE,"file_name");
buff:=CREATE_BUFFER(f,f);p:=SPAN(" ")@r&LINE_END;POSITION(BEGINNING_OF(buff))
;LOOP EXITIF SEARCH(p,FORWARD)=0;POSITION(r);ERASE(r);ENDLOOP;POSITION(
BEGINNING_OF(buff));g:=0;LOOP EXITIF MARK(NONE)=END_OF(buff);x:=
ERASE_CHARACTER(1);IF g = 0 THEN IF x="X" THEN MOVE_VERTICAL(1);ENDIF;IF x=
"V" THEN APPEND_LINE;MOVE_HORIZONTAL(-CURRENT_OFFSET);MOVE_VERTICAL(1);ENDIF;
IF x="+" THEN g:=1;ERASE_LINE;ENDIF;ELSE IF x="-" THEN g:=0;ENDIF;ERASE_LINE;
ENDIF;ENDLOOP;p:="`";POSITION(BEGINNING_OF(buff));LOOP r:=SEARCH(p,FORWARD);
EXITIF r=0;POSITION(r);ERASE(r);COPY_TEXT(ASCII(INT(ERASE_CHARACTER(3))));
ENDLOOP;o:=GET_INFO(COMMAND_LINE,"output_file");WRITE_FILE(buff,o);
ENDPROCEDURE;Unpacker;EXIT;
$ delete/nolog 'f'*
$ CHECKSUM 'P1'
$ IF CHECKSUM$CHECKSUM .eqs. P2 THEN $ EXIT
$ e "-E-CHKSMFAIL, Checksum of ''P1' failed."
$ ENDSUBROUTINE
$START:
$ create/nolog 'f'
XARBITRON - Create statistics to be sent monthly to netsurvey@decwrl.
X
XThis makes heavy use of subroutines that are part of NEWS.  Because of
Xthe intertwining of these subroutines, the EXE winds up being over 600
Xblocks.  This means that I can't reasonably post the EXE to the net.
XUnfortunately, this means that arbitron is only available to sites with
XC compilers.
X
XTo build arbitron, you first have to include the file NEWSRC.PATCH into
XNEWSRC.C, and recompile it.  Then compile and link arbitron.  ARBBUILD.COM
Xis included for this purpose, but you'll probably have to modify it for
Xyour site.
X
XARBITRON.COM should be run monthly.  This will also have to be modified
Xfor your site.
X
XLenny Glassmann
Xlenny@ccavax.camb.com
Xuunet!ccavax!lenny
$ CALL UNPACK AAAREADME.TXT;1 1716919986
$ create/nolog 'f'
X$ cc/include=news_src arbitron
X$ link arbitron,`091.news_build`093news_library/lib,news_src:options_c_link/
Vopt
$ CALL UNPACK ARBBUILD.COM;2 1027810274
$ create/nolog 'f'
X/* ARBITRON.C - Calculate number of readers of each newsgroup for ANU NEWS.
X`009This is meant to be run once per month, with results mailed to
X`009netsurvey@decwrl.dec.com in time to get there before midnight on the
X`009last day of the month.  To be safe, do this on about the 20th of the
X`009month. The results are placed in the file ARBITRON.OUT.
X
X`009The NEWS database is read to determine what is the oldest and newest
X`009item number still on the system for each newsgroup.  Then each
X`009user's NEWSRC file is read to determine what the last item was
X`009that the user read. If the last item read is later than the oldest
X`009one still on the system, then the user is counted as a reader of
X`009the newsgroup.
X
X`009This program makes heavy use of subroutines that are part of NEWS to
X`009read the files.  This is not the most efficient way to read these
X`009files because work is done by those subroutines which is not needed
X`009by this program.  However, it made this easy to write and maintain,
X`009and since it only runs once a month, we don't care how long it takes.
X
X`009Just about all of the numbers generated by this program are estimates
X`009of some kind.  Number of users on the system is the number of
X`009accounts that have logged on interactively in the past 30 days.  Number
X`009of news readers is the number of users who are readers of at least
X`009one newsgroup.  A reader of a newsgroup is someone who has read an
X`009article that has an item number between the lowest and highest
X`009currently stored in the newsgroup.
X
X`009This has only been tested with ANU NEWS V5.9C, VAX C V3.0, VMS V5.2.
X`009If you have another configuration, you'll probably have to at least
X`009recompile this module.
X
X`009This must run from a place that has read access to every NEWSRC
X`009file on the system.  The NEWSRC file must be in the user's
X`009default directory to be counted.
X
X`009Lenny Glassmann
X`009Cambridge Computer Associates
X`00956 Beaver St.
X`009New York, NY 10004
X`009(212) 425-5830
X`009lenny@ccavax.camb.com
X
XThanks to:
X`009Geoff Huston (gih900@csc.anu.oz.au) for ANU NEWS.
X`009Joe Meadows (joe@fhcrcvax.bitnet) for the UAF reading code.
X`009Brian Reid (reid@decwrl.dec.com) for the original UNIX arbitron.
X
XHistory:
X`009Dec 27 1989 - Initial release.
X`009Jan 13 1990 - V1.1 count groups with only 1 item.
X*/
X
X#include "newsinclude"
X#include lnmdef
X#include libdef
X#include "uaf"
X#include "newsdefine"
X
X#define extrnl
X#define iev(v)`009= v
X#define iv(v)`009v
X#include "newsvariables"
X
Xtypedef struct
X    `123
X    char name`091SUBJLEN`093;
X    unsigned int low_item;
X    unsigned int high_item;
X    unsigned short readers;
X    `125 GRP_INFO;
X
Xtypedef struct u_struct
X    `123
X    char *devdir;
X    char *name;
X    struct u_struct *next;
X    `125 USER_STRUCT;
X
Xtypedef unsigned long SYSTIME`0912`093;
X
Xstatic struct FAB uaffab;
Xstatic struct RAB uafrab;
Xstatic struct UAFDEF record;
Xstatic FILE *outfile;
Xstatic GRP_INFO *grparray;
Xstatic SYSTIME curtime;
Xstatic char version`091`093 = "vms-arbitron-1.1";
Xstatic int news_readers = 0;
Xstatic int num_users = 0;
X
X
Xarbitron ()
Xmain_program
X`123
Xchar reg_filename`091`093 = "newsrc";
X$DESCRIPTOR (delta_d, "30 0");`009`009/* Thirty days */
Xregister int grp_num;
XITM_PTR itmptr;
Xregister GRP_PTR group;
Xchar *cp;
Xchar *cp2;
Xint lastread;
Xchar user_reads_news;
Xint devlen;
Xint dirlen;
Xshort namelen;
Xchar username`091UAF$S_USERNAME + 1`093;
XSYSTIME delta_t;
XSYSTIME month_ago;
X
X    /* figure out the date that was 30 days ago */
X    sys$gettim (&curtime);
X    sys$bintim (&delta_d, &delta_t);
X    lib$sub_times (&curtime, &delta_t, &month_ago);
X
X    openfiles ();`009/* open and read NEWS.GROUPS */
X    open_uaf ();`009/* open SYSUAF.DAT */
X    open_outfiles ();
X
X    grparray = malloc (sizeof (GRP_INFO) * ga_size);
X
X    /* inspect each group (group 0 is not used) */
X    for (grp_num=1; grp_num <= ga_size; grp_num++)
X`009`123
X`009map_items (grp_num);`009`009/* read NEWS.ITEMS */
X`009group = ga`091grp_num`093;
X`009strcpy (grparray`091grp_num`093.name, group->grp_name);
X`009itmptr = group->grp_ia;
X`009if (group->grp_count > 0)
X`009    `123
X`009    /* get high and low item numbers (itmptr`0910`093 is not used) */
X`009    grparray`091grp_num`093.low_item = itmptr`0911`093.itm_num;
X`009    grparray`091grp_num`093.high_item = itmptr`091group->grp_count`093.i
Vtm_num;
X`009    `125
X`009else
X`009    `123
X`009    grparray`091grp_num`093.low_item = 0;
X`009    grparray`091grp_num`093.high_item = 0;
X`009    `125
X`009grparray`091grp_num`093.readers = 0;
X`009free (group->grp_ia);`009`009/* no longer need item list */
X`009`125
X
X    while (SYS$GET (&uafrab) & 1)`009/* read every record in the UAF */
X`009`123
X`009cp = username;
X`009cp2 = record.uaf$t_username;
X`009namelen = 0;
X`009while ((*cp2 != ' ') && (namelen++ < UAF$S_USERNAME))
X`009    *cp++ = *cp2++;
X`009*cp = '\0';
X
X`009printf ("%s ", username);
X
X`009if (lib$sub_times (&record.uaf$q_lastlogin_i, &month_ago,
X`009`009`009`009`009`009&delta_t) == LIB$_NEGTIM)
X`009    `123
X`009    printf ("hasn't logged in this month\n");
X`009    continue;`009`009`009`009/* skip this user */
X`009    `125
X
X`009num_users++;
X
X`009devlen = *record.uaf$t_defdev;
X`009dirlen = *record.uaf$t_defdir;
X`009if (devlen + dirlen + strlen (reg_filename) + 1 > sizeof(news_register))
X`009    `123
X`009    printf ("register file name is too long\n");
X`009    exit (0);
X`009    `125
X`009memcpy (news_register, &record.uaf$t_defdev`0911`093, devlen);
X`009memcpy (news_register + devlen, &record.uaf$t_defdir`0911`093, dirlen);
X`009*(news_register + devlen + dirlen) = '\0';
X
X`009/* if more than one user has the same default directory, and therfore
X`009   the same NEWSRC file, we only want to count it once.`009`009     */
X`009if (duplicate_user (news_register, username))
X`009    continue;
X
X`009strcat (news_register, reg_filename);
X
X`009if (access (news_register, 0) != 0)
X`009    `123
X`009    printf ("no NEWSRC file found\n");
X`009    continue;`009`009`009/* skip this user */
X`009    `125
X
X`009clear_profile_flags ();`009/* get rid of profile info from other users *
V/
X`009read_reg_file ();`009/* read the NEWSRC file */
X`009user_reads_news = FALSE;
X
X`009for (grp_num=1; grp_num <= ga_size; grp_num++)
X`009    `123
X`009    group = ga`091grp_num`093;
X`009    if (group->grp_reg_text)
X`009`009`123
X`009`009cp = group->grp_reg_text +
X`009`009`009strlen (group->grp_reg_text) - 1;
X`009`009while (isdigit (*cp))       /* find last non-digit */
X`009`009    --cp;
X`009`009lastread = atoi (++cp);
X`009`009free (group->grp_reg_text);
X`009`009group->grp_reg_text = NULL;
X`009`009if ((grparray`091grp_num`093.low_item != 0)          &&
X`009`009    (lastread >= grparray`091grp_num`093.low_item)   &&
X`009`009    (lastread <= grparray`091grp_num`093.high_item))
X`009`009    `123
X`009`009    grparray`091grp_num`093.readers++;
X`009`009    user_reads_news = TRUE;
X`009`009    `125
X`009`009`125
X`009    `125
X`009if (user_reads_news)
X`009    `123
X`009    news_readers++;
X`009    printf ("reads news\n");
X`009    `125
X`009else
X`009    `123
X`009    printf ("doesn't read news\n");
X`009    `125
X`009`125
X
X    sort_output ();
X    write_header ();
X    write_data ();
X`125
X
Xwrite_header ()
X`123
Xchar *cp;
Xunsigned int ret;
Xchar timbuf`09113`093;
X$DESCRIPTOR (timbuf_d, timbuf);
X$DESCRIPTOR (systbl, "LNM$SYSTEM_TABLE");
X$DESCRIPTOR (addr_log_d, "NEWS_ADDRESS");
Xunsigned short hostlen;
Xchar hostnam`09180`093;
Xstruct
X    `123
X    unsigned short buflen;
X    unsigned short itmcode;
X    char *buf_addr;
X    int *retlen;
X    `125 hostitm`091`093 =
X`009`123`123sizeof (hostnam) - 1, LNM$_STRING, &hostnam, &hostlen`125,
X`009`1230, 0, 0, 0`125`125;
X
X    if ((ret = sys$trnlnm (0, &systbl, &addr_log_d, 0, &hostitm)) != SS$_NOR
VMAL)
X`009lib$stop (ret);
X    hostnam`091hostlen`093 = '\0';
X
X    for (cp = hostnam; *cp; cp++)
X`009*cp = _tolower (*cp);
X    fprintf (outfile, "Host            %s\n", hostnam);
X    fprintf (outfile, "Users           %d\n", num_users);
X    fprintf (outfile, "NetReaders      %d\n", news_readers);
X    sys$asctim (0, &timbuf_d, &curtime, 0);
X    fprintf (outfile, "ReportDate      %c%c%c%.4s\n",
X`009timbuf`0913`093, _tolower(timbuf`0914`093), _tolower(timbuf`0915`093), &
Vtimbuf`0917`093);
X    fprintf (outfile, "SystemType      %s\n", version);
X`125
X
Xwrite_data ()
X`123
X    GRP_INFO group_rec;
X    struct dsc$descriptor rec_d;
X    unsigned int ret;
X
X    rec_d.dsc$w_length = sizeof (group_rec);
X    rec_d.dsc$b_dtype = DSC$K_DTYPE_T;
X    rec_d.dsc$b_class = DSC$K_CLASS_S;
X    rec_d.dsc$a_pointer = &group_rec;
X
X    while ((ret = sor$return_rec (&rec_d)) == SS$_NORMAL)
X`009fprintf (outfile, "%d %s\n", group_rec.readers, group_rec.name);
X
X    if (ret != SS$_ENDOFFILE)
X`009lib$stop (ret);
X`125
X
X/* sort the newsgroup lines in descending order by number of readers */
Xsort_output ()
X`123
X    unsigned int ret;
X    unsigned short sort_buf`091`093 =
X`009`1231, DSC$K_DTYPE_WU, 1,
X`009(char *)&grparray`0910`093.readers - (char *)&grparray`0910`093,
X`009sizeof (grparray`0910`093.readers)`125;
X    unsigned short reclen = sizeof (grparray`0910`093);
X    struct dsc$descriptor rec_d;
X    unsigned int grp_num;
X
X    if ((ret = sor$begin_sort (&sort_buf, &reclen, 0, 0, 0, 0, 0, 0, 0)) !=
X`009`009`009`009`009`009`009`009    SS$_NORMAL)
X`009lib$stop (ret);
X    rec_d.dsc$w_length = sizeof (grparray`0910`093);
X    rec_d.dsc$b_dtype = DSC$K_DTYPE_T;
X    rec_d.dsc$b_class = DSC$K_CLASS_S;
X
X    for (grp_num=1; grp_num <= ga_size; grp_num++)
X`009`123
X`009rec_d.dsc$a_pointer = &grparray`091grp_num`093;
X`009if ((ret = sor$release_rec (&rec_d)) != SS$_NORMAL)
X`009    lib$stop (ret);
X`009`125
X
X    if ((ret = sor$sort_merge (0)) != SS$_NORMAL)
X`009lib$stop (ret);
X`125
X
X
X/* check for users who have the same default directory */
Xduplicate_user (devdir, name)
Xchar *devdir;
Xchar *name;
X`123
X    static USER_STRUCT *top_user = NULL;
X    register USER_STRUCT *userp;
X    register USER_STRUCT *new;
X    USER_STRUCT *prev;
X    char past = FALSE;
X    short cmp;
X
X    prev = NULL;
X    userp = top_user;
X    while (userp && !past)
X`009`123
X`009cmp = strcmp (userp->devdir, devdir);
X`009if (cmp == 0)
X`009    `123
X`009    printf ("duplicated by %s\n", userp->name);
X`009    return (TRUE);
X`009    `125
X`009if (cmp > 0)
X`009    past = TRUE;
X`009else
X`009    `123
X`009    prev = userp;
X`009    userp = userp->next;
X`009    `125
X`009`125
X
+-+-+-+-+-+-+-+-  END  OF PART 1 +-+-+-+-+-+-+-+-