#!/usr/local/bin/perl # count.xbm - Perl script to keep count of users of this # home page. # # Credits: # Original C code: Frans van Hoesel (hoesel@chem.rug.nl) # Original port to Perl: Dan Rich (drich@corp.sgi.com) # Modifications for cgi: Michael Nelson (m.l.nelson@larc.nasa.gov) # Modifications for NSF: Mike Morse (mmorse@nsf.gov) # Modified from NSF version: Spencer Thomas (spencer@umich.edu) # This version is NSF specific, but you can easily modify it # for your own needs. The main NSF-ism is that we run three # servers on the same machine, and this script needs to figure # out which server it is being called in behalf of. # To install: # 1. Look at initialize() and fix everything you need to. # 2. Create the files referenced in initialize(), with the # same owner as your server runs under, or world writable. # 3. Stick the script in cgi-bin. My experience is that the # script file must have the extension ".xbm". # 4. Stick something like this in your home page: # Number of home page visitors: # # # The second line is only if you use our "stats" program, you # can put anything there, or just use the IMG SRC bare. # 5. When the home page is displayed, Mosaic clients will retrieve # the in-line bitmap. This will invoke this script, which # responds with a bitmap it builds on the fly, that looks like # an automobile odometer. If you include the A HREF, the odometer # will be surrounded by a border (and look better), and users # can click the odometer to get some other information. # 6. Don't worry if older WinMosaic versions display "Error" for # the bitmap. The script is doing the Right Thing. &initialize; &incrementCount; # if ($ENV{PATH_INFO} && $ENV{PATH_INFO} =~ /.*\/text(\/.*)?/) { &writeTextCounter; } else { &generateBitmap; &writeBitmap; } exit(0); sub writeTextCounter { print ("Content-type: text/html\n\n"); print ("Visitor counter\n"); print ("You are visitor number $totalReads\n"); } sub writeBitmap { print ("Content-type: image/x-xbitmap\n\n"); if ($isHigh) { printf ("#define count_width %d\n#define count_height 16\n", $len*8); } else { printf ("#define count_width %d\n#define count_height 10\n", $len*8); } printf STDOUT "static char count_bits[] = {\n"; for($i = 0; $i < ($#bytes + 1); $i++) { print("0x$bytes[$i]"); if ($i != $#bytes) { print(","); if (($i+1) % 7 == 0) { print("\n"); } } } print("};\n"); } # generateBitmap() - $count contains number to display # $minLen contains minimum number of digits to display # $isHigh is one for 16 bit high numbers (else 10) # $isInverse is one for reverse video (white on black); sub generateBitmap { $count = $totalReads; @bytes = (); $len = length($count) > $minLen ? length($count) : $minLen; $formattedCount = sprintf("%0${len}d",$count); if ($isHigh) { for ($i = 0; $i < $len*3; $i++ ) { if ($isInverse) { push(@bytes,"ff"); # add three blank rows to each digit } else { push(@bytes,"00"); } } } for ($y=0; $y < 10; $y++) { for ($x=0; $x < $len; $x++) { $digit = substr($formattedCount,$x,1); if ($isInverse) { # $inv = 1 for inverted text $byte = substr(@invdigits[$digit],$y*3,2); } else { $byte = substr(@digits[$digit],$y*3,2); } push(@bytes,$byte); } } if ($isHigh) { for ($i = 0; $i < $len*3; $i++ ) { if ($isInverse) { push(@bytes,"ff"); # add three blank rows to each digit } else { push(@bytes,"00"); } } } } sub initialize { $host = $ENV{REMOTE_ADDR}; $port = $ENV{SERVER_PORT}; $minLen = 7; # minimum number of digits in bigmap $isHigh = 1; # if 1, digits are 16 pixels high, to # allow room for border $isInverse = 1; # If 1, digits are white on black $serverRoot = "/www"; $counterFile = "$serverRoot/logs/counter.txt"; $hostsFile = "$serverRoot/logs/counthosts.txt"; $dbmFile = "$serverRoot/logs/hosts"; $logFile = "$serverRoot/logs/counter.log"; $lockWait = 5; # number of seconds to wait for lock # bitmap for each digit # Each digit is 8 pixels wide, 10 high # @invdigits are white on black, @digits black on white @invdigits = ("c3 99 99 99 99 99 99 99 99 c3", # 0 "cf c7 cf cf cf cf cf cf cf c7", # 1 "c3 99 9f 9f cf e7 f3 f9 f9 81", # 2 "c3 99 9f 9f c7 9f 9f 9f 99 c3", # 3 "cf cf c7 c7 cb cb cd 81 cf 87", # 4 "81 f9 f9 f9 c1 9f 9f 9f 99 c3", # 5 "c7 f3 f9 f9 c1 99 99 99 99 c3", # 6 "81 99 9f 9f cf cf e7 e7 f3 f3", # 7 "c3 99 99 99 c3 99 99 99 99 c3", # 8 "c3 99 99 99 99 83 9f 9f cf e3"); # 9 @digits = ("3c 66 66 66 66 66 66 66 66 3c", # 0 "30 38 30 30 30 30 30 30 30 30", # 1 "3c 66 60 60 30 18 0c 06 06 7e", # 2 "3c 66 60 60 38 60 60 60 66 3c", # 3 "30 30 38 38 34 34 32 7e 30 78", # 4 "7e 06 06 06 3e 60 60 60 66 3c", # 5 "38 0c 06 06 3e 66 66 66 66 3c", # 6 "7e 66 60 60 30 30 18 18 0c 0c", # 7 "3c 66 66 66 3c 66 66 66 66 3c", # 8 "3c 66 66 66 66 7c 60 60 30 1c"); # 9 } sub incrementCount { &makeLogEntry; if (&lockFile == 1) { $count = "0"; return; } &incrementTotalReads; &incrementHosts; &unlockFile; } sub unlockFile { unlink("$counterFile.lock"); } sub lockFile { $lockCount = 0; while (-f "$counterFile.lock") { if ($lockCount > $lockWait) { $count = 0; return 1; # forget it (would be nice to log though) } sleep 1; $lockCount++; } open(LOCK,">$counterFile.lock") || die("Can't open $counterFile.lock: $!\n"); return 0; } sub incrementTotalReads { if (-e $counterFile) { open(COUNT,"$counterFile") || die("Can't open $counterFile: $!\n"); } $totalReads = ; chop $totalReads; close(COUNT); $totalReads++; open(COUNT,">$counterFile") || die "$0: can\'t open $counterFile: $!\n"; print (COUNT "$totalReads\n"); close(COUNT); } sub makeLogEntry { ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime(time); $date = sprintf("%02d/%02d/%02d %02d:%02d:%02d",$year,$mon+1,$mday, $hour,$min,$sec); open(LOG,">>$logFile") || die "$0: can\'t open $logFile: $!\n"; print (LOG "$date\t$host\t$port\n"); close(LOG); } sub incrementHosts { if (-e $hostsFile) { open(COUNT,"$hostsFile") || die("Can't open $hostsFile: $!\n"); } $totalHosts = ; chop $totalHosts; close(COUNT); dbmopen(HOSTS,$dbmFile,0666); if (! $host) { $host = "unknown"; } $HOSTS{$host}++; if ($HOSTS{$host} == 1) { $totalHosts++; } dbmclose(HOSTS); open(COUNT,">$hostsFile") || die "$0: can\'t open $hostsFile: $!\n"; print (COUNT "$totalHosts\n"); close(COUNT); }