Connect / Disconnect corrupting Application
Connect / Disconnect corrupting Application
Hello Everyone,
Has anybody faced this issue before? I'm looking for some help with the situation below:
I have an application POS developed with Delphi 7 + Zeos 7.1.4 with a SQLite Database which has been working for over a year in dozens of stations,
We recently made the update of Zeos Version to the newest (8.0.0) and shortly after that we started seeing Access Violation Errors. Even after applying all patches from SVN, the errors remain.
Starting from the moment the first error happens, you have to logout of the App and Login again. And the error does not happen immediately. It takes a while, usually on the 10 sales mark. Give or take a few.
After debugging, we found out that the error happens when you try to set a String as a parameter of a Query as show below:
zqrDel.ParamByName('NIDTIPO').AsString := CNIDTIPO_USUARIO;
It doesn't always happen on the same query, but always when you try to set a Value of type String to a parameter.
Digging deeper, we found the exact point in Zeos where the error happens:
Its on the file: ZDatasetParam.pas - On the Class TZParam - On the Method: procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);
The line where the error happens is:
If ((FConSettings <> nil) or SetConsettings) and (FConSettings.ClientCodePage.Encoding = ceUTF16) then begin
More precisely, its when the application evaluates the variable FConSettings
It is NOT Null, but it contains a value that points to a place in memory with garbage.
After more investigations, we found out what was causing this variable corruption. The App, between sales, for security reasons, disconnects from the Database, makes a copy from the Database and connects again.
procedure SafeBackup;
begin
zDBMov.Disconnect;
DoBackupDBMOV;
zDBMov.Connect;
end;
After commenting the call to procedure SafeBackup, the app has not had any problems so far. And the problem only happens after this is called several times.
Any Hint or Workaround ?
Has anybody faced this issue before? I'm looking for some help with the situation below:
I have an application POS developed with Delphi 7 + Zeos 7.1.4 with a SQLite Database which has been working for over a year in dozens of stations,
We recently made the update of Zeos Version to the newest (8.0.0) and shortly after that we started seeing Access Violation Errors. Even after applying all patches from SVN, the errors remain.
Starting from the moment the first error happens, you have to logout of the App and Login again. And the error does not happen immediately. It takes a while, usually on the 10 sales mark. Give or take a few.
After debugging, we found out that the error happens when you try to set a String as a parameter of a Query as show below:
zqrDel.ParamByName('NIDTIPO').AsString := CNIDTIPO_USUARIO;
It doesn't always happen on the same query, but always when you try to set a Value of type String to a parameter.
Digging deeper, we found the exact point in Zeos where the error happens:
Its on the file: ZDatasetParam.pas - On the Class TZParam - On the Method: procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);
The line where the error happens is:
If ((FConSettings <> nil) or SetConsettings) and (FConSettings.ClientCodePage.Encoding = ceUTF16) then begin
More precisely, its when the application evaluates the variable FConSettings
It is NOT Null, but it contains a value that points to a place in memory with garbage.
After more investigations, we found out what was causing this variable corruption. The App, between sales, for security reasons, disconnects from the Database, makes a copy from the Database and connects again.
procedure SafeBackup;
begin
zDBMov.Disconnect;
DoBackupDBMOV;
zDBMov.Connect;
end;
After commenting the call to procedure SafeBackup, the app has not had any problems so far. And the problem only happens after this is called several times.
Any Hint or Workaround ?
-
- Platinum Boarder
- Posts: 1995
- Joined: 17.01.2011, 14:17
Re: Connect / Disconnect corrupting Application
Hello,
hunting down memory bugs like this is hard.
You might want to check the VACUUM INTO command to see if you can live without deconnecting.
Best regards,
Jan
hunting down memory bugs like this is hard.
You might want to check the VACUUM INTO command to see if you can live without deconnecting.
Best regards,
Jan
Re: Connect / Disconnect corrupting Application
Hi Jan
Thanks for your help / Hint. In this specific case it solves the problem.
However we have another application where we need to connect and disconnect from time to time in the Database and in this case I can not avoid the Connect/Disconnect cycle.
I know that this kind of bug is very hard to find and fix, BUT I managed to replicate the BUG with a very, very simple application.
I create a SQLite Databse with just one table:
The application has just TWO querys and one Method:
function TDMZeos.InsertRecord( nValor: double; sStatus: string): Boolean;
var
nMax : integer;
begin
ZDB.Connect;
try
qrMax.Close;
qrMax.Open;
nMax := qrMaxNFCFICHA.AsInteger + 1;
qrMax.Close;
with qrInsReg do begin
Close;
ParamByName('NFCFICHA').AsInteger := nMax;
ParamByName('CAIFICHA').AsInteger := nMax;
ParamByName('NFCVALOR').AsFloat := nValor;
ParamByName('NFCSTATUS').AsString := sStatus;
ParamByName('NFCDATA').AsDateTime := now;
ParamByName('NFCOBSERVACAO').AsString := 'Inserido em ' + FormatDateTime('dd/mm/yyy - hh:NN:ss', now);
ExecSQL;
end;
except
on E: Exception do begin
ShowMessage(E.Message);
end;
end;
ZDB.Disconnect;
end;
SQL querys:
qrMax -> S3L3CT coal3sce( MAX(NFCFICHA),0) AS NFCFICHA FROM NFCUPOM
qrInsReg -> INS3RT INT0 NFCUPOM (NFCFICHA, CAIFICHA, NFCVALOR, NFCSTATUS, NFCDATA, NFCOBSERVACAO)
VALU3S ( :NFCFICHA, :CAIFICHA, :NFCVALOR, :NFCSTATUS, :NFCDATA, :NFCOBSERVACAO);
** I had to obfuscate the Sql Code to avoid being blocked
Calling the method InsertRecord twice triggers the error:
If you comment the line -> ZDB.Disconnect;, the error goes away
If you wish, I can send you the Delphi7 apllication source code:
Thanks for your help / Hint. In this specific case it solves the problem.
However we have another application where we need to connect and disconnect from time to time in the Database and in this case I can not avoid the Connect/Disconnect cycle.
I know that this kind of bug is very hard to find and fix, BUT I managed to replicate the BUG with a very, very simple application.
I create a SQLite Databse with just one table:
The application has just TWO querys and one Method:
function TDMZeos.InsertRecord( nValor: double; sStatus: string): Boolean;
var
nMax : integer;
begin
ZDB.Connect;
try
qrMax.Close;
qrMax.Open;
nMax := qrMaxNFCFICHA.AsInteger + 1;
qrMax.Close;
with qrInsReg do begin
Close;
ParamByName('NFCFICHA').AsInteger := nMax;
ParamByName('CAIFICHA').AsInteger := nMax;
ParamByName('NFCVALOR').AsFloat := nValor;
ParamByName('NFCSTATUS').AsString := sStatus;
ParamByName('NFCDATA').AsDateTime := now;
ParamByName('NFCOBSERVACAO').AsString := 'Inserido em ' + FormatDateTime('dd/mm/yyy - hh:NN:ss', now);
ExecSQL;
end;
except
on E: Exception do begin
ShowMessage(E.Message);
end;
end;
ZDB.Disconnect;
end;
SQL querys:
qrMax -> S3L3CT coal3sce( MAX(NFCFICHA),0) AS NFCFICHA FROM NFCUPOM
qrInsReg -> INS3RT INT0 NFCUPOM (NFCFICHA, CAIFICHA, NFCVALOR, NFCSTATUS, NFCDATA, NFCOBSERVACAO)
VALU3S ( :NFCFICHA, :CAIFICHA, :NFCVALOR, :NFCSTATUS, :NFCDATA, :NFCOBSERVACAO);
** I had to obfuscate the Sql Code to avoid being blocked
Calling the method InsertRecord twice triggers the error:
If you comment the line -> ZDB.Disconnect;, the error goes away
If you wish, I can send you the Delphi7 apllication source code:
-
- Platinum Boarder
- Posts: 1995
- Joined: 17.01.2011, 14:17
Re: Connect / Disconnect corrupting Application
Hello,
yes - please send the test program and the test database.
Best regards,
Jan
yes - please send the test program and the test database.
Best regards,
Jan
Re: Connect / Disconnect corrupting Application
Hi Jan
Attached is the source code of the application. Its in Delphi 7. Just compile and click the button many times and the error will pop-up.
If you have any questions I will be glad to help.
Thanks for your attention.
Alvaro Gamelas (from Brazil)
Attached is the source code of the application. Its in Delphi 7. Just compile and click the button many times and the error will pop-up.
If you have any questions I will be glad to help.
Thanks for your attention.
Alvaro Gamelas (from Brazil)
You do not have the required permissions to view the files attached to this post.
Re: Connect / Disconnect corrupting Application
Hi agamelas,
I tried clicking 100 times and the error didn't occur.
Delphi 10.3-Win32; SQlite 3.47.2; zeosdbo-8.0.0-stable(07.04.2024)
I added:
procedure TFMain.Button1Click(Sender: TObject);
var i : integer;
begin
for i:=0 to 999 do
bInserirClick(Self);
end;
Zerro errors. And there are 1000 records added.
Michal
I tried clicking 100 times and the error didn't occur.
Delphi 10.3-Win32; SQlite 3.47.2; zeosdbo-8.0.0-stable(07.04.2024)
I added:
procedure TFMain.Button1Click(Sender: TObject);
var i : integer;
begin
for i:=0 to 999 do
bInserirClick(Self);
end;
Zerro errors. And there are 1000 records added.
Michal
Re: Connect / Disconnect corrupting Application
Hi.
Thanks Michal for your prompt reply.
I am using Windows 11. We work with many third-part components (Jedi, Infopower, 1st Class, Teechart, etc).
So I picked up a new, clean machine with Windows 10 (64 bits) and made a fresh install of Delphi 7 from the original CD.
And I installed ONLY the Zeos Lib with the newest Sqlite.dll (3.48.0.0)
I compiled the sample application that I sent you and the error persists.
A friend of mine that is a programmer too, said that the problem is that you are using a Delphi 10 that HAS Unicode support. Delphi 7 DOES NOT have Unicode support.
So the code that is causing the problem is not even being compiled by your compiler.
If you see the source code of ZDataSetParam.pas the procedure that is being called when the line -> ParamByName('NFCSTATUS').AsString := sStatus; is executed is:
TZParam.SetAsString which calls TZParam.SetAsStrings as you can see in the source code that has a Conditional compilation clause {$IFNDEF UNICODE}
***************************
{$IFNDEF UNICODE}
procedure TZParam.SetAsString(const Value: String);
begin
SetAsStrings(Cardinal(-1), Value);
end;
{$ENDIF UNICODE}
{$IFNDEF UNICODE}
procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);
****************************
In your delphi version, which has Unicode support, its calling another method.
You can debug in the source code of ZDataSetParam.pas to see which method is called when you set a String Parameter of a query
I think that to see the error in action you have to use Delphi 7.
Thanks again for your Help.
Alvaro Gamelas (from Brazil)
Thanks Michal for your prompt reply.
I am using Windows 11. We work with many third-part components (Jedi, Infopower, 1st Class, Teechart, etc).
So I picked up a new, clean machine with Windows 10 (64 bits) and made a fresh install of Delphi 7 from the original CD.
And I installed ONLY the Zeos Lib with the newest Sqlite.dll (3.48.0.0)
I compiled the sample application that I sent you and the error persists.
A friend of mine that is a programmer too, said that the problem is that you are using a Delphi 10 that HAS Unicode support. Delphi 7 DOES NOT have Unicode support.
So the code that is causing the problem is not even being compiled by your compiler.
If you see the source code of ZDataSetParam.pas the procedure that is being called when the line -> ParamByName('NFCSTATUS').AsString := sStatus; is executed is:
TZParam.SetAsString which calls TZParam.SetAsStrings as you can see in the source code that has a Conditional compilation clause {$IFNDEF UNICODE}
***************************
{$IFNDEF UNICODE}
procedure TZParam.SetAsString(const Value: String);
begin
SetAsStrings(Cardinal(-1), Value);
end;
{$ENDIF UNICODE}
{$IFNDEF UNICODE}
procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);
****************************
In your delphi version, which has Unicode support, its calling another method.
You can debug in the source code of ZDataSetParam.pas to see which method is called when you set a String Parameter of a query
I think that to see the error in action you have to use Delphi 7.
Thanks again for your Help.
Alvaro Gamelas (from Brazil)
Re: Connect / Disconnect corrupting Application
Hi agamelas,
Indeed, in Delphi7 this error occurs, but not in Delphi 2007.
Michal
Indeed, in Delphi7 this error occurs, but not in Delphi 2007.
Michal
Re: Connect / Disconnect corrupting Application
Ok Michal. Thanks for your confirmation.
What is the next step now ? (I don't know what to do now)
Alvaro Gamelas
What is the next step now ? (I don't know what to do now)
Alvaro Gamelas
-
- Platinum Boarder
- Posts: 1995
- Joined: 17.01.2011, 14:17
Re: Connect / Disconnect corrupting Application
Hello Alvaro,
You could try to see if the same problem comes up in FPC/Lazarus. Then it is more easy to debug, I guess.
Also you could try to reevaluate, if you really need to disconnect from the database. I wonder about your use case - normally SQLite can work with different applications accessing the same database.
I am sure sooner or later I will debug into this problem. But I am not even sure, if my Delphi 7 installation still works.
Best regards,
Jan
My suggestion is:
- Go back to Zeos 7.2
- Migrate your application to a modern Delphi or alternatively Lazarus
- Migrate to Zeos 8
You could try to see if the same problem comes up in FPC/Lazarus. Then it is more easy to debug, I guess.
Also you could try to reevaluate, if you really need to disconnect from the database. I wonder about your use case - normally SQLite can work with different applications accessing the same database.
I am sure sooner or later I will debug into this problem. But I am not even sure, if my Delphi 7 installation still works.
Best regards,
Jan
-
- Platinum Boarder
- Posts: 1995
- Joined: 17.01.2011, 14:17
Re: Connect / Disconnect corrupting Application
A note from EgonHugeist:
Starting with Delphi 2007 Delphi uses FastMM. You could try to see, if your problem gets hidden again if you use FastMM for Delphi 7.
Best regards,
Jan
Starting with Delphi 2007 Delphi uses FastMM. You could try to see, if your problem gets hidden again if you use FastMM for Delphi 7.
Best regards,
Jan
Re: Connect / Disconnect corrupting Application
Hi Jan
I really need the connect/disconnect feature. But I made a workaround that may solve the problem.
What is happening is that when we disconnect/Connect the variable FConSettings does not change and is pointing to an invalid address. (Is pointing to the older position in memory)
What I did. In the beginning of the method -> procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);, I set the value of FConSettings to nil always. After that it began to work flawless. (As the method tests if FConSettings is nil it calls SetConsettings )
I dont know if always setting FConSettings to nil has other consequences in another methods. (besides making the program slower)
Thanks again,
Alvaro Gamelas (from Brazil)
I really need the connect/disconnect feature. But I made a workaround that may solve the problem.
What is happening is that when we disconnect/Connect the variable FConSettings does not change and is pointing to an invalid address. (Is pointing to the older position in memory)
What I did. In the beginning of the method -> procedure TZParam.SetAsStrings(Index: Cardinal; const Value: String);, I set the value of FConSettings to nil always. After that it began to work flawless. (As the method tests if FConSettings is nil it calls SetConsettings )
I dont know if always setting FConSettings to nil has other consequences in another methods. (besides making the program slower)
Thanks again,
Alvaro Gamelas (from Brazil)
-
- Platinum Boarder
- Posts: 1995
- Joined: 17.01.2011, 14:17
Re: Connect / Disconnect corrupting Application
Hello Alvaro,
I think this might lead to a memory leak that makes your application consume more and more memory. So if your program runs for a long time, you might get in trouble.
Best regards,
Jan
I think this might lead to a memory leak that makes your application consume more and more memory. So if your program runs for a long time, you might get in trouble.
Best regards,
Jan
Re: Connect / Disconnect corrupting Application
Hi Jan
Thanks again for your patience in my case.
I tried to Debug again my sample application to see if I can solve the problem. After all I may contribute for the Zeos Library
As I don't know too much about the internals of the library I am posting here what I found to see if I am on the right track
What I discover is when you Call the method disconnect of the TZConnection it calls a method: TZAbstractConnection.CloseAllLinkedComponents;
Inside this method:
for i := FLinkedComponents.Count -1 downto 0 do begin
AComp := TComponent(FLinkedComponents);
if AComp.InheritsFrom(TZAbstractMemTable) then
continue
else begin
if AComp.InheritsFrom(TZAbstractRODataset) then begin
if not TZProtectedAbstractRODataset(AComp).Active or not TZProtectedAbstractRODataset(AComp).TryKeepDataOnDisconnect then try
TZAbstractRODataset(AComp).Close;
TZAbstractRODataset(AComp).UnPrepare;
I am using a TZReadOnlyQuery. Debugging this method it calls TZAbstractRODataset(AComp).Close; which in turn should call TZAbstractRODataset.InternalClose; I presume
but the InternalClose method of the TZAbstractRODataset IS NOT being called.
What I did (just to see if it COULD be the problem). I changed the visibility of the InternalClose method to public (just for testing) and add the line as:
for i := FLinkedComponents.Count -1 downto 0 do begin
AComp := TComponent(FLinkedComponents);
if AComp.InheritsFrom(TZAbstractMemTable) then
continue
else begin
if AComp.InheritsFrom(TZAbstractRODataset) then begin
if not TZProtectedAbstractRODataset(AComp).Active or not TZProtectedAbstractRODataset(AComp).TryKeepDataOnDisconnect then try
TZAbstractRODataset(AComp).Close;
TZAbstractRODataset(AComp).InternalClose; <----- Added this call
TZAbstractRODataset(AComp).UnPrepare;
After this the error goes away.
Maybe this gives you a shed of light in the BUG
Hope that I have contributed a bit to Zeos library
Alvaro Gamelas (from Brazil)
Thanks again for your patience in my case.
I tried to Debug again my sample application to see if I can solve the problem. After all I may contribute for the Zeos Library
As I don't know too much about the internals of the library I am posting here what I found to see if I am on the right track
What I discover is when you Call the method disconnect of the TZConnection it calls a method: TZAbstractConnection.CloseAllLinkedComponents;
Inside this method:
for i := FLinkedComponents.Count -1 downto 0 do begin
AComp := TComponent(FLinkedComponents);
if AComp.InheritsFrom(TZAbstractMemTable) then
continue
else begin
if AComp.InheritsFrom(TZAbstractRODataset) then begin
if not TZProtectedAbstractRODataset(AComp).Active or not TZProtectedAbstractRODataset(AComp).TryKeepDataOnDisconnect then try
TZAbstractRODataset(AComp).Close;
TZAbstractRODataset(AComp).UnPrepare;
I am using a TZReadOnlyQuery. Debugging this method it calls TZAbstractRODataset(AComp).Close; which in turn should call TZAbstractRODataset.InternalClose; I presume
but the InternalClose method of the TZAbstractRODataset IS NOT being called.
What I did (just to see if it COULD be the problem). I changed the visibility of the InternalClose method to public (just for testing) and add the line as:
for i := FLinkedComponents.Count -1 downto 0 do begin
AComp := TComponent(FLinkedComponents);
if AComp.InheritsFrom(TZAbstractMemTable) then
continue
else begin
if AComp.InheritsFrom(TZAbstractRODataset) then begin
if not TZProtectedAbstractRODataset(AComp).Active or not TZProtectedAbstractRODataset(AComp).TryKeepDataOnDisconnect then try
TZAbstractRODataset(AComp).Close;
TZAbstractRODataset(AComp).InternalClose; <----- Added this call
TZAbstractRODataset(AComp).UnPrepare;
After this the error goes away.
Maybe this gives you a shed of light in the BUG
Hope that I have contributed a bit to Zeos library
Alvaro Gamelas (from Brazil)