User log, tools
Volume Number: | | 5
|
Issue Number: | | 5
|
Column Tag: | | Programmer's Workshop
|
User Log and Programming Tools
By Dave Kelly, MacTutor Editorial Board
It was just a few short weeks ago that I was cleaning out our nVir attack. Yea, about 8 out of 10 of our Macs had the dreaded disease. The guy that wrote it probably thinks he is real funny right now. Ha, Ha. Well, after about 15-20 hours worth of checking floppy disks and cleaning systems Im not laughing. That just goes to show you that even when you think you are fine it could still get you. At least my own Mac at home was still clean.
Making a fuss like that in a big company doesnt go over very well. The management has sent out an edict that all Macintoshes must be checked for viruses once each day. Thats a bit much, wouldnt you say? In addition, they want every user (there are many more users than there are Macs) to log in each disk that they insert into a Macintosh. WHAT? you say. Thats right, write down on a log sheet your name and what disk you inserted into the system. I dont need to tell you that most people dont bother to write anything on the log sheets. This was bound to backfire somehow. Probably someone will think that few people use the Macintosh (this is obviously not true since it is hard to even get on a Mac during the day, and I have to bring my work home) and not allow us to buy more (much needed) computers.
There are a few options open to solve this problem. We could just lock up everyones floppy disks so that they cant take them home, or buy a commercially available activity tracking system such as MacInUse to log each user in, or we could write a quick and dirty program to do at least part of the job for us. Buying a copy of software for every Mac we have doesnt appeal to anyone since we do have a strict budget. I hope they realize how much free use of equipment they get when employees to work at home on their own computers. We must be able to backup and transport data via floppy disk. I opted to write a short program to do the job. I didnt have the time to make it do everything, but it will meet the basic need to log everyone in.
HOW IT WORKS
The trick involved in this program is to get the program to run again after any application runs and when it quits, return back to the Finder. I didnt want to remove any features from the Macintosh such as the Finder. It would be easier to just replace the Finder with another application but that would make it hard for users to copy their own disks, etc. There are two global memory locations that are used to make this work. The first one, CurApName at $910 gives us the name of our application in case someone changed the name. The second one, FinderName at $2E0 is the name of the finder which is used to tell the system which program to run when an application quits. By putting CurApName into FinderName our application will run after each time another application quits provided that our application is put in the system folder.
The program Who? is a very simple program to record usage and on the computer. You may use 1 or more passwords (in case there are several people using the system). The passwords are stored in a STR# resource in the application. The passwords are not encrypted because most of the users that I come in contact with dont have ResEdit or even if they did they wouldnt know how to use it. The ones that do are going to get past this sort of password protection anyway. Remember, the main purpose is not to keep people out, but to log activity. You may edit passwords by changing the STR# resource.
When the program runs, a modal dialog box comes up to request the password and users name. The user must type in a name or the program will not end. (Enter Your Name here! is not a valid user name). When all the information has been filled in the user selects the OK button and if the password matches one of the STR# resource items, the program continues. At this point, the FinderName global is updated to the current application name as explained above. Next the available volumes are searched so that they can also be logged in. The purpose of this was to give some traceability if the virus appears again. We want to be able to trace to some extent who may have disks that are infected. This is not foolproof however because the user can always switch disks while running under another application. The function GetIndVolume for searching for mounted volumes is found in technical note #24 by Bryan Stearns. This is a good one to keep around for future use.
The Who? program finishes by fetching the time and date and storing the data to a file on disk, then launching the Finder. The log-in data file is set to invisible to make it harder to erase from the disk. It can be accessed via a word processor to examine who? has been using the Mac. I included a way to reset the FinderName back to Finder and make the log-in file visible. By typing ShowLog (case sensitive) for the password, the log will become a visible file. By using RestoreFinder (also case sensitive) for the user name, the FinderName global will be set to Finder. Thats about all there is to it. To set up the application, drag it into the system folder and then select it. Next choose Set StartUp from the Finders Special menu. If you use RestoreFinder be sure to Set StartUp again or the system will get confused the next time you startup.
There is a lot of things that could be done to change this program to add features which you may feel are needed. While this by no means keeps the virus bug from coming, it does provide a bit of traceability to help in the clean up process.
program who;
uses
filestuff;
const
DialogID = 4912;{Resource ID number}
FinderName = $2E0;{Global location of Finder name}
CurApName = $910; {Global location of Current Application name}
type {this type sets up structure for launching applications}
pLaunchStruct = ^LaunchStruct;
LaunchStruct = record
pfName: StringPtr;
param: integer;
end; { LaunchStruct}
var
itemhit: integer;
theDialog: Dialogptr;
itemtype, index: integer;
item2, item3: handle;
box: rect;
text2, text3, thepassword: str255;
access, done: boolean;
Filenameptr: STRINGptr;
fName: str255;
Finderptr: stringptr;
pMyLaunch: pLaunchStruct;
myLaunch: LaunchStruct;
d: DateTimeRec;
datafile: str255;
data: text;
procedure Launchit (pLnch: pLaunchStruct); {Launch an application}
inline
$205F, $A9F2;
procedure writedata;{store the log in file to disk}
var
whichvol, VolRefNum: integer;
volName: str255;
Result: OSErr;
fndrInfo: FInfo;
flgptr: ptr;
begin
datafile := Log in.data; {set filename for data file}
open(data, datafile); {open the file}
seek(data, maxlongint); {find the end of the file}
writeln(data, text3, , text2, , d.Month : 3, /, d.Day : 2,
/, d.Year : 4, d.Hour : 3, :, d.Minute : 2, :, d.Second : 2);
whichVol := 1; { start searching for disks in 1st vol.}
repeat { find all the volumes that are currently mounted}
begin
Result := GetIndVolume(whichVol, volName, VolRefNum);
if Result <> nsvErr then
begin {store the volumes on line to system disk}
writeln(data, whichvol, : , volName);
whichVol := whichVol + 1;
end;
end;
until Result = nsvErr;
close(data);
Result := GetFInfo(datafile, 0, fndrInfo);
if BitAnd(fndrInfo.fdFlags, fInvisible) <> 16384 then
fndrInfo.fdFlags := fndrInfo.fdFlags + fInvisible;
if text2 = ShowLog then
fndrInfo.fdFlags := fndrInfo.fdFlags - fInvisible;
Result := SetFInfo(datafile, 0, fndrInfo);
end;
begin {start of main program}
initcursor;
access := false;
text2 := Enter Password here!;
text3 := Enter Your Name here!;
theDialog := getnewdialog(dialogID, nil, POINTER(-1));
getditem(thedialog, 2, itemtype, item2, box);
setItext(item2, text2);
getditem(thedialog, 3, itemtype, item3, box);
setItext(item3, text3);
repeat
SelIText(theDialog, 2, 0, 32767);
itemhit := 10;
repeat
modaldialog(nil, itemhit);
until itemhit = 1;
getItext(item2, text2);
getItext(item3, text3);
done := false;
index := 1;
repeat
begin
getindstring(thepassword, DialogID, index);
if (text2 = thepassword) and (thepassword <> ) then
access := true;
if access = true then
done := true;
if thepassword = then
done := true;
index := index + 1;
end;
until done = true;
if access = false then
setItif text3 = Enter Your Name here! then
access := false;
until access = true;
closedialog(theDialog);
Filenameptr := STRINGptr(CurApName);
fName := Filenameptr^;
Filenameptr := STRINGptr(FinderName);
Filenameptr^ := fName;
if text3 = RestoreFinder then
Filenameptr^ := Finder;
GetTime(d);
WriteData;
pMyLaunch := @myLaunch;
fName := Finder;
with pMyLaunch^ do
begin
pfname := @fName;
param := 0;
end;
LaunchIt(pMyLaunch);
end.
unit Filestuff;
{ From Macintosh Technical Notes #24 }
{ by Bryan Stearns}
interface
function GetIndVolume (whichVol: integer; var volName: str255; var volRefNum:
Integer): OSErr;
implementation
function GetIndVolume; {(whichVol: integer, var volName: str255, var
volRefNum: Integer): OSErr;}
{Return the name and vRefNum of volume specified by whichVol}
var
volPB: HParamBlockRec;
error: OSErr;
begin
with volPB do {makes it easier to fill in!}
begin
ioNamePtr := @volName; {make sure it returns the name}
ioVRefNum := 0; {0 means use ioVolIndex}
ioVolIndex := whichVol; {use this to determine the volume}
end; {with}
error := PBHGetVInfo(@volPB, false);{do it}
if error = noErr then
begin
volRefNum := volPB.ioVRefNum;
end; {if no error}
{other information is available from this record; see the FILE }
{Managers description of PBHGetVInfo for more details}
GetIndVolume := error; {return error code}
end;
end. {of unit}
WHO.rsc
resource ICN# (128, purgeable)
{
0x07ff 0xffe0 0x39e0 0x079c
0xff70 0x0eff 0x83d8 0x1bc1
0x80fc 0x3f01 0x81ce 0x7381
0x8303 0xc0c1 0x8201 0x8041
0x8631 0x8c61 0x8631 0x8c61
0x8631 0x8c61 0x8201 0x8041
0x8303 0xc0c1 0x81cf 0xf381
0x81ff 0xff81 0x8101 0x8081
0x8301 0x80c1 0x8300 0x00c1
0x8300 0x00c1 0x8700 0x00e1
0x8700 0x00e1 0xa500 0x00a5
0xa500 0x00a5 0xa580 0x01a5
0xa7bb 0xdde5 0xa6f9 0x9f65
0xacc0 0x0335 0xb8ee 0x771d
0xb0f4 0x2f0d 0xe1fb 0xdf87
0xc3ff 0xffc3 0x82d7 0xeb41 ;
0x07ff 0xffe0 0x3fff 0xfffc
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
0xffff 0xffff 0xffff 0xffff
};
userdefined resource Dave (0)
{
long:0x2557686f;
long:0x3f205665;
long:0x7273696f;
long:0x6e20312e;
long:0x30310da9;
long:0x31393839;
long:0x20627920;
long:0x44617665;
long:0x204b656c;
integer:27769};
resource BNDL (130)
{
Dave, 0,
{
FREF, {1, 129};
ICN#, {0, 128}
}
};
resource FREF (129)
{
APPL,
0,
};
resource STR# (4912 , Passwords)
{
MacTutor;
ShowLog
};
resource DLOG (4912)
{
{64, 32, 180, 354},
0,
visible,
noGoAway,
0x0,
4912,
Please Log in\0xc9
};
resource DITL (4912)
{
{
{76, 128, 96, 188},
Button {enabled, OK};
{15, 123, 34, 303},
EditText {enabled, Enter Password Here!};
{42, 123, 61, 303},
EditText {enabled, Enter Your Name Here!};
{15, 43, 33, 116},
StaticText {enabled, Password:};
{42, 16, 61, 119},
StaticText {enabled, Name/Initials:}
}
};
Mac Programming Tools
Some mention has been made in the past regarding programming tools which create the Macintosh user interface. In October, 1988, MacTutor presented a comparison of Prototyper 1.0, MacExpress, and FaceIt all products to help programmers with the user interface. In order to update the list of these programs, I would be amiss if I did not mention Invention Softwares Professional Programmers EXTENDER (Full Source Code Version) and say a few words about the new release of Prototyper 2.0.
EXTENDER consists of a very large set of routines which may be used to implement all or portions of the Macintosh interface. It consists of C or Pascal source code, your choice, which you must compile with your application. Unlike the FaceIt which must have the library file available when running the application, EXTENDER allows you to use and compile all or parts of the routines as needed. The main problem with this is that the possible routines you may use are are as numerous as the first three volumes of Inside Macintosh.
EXTENDER comes with your choice of language disks and a huge binder which reminded me of the early Macintosh development days when Inside Macintosh was only available as loose leaf pages. (You can still get Inside Macintosh as loose leaf pages if you want). Routines included are:
Menu
Windows
Controls
Scroll Bars
Dialogs and Alert Boxes
Text Editing
Bit Maps
Graphics
Printing
The List Manager
Popup menus
Input/Output routines
Event handling
Each of the categories above are organized in a tutorial/reference format so you can study and learn how to implement the routines by observing and practicing the demos provided. Be prepared to do a lot of reading.
There are updates available from time to time which will correct some of the problems which come up. If there are problems it is nearly always more difficult to debug someone elses code than to debug your own. The problem is that whenever generic routines are used there are cases or exceptions when you want to do something just a bit different. In EXTENDER that requires modifying code that someone else wrote.
Prototyper 2.0 is a major new release from SmethersBarnes. If you thought that Prototyper 1.0 was great, youll love this version. Dozens of new features have been added to Prototyper based on comments of programmers and user interface designers. In particular Prototyper 2.0 adds:
New Code Generators
Enhanced Screen Layout Capabilities
Macintosh User Interface Feature Additions
So whats new? The following is a list of new features which will convince you that you should upgrade or buy Prototyper 2.0:
C code generation
C programmers have been begging for this since version 1.0 was released. Two separate code generators are included with Prototyper 2.0, one for C generation and one for Pascal. The Pascal generator creates source code for LightSpeed Pascal 2.0, Turbo Pascal, MPW Pascal, and TML Pascal II. The C generator creates source code for LightSpeed C, MPW C 2.0 and MPW C 3.0.
Screen Layout
The layout features you have seen in version 1.0 have been improved so that you can manipulate zones similar to the way you would in drawing programs. It is much easier to group and align objects. Screen grids and a new coordinate window let you position zones exactly down to the pixel.
Popup & hierarchical menus
The menu editor now supports one level of hierarchical menus. Most applications will probably never need more than one hierarchical level. Popup menus are included in the window palettes so you can place them anywhere in your window.
Set location
The window editor has been enhanced to support any size of monitor. Self centering options may be selected to center the window for any size screen. The screen grid mentioned above helps align objects on the window also.
Linking
You can now link menus, buttons, and menu items to enable/disable each other or to open or close windows. The standard dialogs for Print and Page Setup can be linked to open. The flow of an entire application can be establish through this simple mechanism. It is as easy as linking HyperCards together.
Other Features
You can now import and export resources from other applications. A Print to file option lets you output your screen designs to a MacPaint file to be used for documentation or other purposes. Demo windows have been added to allow you to created canned graphics and word processing windows. Prototyper 2.0 includes a new and much improved manual. The manual includes a tutorial, and detailed information about every aspect of the program.
After using EXTENDER and Prototyper for some time now I will have to say the I prefer Prototyper. EXTENDER has so many routines to become familiar with that I feel it would take about as long to learn it as to use Inside Macintosh and get to know the routines that are standard. With tools like Prototyper around I can create the structural part of my user interface with the minimum of discomfort.