lazarus-ccr/components/flashfiler/sourcelaz/fflllog.pas
2016-12-07 13:31:59 +00:00

545 lines
14 KiB
ObjectPascal

{*********************************************************}
{* FlashFiler: Logging facility *}
{*********************************************************}
(* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is TurboPower FlashFiler
*
* The Initial Developer of the Original Code is
* TurboPower Software
*
* Portions created by the Initial Developer are Copyright (C) 1996-2002
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* ***** END LICENSE BLOCK ***** *)
{$I ffdefine.inc}
unit fflllog;
interface
uses
Classes,
ExtCtrls, {!!.06}
SysUtils,
Windows,
ffllbase;
type
{ Base class for event logs. }
TffBaseLog = class(TffComponent)
protected { private }
{ Property variables }
FCache : Boolean; {!!.06}
FCacheLimit : Integer; {!!.06}
FEnabled : Boolean;
FFileName : TFileName;
{ Internal variables }
blLogCS : TRTLCriticalSection;
{Begin !!.06}
blTimer : TTimer;
{ When caching, flushes cache during periods of inactivity. The timer
is enabled only when caching is enabled and something is written to
the log. The timer is reset as more stuff is added to the log. }
{End !!.06}
{ Property methods }
function blGetFileName : TFileName;
protected
procedure blLockLog;
procedure blUnlockLog;
function blGetEnabled : Boolean;
procedure blOnTimer(Sender : TObject); virtual; {!!.06}
procedure blSetEnabled(const Value : Boolean); virtual;
procedure blSetFileName(const Value : TFileName); virtual;
procedure Clear; virtual;
public
constructor Create(aOwner : TComponent); override;
destructor Destroy; override;
procedure Flush; virtual; {!!.06}
procedure WriteBlock(const S : string; Buf : pointer;
BufLen : TffMemSize); virtual; abstract;
{ Use this method to write a block of data to the event log. }
procedure WriteString(const aMsg : string); virtual; abstract;
{ Used to write a string to the event log. }
procedure WriteStringFmt(const aMsg : string; args : array of const); virtual; abstract;
{ Used to write a formatted string to the event log. }
procedure WriteStrings(const Msgs : array of string); virtual; abstract;
{ Used to write a block of strings to the event log. }
{ Properties }
{Begin !!.06}
property CacheEnabled : Boolean
read FCache
write FCache
default True;
{ If True then log lines are cached in memory and flushed to
disk once the CacheLimit has been reached. }
property CacheLimit : Integer
read FCacheLimit
write FCacheLimit
default 500;
{ The maximum number of log lines that may be retained in
memory. Not used if CacheEnabled is set to False. }
{End !!.06}
property Enabled : Boolean
read blGetEnabled
write blSetEnabled
default False; {!!.01}
{ Enable/disable event logging. }
property FileName : TFileName
read blGetFileName write blSetFileName;
{ The file to which the event log is written. }
end;
TffEventLog = class(TffBaseLog)
protected
FLog : TStringList; {!!.06}
FLogSize : Integer; {!!.06}
FTruncateSize : Integer; {!!.06}
FMaxSize : Integer; {!!.06}
FWriteBlockData : Boolean; {!!.06}
procedure elTruncateCheck(const Stream : TStream); {!!.06}
procedure elWritePrim(const LogStr : string); virtual; {!!.05}
public
constructor Create(aOwner : TComponent); override;
destructor Destroy; override;
procedure Flush; override; {!!.06}
{ Flushes the contents of the cache to the log. } {!!.06}
procedure WriteBlock(const S : string; Buf : pointer;
BufLen : TffMemSize); override;
procedure WriteString(const aMsg : string); override;
procedure WriteStringFmt(const aMsg : string; args : array of const); override;
procedure WriteStrings(const Msgs : array of string); override;
published
{ Inherited properties }
property CacheEnabled; {!!.06}
property CacheLimit; {!!.06}
property Enabled;
property FileName;
{Begin !!.06}
property MaxSize : Integer
read FMaxSize
write FMaxSize
default 50;
{ Max size (in megabytes) of the log file. Once the log file
reaches this size it will be truncated to TruncateSize. By
default, the log is truncated at 50MB. }
property TruncateSize : Integer
read FTruncateSize
write FTruncateSize
default ffcl_1KB;
{ Kilobytes of log kept when truncated. By default, 1MB is kept
when the log is truncated. See MaxSize. }
property WriteBlockData : Boolean
read FWriteBlockData
write FWriteBlockData
default False;
{ If set to False then data passed to WriteBlock is *not*
written to the log. }
{End !!.06}
end;
{Begin !!.06}
const
ffc_FlushTimerInterval : Cardinal = 1000;
{End !!.06}
implementation
const
ffcsSpaces13 = ' ';
ffcsSpaces44 = ffcsSpaces13 + ffcsSpaces13 + ffcsSpaces13 + ' ';
ffcsFormat = '%s %12d %8d %s' + ffcCRLF;
{===TffBaseLog=======================================================}
constructor TffBaseLog.Create(aOwner : TComponent);
begin
inherited Create(aOwner);
InitializeCriticalSection(blLogCS);
FCache := True;
FCacheLimit := 500;
{Begin !!.06}
blTimer := TTimer.Create(nil);
blTimer.Enabled := False;
blTimer.Interval := ffc_FlushTimerInterval;
blTimer.OnTimer := blOnTimer;
{End !!.06}
end;
{--------}
destructor TffBaseLog.Destroy;
begin
FFNotifyDependents(ffn_Destroy); {!!.11}
blTimer.Free; {!!.05}
DeleteCriticalSection(blLogCS);
inherited Destroy;
end;
{--------}
function TffBaseLog.blGetEnabled : Boolean;
begin
blLockLog;
try
Result := FEnabled;
finally
blUnlockLog;
end;
end;
{--------}
function TffBaseLog.blGetFileName : TFileName;
begin
blLockLog;
try
Result := FFileName;
finally
blUnlockLog;
end;
end;
{--------}
procedure TffBaseLog.blLockLog;
begin
if IsMultiThread then
EnterCriticalSection(blLogCS);
end;
{Begin !!.06}
{--------}
procedure TffBaseLog.blOnTimer(Sender : TObject);
begin
blLockLog;
try
blTimer.Enabled := False;
Flush;
finally
blUnlockLog;
end;
end;
{End !!.06}
{--------}
procedure TffBaseLog.blSetEnabled(const Value : Boolean);
begin
blLockLog;
try
FEnabled := Value;
finally
blUnlockLog;
end;
end;
{--------}
procedure TffBaseLog.blSetFileName(const Value : TFileName);
begin
blLockLog;
try
FFileName := Value;
finally
blUnlockLog;
end;
end;
{--------}
procedure TffBaseLog.blUnlockLog;
begin
if IsMultiThread then
LeaveCriticalSection(blLogCS);
end;
{Begin !!.06}
{--------}
procedure TffBaseLog.Clear;
begin
{ Do nothing }
end;
{--------}
procedure TffBaseLog.Flush;
begin
{ Do nothing }
end;
{End !!.06}
{====================================================================}
{===TffEventLog======================================================}
{Begin !!.06}
constructor TffEventLog.Create(aOwner : TComponent);
begin
inherited Create(aOwner);
FLog := TStringList.Create;
FLogSize := 0;
FWriteBlockData := False;
FMaxSize := 50;
FTruncateSize := ffcl_1KB;
end;
{--------}
destructor TffEventLog.Destroy;
begin
Flush;
FLog.Free;
inherited;
end;
{--------}
procedure TffEventLog.elTruncateCheck(const Stream : TStream);
var
TruncBytes,
MaxBytes : Integer;
TempStr : string;
begin
{ Convert MaxSize to Bytes. }
MaxBytes := (FMaxSize * ffcl_1MB);
{ Is it time to truncate this log file? }
if ((FMaxSize <> 0) and
(FLogSize > MaxBytes)) then begin
{ Convert the truncate size to bytes. }
TruncBytes := (FTruncateSize * ffcl_1KB);
{ Position the log to the portion we want to keep. }
Stream.Seek(TruncBytes * -1, soFromEnd);
{ Preserve the part we want to keep. }
SetLength(TempStr, TruncBytes);
Stream.Read(TempStr[1], TruncBytes);
{ Truncate the file. }
Stream.Size := TruncBytes;
{ Position to the beginning of the file and write the preserved
portion of the log. }
Stream.Position := 0;
Stream.Write(TempStr[1], TruncBytes);
{ Reset the log's size. }
FLogSize := TruncBytes;
end;
end;
{--------}
{End !!.06}
procedure TffEventLog.elWritePrim(const LogStr : string);
{Rewritten !!.06}
var
FileStm : TFileStream;
LogMode : Word;
begin
{ Assumption: Log file locked for use by this thread. }
if FCache then begin
blTimer.Enabled := False;
if FLog.Count = FCacheLimit then
Flush;
blTimer.Enabled := True;
FLog.Add(LogStr);
end
else begin
{ Check whether file exists, set flags appropriately }
if FileExists(FFileName) then
LogMode := (fmOpenReadWrite or fmShareDenyWrite)
else
LogMode := (fmCreate or fmShareDenyWrite);
{ Open file, write string, close file }
FileStm := TFileStream.Create(FFileName, LogMode);
try
elTruncateCheck(FileStm);
FileStm.Seek(0, soFromEnd);
FLogSize := FLogSize +
FileStm.Write(LogStr[1], Length(LogStr));
finally
FileStm.Free;
end;
end;
end;
{Begin !!.06}
{--------}
procedure TffEventLog.Flush;
var
Inx : Integer;
aStr : string;
FileStm : TFileStream;
LogMode : Word;
begin
{ Assumption: Log file locked for use by this thread. }
if FCache and (FLog.Count > 0) and (FFileName <> '') then begin
{ Check whether file exists, set flags appropriately }
if FileExists(FFileName) then
LogMode := (fmOpenReadWrite or fmShareDenyWrite)
else
LogMode := (fmCreate or fmShareDenyWrite);
{ Open file, write string, close file }
FileStm := TFileStream.Create(FFileName, LogMode);
try
elTruncateCheck(FileStm);
FileStm.Seek(0, soFromEnd);
for Inx := 0 to Pred(FLog.Count) do begin
aStr := FLog.Strings[Inx];
FLogSize := FLogSize +
FileStm.Write(aStr[1], Length(aStr));
end;
finally
FileStm.Free;
end;
FLog.Clear;
end;
end;
{End !!.06}
{--------}
procedure TffEventLog.WriteBlock(const S : string; Buf : pointer;
BufLen : TffMemSize);
const
HexPos : array [0..15] of byte =
(1, 4, 7, 10, 14, 17, 20, 23, 27, 30, 33, 36, 40, 43, 46, 49);
HexChar : array [0..15] of char =
'0123456789abcdef';
var
B : PffByteArray absolute Buf;
ThisWidth,
i, j : integer;
Line : string[70];
Work : byte;
begin
{Begin !!.06}
if FWriteBlockData then begin
blLockLog;
try
WriteStringFmt('%s (Size: %d)', [S, BufLen]);
if (BufLen = 0) or (Buf = nil) then
elWritePrim(ffcsSpaces13 + 'buffer is nil' + ffcCRLF)
else begin
if (BufLen > 1024) then begin
elWritePrim(ffcsSpaces13 + '(writing first 1K of buffer only)' + ffcCRLF);
BufLen := 1024;
end;
for i := 0 to ((BufLen-1) shr 4) do begin
FillChar(Line, 70, ' ');
Line[0] := #70;
Line[53] := '['; Line[70] := ']';
if (BufLen >= 16) then
ThisWidth := 16
else
ThisWidth := BufLen;
for j := 0 to ThisWidth-1 do begin
Work := B^[(i shl 4) + j];
Line[HexPos[j]] := HexChar[Work shr 4];
Line[HexPos[j]+1] := HexChar[Work and $F];
if (Work < 32) or (Work >= $80) then
Work := ord('.');
Line[54+j] := char(Work);
end;
elWritePrim(ffcsSpaces13 + Line + ffcCRLF);
dec(BufLen, ThisWidth);
end;
end;
finally
blUnlockLog;
end;
end; { if }
{End !!.06}
end;
{--------}
procedure TffEventLog.WriteString(const aMsg : string);
var
LogStr : string;
begin
{ Bail if logging isn't turned on }
if not FEnabled then Exit;
blLockLog;
try
{ Create appropriate string for log }
LogStr := format(ffcsFormat,
[DateTimeToStr(Now), getTickCount,
getCurrentThreadID, aMsg]);
elWritePrim(LogStr);
finally
blUnlockLog;
end;
end;
{--------}
procedure TffEventLog.WriteStringFmt(const aMsg : string; args : array of const);
var
LogStr : string;
begin
{ Bail if logging isn't turned on }
if not FEnabled then Exit;
blLockLog;
try
{ Create appropriate string for log }
LogStr := format(ffcsFormat,
[DateTimeToStr(Now), getTickCount,
getCurrentThreadID, format(aMsg, args)]);
elWritePrim(LogStr);
finally
blUnlockLog;
end;
end;
{--------}
procedure TffEventLog.WriteStrings(const Msgs : array of string);
var
Index : longInt;
LogStr : string;
MsgStr : string;
begin
{ Bail if logging isn't turned on }
if not FEnabled then Exit;
blLockLog;
try
for Index := 0 to high(Msgs) do begin
{ Create appropriate string for log }
MsgStr := Msgs[Index];
if (length(MsgStr) = 0) then
LogStr := ffcCRLF
else if(MsgStr[1] = ' ') then
LogStr := ffcsSpaces44 + MsgStr + ffcCRLF
else
LogStr := format(ffcsFormat,
[DateTimeToStr(Now), getTickCount,
getCurrentThreadID, MsgStr]);
elWritePrim(LogStr);
end;
finally
blUnlockLog;
end;
end;
{====================================================================}
end.