utmp.pp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. unit Utmp;
  2. interface
  3. uses
  4. {$ifdef ver1_0}
  5. Linux
  6. {$else}
  7. BaseUnix,
  8. Unix
  9. {$endif}
  10. ;
  11. const
  12. Device_name_length = 12;
  13. Host_name_length = 245;
  14. User_name_length = 32;
  15. type
  16. tTime = Longint;
  17. tPid = Longint;
  18. tIP_Address = array[1..4] of Byte;
  19. tDevice_name = String[Device_name_length];
  20. tUser_name = String[User_name_length];
  21. tHost_name = String[Host_name_length];
  22. tLogin_type = (Unkown, Run_level, Boot_time, New_time, Old_time,
  23. Init_process, Login_process, User_process, Dead_process);
  24. tLogin_types = set of tLogin_type;
  25. tParameter_type = (Include, Exclude);
  26. tUser = packed record
  27. Type_of_login : tLogin_type;
  28. Pid : tPid;
  29. Device : tDevice_name;
  30. Tty_name : String[4];
  31. Login_time : tTime;
  32. User_name : tUser_name;
  33. Host_name : tHost_name;
  34. IP_Address : tIP_Address;
  35. end;
  36. pUser = ^tUser;
  37. Const
  38. DefaultLoginType : TLogin_Types = [User_Process];
  39. Login_type_names : array [TLogin_type] of string[20] =
  40. ('Unkown', 'Run level', 'Boot time','New time', 'Old time',
  41. 'Init process', 'Login process', 'User process', 'Dead process');
  42. All_Login_types : TLogin_types = [Unkown, Run_level, Boot_time, New_time, Old_time,
  43. Init_process, Login_process, User_process, Dead_process];
  44. procedure Read_logged_users;
  45. function Get_next_user : tUser;
  46. function Get_next_user(var Last : Boolean) : tUser;
  47. procedure Set_search_parameters(
  48. const Parameter_type : tParameter_type;
  49. Login_types : tLogin_types);
  50. procedure Reset_user_search;
  51. function More_users : Boolean;
  52. function Number_of_logged_users : Word;
  53. {Low level rutines}
  54. function Number_of_utmp_entries : Word;
  55. procedure Set_utmp_file(const File_name : String);
  56. type
  57. texitstatus = record
  58. e_termination,
  59. e_exit : integer;
  60. end;
  61. tLL_Utmp = record
  62. ut_type : integer;
  63. ut_pid : longint;
  64. ut_line : tdevice_name;
  65. ut_id : array[1..4] of char;
  66. ut_user : tuser_name;
  67. ut_host : thost_name;
  68. ut_exit : texitstatus;
  69. ut_session : longint;
  70. ut_tv : Array [1..2] of longint;
  71. ut_addr : Array[1..4] of longint;
  72. pad : array [1..48] of char
  73. end;
  74. pUser_list = ^tUser_list;
  75. tUser_list = record
  76. User : tUser;
  77. Next : pUser_list;
  78. end;
  79. implementation
  80. Type
  81. tSearch_parameters = record
  82. Type_of_login_types : tParameter_type;
  83. Login_types : tLogin_types;
  84. end;
  85. var
  86. User_list : pUser_list;
  87. Current_user : pUser_list;
  88. Utmp_file : String;
  89. Search_parameters : tSearch_parameters;
  90. procedure Set_search_parameters(
  91. const Parameter_type : tParameter_type;
  92. Login_types : tLogin_types);
  93. begin
  94. Search_parameters.Type_of_login_types := Parameter_type;
  95. Search_parameters.Login_types := Login_types;
  96. end;
  97. function More_users : Boolean;
  98. var
  99. UL : pUser_list;
  100. Last : Boolean;
  101. begin
  102. UL := Current_user;
  103. Last := True;
  104. while (UL <> nil) and Last do begin
  105. if Search_parameters.Type_of_login_types = Exclude then begin
  106. Last := (UL^.User.Type_of_login in Search_parameters.Login_types);
  107. end else begin
  108. Last := not (UL^.User.Type_of_login in Search_parameters.Login_types);
  109. end;
  110. UL := UL^.Next;
  111. end;
  112. More_users := not Last;
  113. end;
  114. function Number_of_logged_users : Word;
  115. var
  116. I : Word;
  117. UL: pUser_list;
  118. begin
  119. I := 0;
  120. UL := User_list;
  121. while UL <> nil do begin
  122. if UL^.User.Type_of_login = User_process then begin
  123. I := I + 1;
  124. end;
  125. UL := UL^.Next;
  126. end;
  127. Number_of_logged_users := I;
  128. end;
  129. function Get_next_user : tUser;
  130. var
  131. Found : Boolean;
  132. User : pUser;
  133. begin
  134. if Current_user <> nil then begin
  135. Found := False;
  136. while (Current_user <> nil) and not Found do begin
  137. User := @Current_user^.User;
  138. if Search_parameters.Type_of_login_types = Exclude then begin
  139. Found := not (User^.Type_of_login in Search_parameters.Login_types);
  140. end else begin
  141. Found := (User^.Type_of_login in Search_parameters.Login_types);
  142. end;
  143. Current_user := Current_user^.Next;
  144. end;
  145. if Found then begin
  146. Get_next_user := User^;
  147. end else begin
  148. New(User);
  149. FillChar(User^, SizeOf(tUser), 0);
  150. Get_next_user := User^;
  151. Dispose(User);
  152. end;
  153. end else begin
  154. New(User);
  155. FillChar(User^, SizeOf(tUser), 0);
  156. Get_next_user := User^;
  157. Dispose(User);
  158. end;
  159. end;
  160. function Get_next_user(var Last : Boolean) : tUser;
  161. var
  162. Found : Boolean;
  163. User : pUser;
  164. UL : pUser_list;
  165. begin
  166. if Current_user <> nil then begin
  167. Found := False;
  168. while (Current_user <> nil) and not Found do begin
  169. User := @Current_user^.User;
  170. if Search_parameters.Type_of_login_types = Exclude then begin
  171. Found := not (User^.Type_of_login in Search_parameters.Login_types);
  172. end else begin
  173. Found := (User^.Type_of_login in Search_parameters.Login_types);
  174. end;
  175. Current_user := Current_user^.Next;
  176. end;
  177. if Found then begin
  178. Get_next_user := User^;
  179. UL := Current_user;
  180. Last := True;
  181. while (UL <> nil) and Last do begin
  182. if Search_parameters.Type_of_login_types = Exclude then begin
  183. Last := (UL^.User.Type_of_login in Search_parameters.Login_types);
  184. end else begin
  185. Last := not (UL^.User.Type_of_login in Search_parameters.Login_types);
  186. end;
  187. UL := UL^.Next;
  188. end;
  189. end else begin
  190. New(User);
  191. FillChar(User^, SizeOf(tUser), 0);
  192. Get_next_user := User^;
  193. Dispose(User);
  194. end;
  195. end else begin
  196. New(User);
  197. FillChar(User^, SizeOf(tUser), 0);
  198. Get_next_user := User^;
  199. Dispose(User);
  200. end;
  201. end;
  202. procedure Reset_user_search;
  203. begin
  204. Current_user := User_list;
  205. end;
  206. procedure Set_utmp_file(const File_name : String);
  207. begin
  208. Utmp_file := File_name;
  209. end;
  210. function Number_of_utmp_entries : Word;
  211. var
  212. S : Stat;
  213. begin
  214. {$ifdef ver1_0}FStat{$else}fpstat{$endif}(Utmp_file, S);
  215. Number_of_utmp_entries := {$ifdef ver1_0}S.Size{$else}s.st_size{$endif} div System.SizeOf(tLL_Utmp);
  216. end;
  217. procedure Read_logged_users;
  218. procedure Read_entry(var F : File; var Entry : tUser; var User : Boolean);
  219. var
  220. LL_Entry : tLL_Utmp;
  221. I : Byte;
  222. begin
  223. BlockRead(F, LL_Entry, SizeOf(tLL_Utmp));
  224. //Byte(Entry.Type_of_login) := LL_Entry.ut_type;
  225. Entry.Type_of_login := tLogin_type(LL_Entry.ut_type);
  226. Entry.Pid := Longint(LL_Entry.ut_id);
  227. Entry.Device:=LL_entry.ut_line;
  228. Entry.TTy_Name:=LL_Entry.ut_id;
  229. Entry.Login_time := LL_Entry.ut_tv[1];
  230. Entry.User_name:=LL_entry.ut_user;
  231. Entry.Host_name:=LL_entry.ut_host;
  232. For I:=1 to 4 do
  233. Entry.Ip_Address[I] := LL_Entry.ut_addr[i];
  234. end;
  235. var
  236. F : File;
  237. I : Longint;
  238. UL: pUser_list;
  239. U : Boolean;
  240. begin
  241. System.Assign(F, Utmp_file);
  242. {$IFOPT I+}
  243. {$DEFINE I_was_on}
  244. {$ENDIF}
  245. {$I-}
  246. System.Reset(F,1);
  247. {$IFDEF I_was_on}
  248. {$UNDEF I_was_on}
  249. {$I+}
  250. {$ENDIF}
  251. UL := User_list;
  252. while UL <> nil do begin
  253. User_list := UL;
  254. UL := UL^.Next;
  255. Dispose(User_list);
  256. end;
  257. User_list := nil;
  258. if System.IOResult = 0 then begin
  259. for I := 1 to Number_of_utmp_entries do begin
  260. if User_list = nil then begin
  261. New(User_list);
  262. UL := User_list;
  263. UL^.Next := nil;
  264. end else begin
  265. New(UL^.Next);
  266. UL := UL^.Next;
  267. UL^.Next := nil;
  268. end;
  269. Read_entry(F, UL^.User, U);
  270. end;
  271. UL^.Next := nil;
  272. end else begin
  273. User_list := nil;
  274. end;
  275. System.Close(F);
  276. Current_user := User_list;
  277. end;
  278. begin
  279. User_list := nil;
  280. Current_user := nil;
  281. Utmp_file := '/var/run/utmp';
  282. Set_search_parameters(Include,DefaultLoginType);
  283. end.
  284. $Log$
  285. Revision 1.4 2003-09-29 19:28:09 marco
  286. * stat record fixes.
  287. Revision 1.3 2003/09/27 12:13:50 peter
  288. * fixed for unix
  289. Revision 1.2 2002/09/07 15:43:06 peter
  290. * old logs removed and tabs fixed
  291. Revision 1.1 2002/01/29 17:55:23 peter
  292. * splitted to base and extra
  293. }