ViewVC Help
View File | Revision Log | Show Annotations | Download File | View Changeset | Root Listing
root/public/ibx/trunk/runtime/IBDatabase.pas
(Generate patch)

Comparing ibx/trunk/runtime/IBDatabase.pas (file contents):
Revision 35 by tony, Tue Jan 26 14:38:47 2016 UTC vs.
Revision 143 by tony, Fri Feb 23 12:11:21 2018 UTC

# Line 27 | Line 27
27   {    IBX For Lazarus (Firebird Express)                                  }
28   {    Contributor: Tony Whyman, MWA Software http://www.mwasoftware.co.uk }
29   {    Portions created by MWA Software are copyright McCallum Whyman      }
30 < {    Associates Ltd 2011                                                 }
30 > {    Associates Ltd 2011 - 2018                                               }
31   {                                                                        }
32   {************************************************************************}
33  
# Line 35 | Line 35 | unit IBDatabase;
35  
36   {$Mode Delphi}
37  
38 + {$codepage UTF8}
39 +
40   interface
41  
42   uses
# Line 43 | Line 45 | uses
45   {$ELSE}
46    unix,
47   {$ENDIF}
48 <  SysUtils, Classes, FPTimer, IBHeader, IBExternals, DB,
47 <  IB, CustApp;
48 >  SysUtils, Classes, FPTimer, IBExternals, DB, IB, CustApp, IBTypes;
49  
50   const
51    DPBPrefix = 'isc_dpb_';
# Line 115 | Line 116 | const
116      'set_db_readonly',
117      'set_db_sql_dialect',
118      'gfix_attach',
119 <    'gstat_attach'
120 <  );
119 >    'gstat_attach',
120 >    'set_db_charset',
121 >    'gsec_attach',
122 >    'address_path' ,
123 >    'process_id',
124 >    'no_db_triggers',
125 >    'trusted_auth',
126 >    'process_name',
127 >    'trusted_role',
128 >    'org_filename',
129 >    'utf8_ilename',
130 >    'ext_call_depth',
131 >    'auth_block',
132 >    'client_version',
133 >    'remote_protocol',
134 >    'host_name',
135 >    'os_user',
136 >    'specific_auth_data',
137 >    'auth_plugin_list',
138 >    'auth_plugin_name',
139 >    'config',
140 >    'nolinger',
141 >    'reset_icu',
142 >    'map_attach'
143 >    );
144  
145    TPBPrefix = 'isc_tpb_';
146    TPBConstantNames: array[1..isc_tpb_last_tpb_constant] of string = (
# Line 139 | Line 163 | const
163      'rec_version',
164      'no_rec_version',
165      'restart_requests',
166 <    'no_auto_undo'
166 >    'no_auto_undo',
167 >    'lock_timeout'
168    );
169  
170   type
# Line 155 | Line 180 | type
180    { TIBDatabase }
181    TIBDataBase = class(TCustomConnection)
182    private
183 +    FAttachment: IAttachment;
184 +    FCreateDatabase: boolean;
185 +    FCreateIfNotExists: boolean;
186      FAllowStreamedConnected: boolean;
187      FHiddenPassword: string;
188 <    FIBLoaded: Boolean;
188 >    FOnCreateDatabase: TNotifyEvent;
189      FOnLogin: TIBDatabaseLoginEvent;
190      FSQLHourGlass: Boolean;
191      FTraceFlags: TTraceFlags;
164    FDBSQLDialect: Integer;
192      FSQLDialect: Integer;
193      FOnDialectDowngradeWarning: TNotifyEvent;
167    FCanTimeout: Boolean;
194      FSQLObjects: TList;
195      FTransactions: TList;
196      FDBName: TIBFileName;
197      FDBParams: TStrings;
198      FDBParamsChanged: Boolean;
173    FDPB: PChar;
174    FDPBLength: Short;
175    FHandle: TISC_DB_HANDLE;
176    FHandleIsShared: Boolean;
199      FOnIdleTimer: TNotifyEvent;
200      FDefaultTransaction: TIBTransaction;
201      FInternalTransaction: TIBTransaction;
180    FStreamedConnected: Boolean;
202      FTimer: TFPTimer;
203      FUserNames: TStringList;
204      FDataSets: TList;
205      FLoginCalled: boolean;
206 <    FCharSetSizes: array of integer;
186 <    FCharSetNames: array of string;
206 >    FUseDefaultSystemCodePage: boolean;
207      procedure EnsureInactive;
208 +    function GetAuthenticationMethod: string;
209      function GetDBSQLDialect: Integer;
210 <    function GetSQLDialect: Integer;
210 >    function GetDefaultCharSetID: integer;
211 >    function GetDefaultCharSetName: AnsiString;
212 >    function GetDefaultCodePage: TSystemCodePage;
213 >    function GetRemoteProtocol: string;
214      procedure SetSQLDialect(const Value: Integer);
215      procedure ValidateClientSQLDialect;
216      procedure DBParamsChange(Sender: TObject);
217      procedure DBParamsChanging(Sender: TObject);
218      function GetSQLObject(Index: Integer): TIBBase;
219      function GetSQLObjectCount: Integer;
196    function GetDBParamByDPB(const Idx: Integer): String;
220      function GetIdleTimer: Integer;
221      function GetTransaction(Index: Integer): TIBTransaction;
222      function GetTransactionCount: Integer;
223 <    function Login: Boolean;
201 <    procedure LoadCharSetInfo;
223 >    function Login(var aDatabaseName: string): Boolean;
224      procedure SetDatabaseName(const Value: TIBFileName);
225      procedure SetDBParamByDPB(const Idx: Integer; Value: String);
226      procedure SetDBParams(Value: TStrings);
# Line 229 | Line 251 | type
251      procedure CloseDataSets;
252      procedure CheckActive;
253      procedure CheckInactive;
254 <    procedure CreateDatabase;
254 >    procedure CreateDatabase; overload;
255 >    procedure CreateDatabase(createDatabaseSQL: string); overload;
256      procedure DropDatabase;
257      procedure ForceClose;
258      procedure GetFieldNames(const TableName: string; List: TStrings);
# Line 237 | Line 260 | type
260      function IndexOfDBConst(st: String): Integer;
261      function TestConnected: Boolean;
262      procedure CheckDatabaseName;
240    function Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
263      function AddTransaction(TR: TIBTransaction): Integer;
264      function FindTransaction(TR: TIBTransaction): Integer;
265      function FindDefaultTransaction(): TIBTransaction;
266      procedure RemoveTransaction(Idx: Integer);
267      procedure RemoveTransactions;
246    procedure SetHandle(Value: TISC_DB_HANDLE);
268  
269 <    property Handle: TISC_DB_HANDLE read FHandle;
269 >    property Attachment: IAttachment read FAttachment;
270 >    property DBSQLDialect : Integer read GetDBSQLDialect;
271      property IsReadOnly: Boolean read GetIsReadOnly;
250    property DBParamByDPB[const Idx: Integer]: String read GetDBParamByDPB
251                                                      write SetDBParamByDPB;
272      property SQLObjectCount: Integer read GetSQLObjectCount;
273      property SQLObjects[Index: Integer]: TIBBase read GetSQLObject;
254    property HandleIsShared: Boolean read FHandleIsShared;
274      property TransactionCount: Integer read GetTransactionCount;
275      property Transactions[Index: Integer]: TIBTransaction read GetTransaction;
276      property InternalTransaction: TIBTransaction read FInternalTransaction;
277 +    property DefaultCharSetName: AnsiString read GetDefaultCharSetName;
278 +    property DefaultCharSetID: integer read GetDefaultCharSetID;
279 +    property DefaultCodePage: TSystemCodePage read GetDefaultCodePage;
280 +    property AuthenticationMethod: string read GetAuthenticationMethod;
281 +    property RemoteProtocol: string read GetRemoteProtocol;
282  
283    published
284      property Connected;
285 +    property CreateIfNotExists: boolean read FCreateIfNotExists write FCreateIfNotExists;
286      property AllowStreamedConnected: boolean read FAllowStreamedConnected
287               write FAllowStreamedConnected;
288      property DatabaseName: TIBFileName read FDBName write SetDatabaseName;
# Line 266 | Line 291 | type
291      property DefaultTransaction: TIBTransaction read FDefaultTransaction
292                                                   write SetDefaultTransaction;
293      property IdleTimer: Integer read GetIdleTimer write SetIdleTimer;
294 <    property SQLDialect : Integer read GetSQLDialect write SetSQLDialect default 3;
294 >    property SQLDialect : Integer read FSQLDialect write SetSQLDialect default 3;
295      property SQLHourGlass: Boolean read FSQLHourGlass write FSQLHourGlass default true;
271    property DBSQLDialect : Integer read FDBSQLDialect;
296      property TraceFlags: TTraceFlags read FTraceFlags write FTraceFlags;
297 +    property UseDefaultSystemCodePage: boolean read FUseDefaultSystemCodePage
298 +                                               write FUseDefaultSystemCodePage;
299      property AfterConnect;
300      property AfterDisconnect;
301      property BeforeConnect;
302      property BeforeDisconnect;
303 +    property OnCreateDatabase: TNotifyEvent read FOnCreateDatabase write FOnCreateDatabase;
304      property OnLogin: TIBDatabaseLoginEvent read FOnLogin write FOnLogin;
305      property OnIdleTimer: TNotifyEvent read FOnIdleTimer write FOnIdleTimer;
306      property OnDialectDowngradeWarning: TNotifyEvent read FOnDialectDowngradeWarning write FOnDialectDowngradeWarning;
307    end;
308  
309 <  { TIBTransaction }
309 >  TDefaultEndAction = TARollback..TACommit;
310  
311 <  TTransactionAction         = (TARollback, TACommit, TARollbackRetaining, TACommitRetaining);
311 >  { TIBTransaction }
312  
313    TIBTransaction = class(TComponent)
314    private
315 +    FTransactionIntf: ITransaction;
316      FAfterDelete: TNotifyEvent;
317      FAfterEdit: TNotifyEvent;
318      FAfterExecQuery: TNotifyEvent;
# Line 292 | Line 320 | type
320      FAfterPost: TNotifyEvent;
321      FAfterTransactionEnd: TNotifyEvent;
322      FBeforeTransactionEnd: TNotifyEvent;
295    FIBLoaded: Boolean;
296    FCanTimeout         : Boolean;
323      FDatabases          : TList;
324      FOnStartTransaction: TNotifyEvent;
325      FSQLObjects         : TList;
326      FDefaultDatabase    : TIBDatabase;
301    FHandle             : TISC_TR_HANDLE;
302    FHandleIsShared     : Boolean;
327      FOnIdleTimer          : TNotifyEvent;
328      FStreamedActive     : Boolean;
329 <    FTPB                : PChar;
306 <    FTPBLength          : Short;
329 >    FTPB                : ITPB;
330      FTimer              : TFPTimer;
331 <    FDefaultAction      : TTransactionAction;
331 >    FDefaultAction      : TDefaultEndAction;
332      FTRParams           : TStrings;
333      FTRParamsChanged    : Boolean;
334      FInEndTransaction   : boolean;
# Line 328 | Line 351 | type
351      function GetIdleTimer: Integer;
352      procedure BeforeDatabaseDisconnect(DB: TIBDatabase);
353      procedure SetActive(Value: Boolean);
331    procedure SetDefaultAction(Value: TTransactionAction);
354      procedure SetDefaultDatabase(Value: TIBDatabase);
355      procedure SetIdleTimer(Value: Integer);
356      procedure SetTRParams(Value: TStrings);
# Line 341 | Line 363 | type
363  
364    protected
365      procedure Loaded; override;
344    procedure SetHandle(Value: TISC_TR_HANDLE);
366      procedure Notification( AComponent: TComponent; Operation: TOperation); override;
367  
368    public
369      constructor Create(AOwner: TComponent); override;
370      destructor Destroy; override;
350    function Call(ErrCode: ISC_STATUS; RaiseError: Boolean): ISC_STATUS;
371      procedure Commit;
372      procedure CommitRetaining;
373      procedure Rollback;
# Line 368 | Line 388 | type
388      property Databases[Index: Integer]: TIBDatabase read GetDatabase;
389      property SQLObjectCount: Integer read GetSQLObjectCount;
390      property SQLObjects[Index: Integer]: TIBBase read GetSQLObject;
371    property Handle: TISC_TR_HANDLE read FHandle;
372    property HandleIsShared: Boolean read FHandleIsShared;
391      property InTransaction: Boolean read GetInTransaction;
392 <    property TPB: PChar read FTPB;
393 <    property TPBLength: Short read FTPBLength;
392 >    property TransactionIntf: ITransaction read FTransactionIntf;
393 >    property TPB: ITPB read FTPB;
394    published
395      property Active: Boolean read GetInTransaction write SetActive;
396      property DefaultDatabase: TIBDatabase read FDefaultDatabase
397                                             write SetDefaultDatabase;
398      property IdleTimer: Integer read GetIdleTimer write SetIdleTimer default 0;
399 <    property DefaultAction: TTransactionAction read FDefaultAction write SetDefaultAction default taCommit;
399 >    property DefaultAction: TDefaultEndAction read FDefaultAction write FDefaultAction default taCommit;
400      property Params: TStrings read FTRParams write SetTRParams;
401      property OnIdleTimer: TNotifyEvent read FOnIdleTimer write FOnIdleTimer;
402      property BeforeTransactionEnd: TNotifyEvent read FBeforeTransactionEnd
# Line 429 | Line 447 | type
447      procedure DoBeforeTransactionEnd(Action: TTransactionAction); virtual;
448      procedure DoAfterTransactionEnd; virtual;
449      procedure DoTransactionFree; virtual;
432    function GetDBHandle: PISC_DB_HANDLE; virtual;
433    function GetTRHandle: PISC_TR_HANDLE; virtual;
450      procedure SetDatabase(Value: TIBDatabase); virtual;
451      procedure SetTransaction(Value: TIBTransaction); virtual;
452    public
# Line 443 | Line 459 | type
459      procedure DoAfterDelete(Sender: TObject); virtual;
460      procedure DoAfterInsert(Sender: TObject); virtual;
461      procedure DoAfterPost(Sender: TObject); virtual;
446    function GetCharSetSize(CharSetID: integer): integer;
447    function GetDefaultCharSetSize: integer;
448    function GetCharSetName(CharSetID: integer): string;
449    function GetDefaultCharSetName: string;
462      procedure HandleException(Sender: TObject);
463      procedure SetCursor;
464      procedure RestoreCursor;
# Line 465 | Line 477 | type
477      property OnTransactionFree: TNotifyEvent read FOnTransactionFree write FOnTransactionFree;
478      property Database: TIBDatabase read FDatabase
479                                      write SetDatabase;
468    property DBHandle: PISC_DB_HANDLE read GetDBHandle;
480      property Owner: TObject read FOwner;
470    property TRHandle: PISC_TR_HANDLE read GetTRHandle;
481      property Transaction: TIBTransaction read FTransaction
482                                            write SetTransaction;
483    end;
484  
485 < procedure GenerateDPB(sl: TStrings; var DPB: string; var DPBLength: Short);
486 < procedure GenerateTPB(sl: TStrings; var TPB: string; var TPBLength: Short);
485 > function GenerateDPB(sl: TStrings): IDPB;
486 > function GenerateTPB(sl: TStrings): ITPB;
487  
488  
489   implementation
490  
491 < uses IBIntf, IBSQLMonitor, IBCustomDataSet, IBDatabaseInfo, IBSQL, IBUtils,
492 <     typInfo;
491 > uses  IBSQLMonitor, IBCustomDataSet, IBDatabaseInfo, IBSQL, IBUtils,
492 >     typInfo, FBMessages, IBErrorCodes {$IFDEF WINDOWS}, Windirs {$ENDIF};
493  
494   { TIBDatabase }
495  
496 < constructor TIBDataBase.Create(AOwner: TComponent);
487 < {$ifdef WINDOWS}
488 < var acp: uint;
489 < {$endif}
496 > constructor TIBDataBase.Create(AOwner: TComponent);
497   begin
498    inherited Create(AOwner);
492  FIBLoaded := False;
493  CheckIBLoaded;
494  FIBLoaded := True;
499    LoginPrompt := True;
500    FSQLObjects := TList.Create;
501    FTransactions := TList.Create;
# Line 502 | Line 506 | begin
506       (AOwner is TCustomApplication) and
507       TCustomApplication(AOWner).ConsoleApplication then
508      LoginPrompt := false;
505  {$ifdef UNIX}
506  if csDesigning in ComponentState then
507    FDBParams.Add('lc_ctype=UTF8');
508  {$else}
509  {$ifdef WINDOWS}
510  if csDesigning in ComponentState then
511  begin
512    acp := GetACP;
513    if (acp >= 1250) and (acp <= 1254) then
514      FDBParams.Values['lc_ctype'] := Format('WIN%d',[acp]);
515  end;
516  {$endif}
517  {$endif}
509    FDBParamsChanged := True;
510    TStringList(FDBParams).OnChange := DBParamsChange;
511    TStringList(FDBParams).OnChanging := DBParamsChanging;
521  FDPB := nil;
522  FHandle := nil;
512    FUserNames := nil;
513    FInternalTransaction := TIBTransaction.Create(self);
514    FInternalTransaction.DefaultDatabase := Self;
# Line 527 | Line 516 | begin
516    FTimer.Enabled := False;
517    FTimer.Interval := 0;
518    FTimer.OnTimer := TimeoutConnection;
530  FDBSQLDialect := 1;
519    FSQLDialect := 3;
520    FTraceFlags := [];
521    FDataSets := TList.Create;
522    CheckStreamConnect;
523   end;
524  
525 < destructor TIBDataBase.Destroy;
525 > destructor TIBDataBase.Destroy;
526   var
527    i: Integer;
528   begin
529 <  if FIBLoaded then
530 <  begin
531 <    IdleTimer := 0;
532 <    if FHandle <> nil then
533 <      ForceClose;
534 <    for i := 0 to FSQLObjects.Count - 1 do
535 <      if FSQLObjects[i] <> nil then
536 <        SQLObjects[i].DoDatabaseFree;
537 <    RemoveSQLObjects;
538 <    RemoveTransactions;
539 <    FInternalTransaction.Free;
540 <    FreeMem(FDPB);
541 <    FDPB := nil;
554 <    FDBParams.Free;
555 <    FSQLObjects.Free;
556 <    FUserNames.Free;
557 <    FTransactions.Free;
558 <  end;
529 >  IdleTimer := 0;
530 >  if FAttachment <> nil then
531 >    ForceClose;
532 >  for i := 0 to FSQLObjects.Count - 1 do
533 >    if FSQLObjects[i] <> nil then
534 >      SQLObjects[i].DoDatabaseFree;
535 >  RemoveSQLObjects;
536 >  RemoveTransactions;
537 >  FInternalTransaction.Free;
538 >  FDBParams.Free;
539 >  FSQLObjects.Free;
540 >  FUserNames.Free;
541 >  FTransactions.Free;
542    FDataSets.Free;
543    inherited Destroy;
544   end;
545  
563 function TIBDataBase.Call(ErrCode: ISC_STATUS; RaiseError: Boolean
564   ): ISC_STATUS;
565 begin
566  result := ErrCode;
567  FCanTimeout := False;
568  if RaiseError and (ErrCode > 0) then
569    IBDataBaseError;
570 end;
571
546   procedure TIBDataBase.CheckActive;
547   begin
548    if StreamedConnected and (not Connected) then
549      Loaded;
550 <  if FHandle = nil then
550 >  if FAttachment = nil then
551      IBError(ibxeDatabaseClosed, [nil]);
552   end;
553  
# Line 581 | Line 555 | end;
555   begin
556    if csDesigning in ComponentState then
557    begin
558 <    if FHandle <> nil then
558 >    if FAttachment <> nil then
559        Close;
560    end
561   end;
562  
563 + function TIBDataBase.GetAuthenticationMethod: string;
564 + begin
565 +  CheckActive;
566 +  Result := Attachment.GetAuthenticationMethod;
567 + end;
568 +
569   procedure TIBDataBase.CheckInactive;
570   begin
571 <  if FHandle <> nil then
571 >  if FAttachment <> nil then
572      IBError(ibxeDatabaseOpen, [nil]);
573   end;
574  
575   procedure TIBDataBase.CheckDatabaseName;
576   begin
577 <  if (FDBName = '') then
577 >  if (Trim(FDBName) = '') then
578      IBError(ibxeDatabaseNameMissing, [nil]);
579   end;
580  
# Line 632 | Line 612 | end;
612   begin
613    if Connected then
614      InternalClose(False);
635  FDBSQLDialect := 1;
636  SetLength(FCharSetSizes,0);
637  SetLength(FCharSetNames,0);
615   end;
616  
617 < procedure TIBDataBase.CreateDatabase;
641 < var
642 <  tr_handle: TISC_TR_HANDLE;
617 >  procedure TIBDataBase.CreateDatabase;
618   begin
619    CheckInactive;
620 <  tr_handle := nil;
621 <  Call(
622 <    isc_dsql_execute_immediate(StatusVector, @FHandle, @tr_handle, 0,
623 <                               PChar('CREATE DATABASE ''' + FDBName + ''' ' + {do not localize}
624 <                               Params.Text), SQLDialect, nil),
625 <    True);
620 >  CheckDatabaseName;
621 >  FCreateDatabase := true;
622 >  Connected := true;
623 > end;
624 >
625 > procedure TIBDataBase.CreateDatabase(createDatabaseSQL: string);
626 > begin
627 >  CheckInactive;
628 >  FAttachment := FirebirdAPI.CreateDatabase(createDatabaseSQL,SQLDialect);
629 >  FDBName := Attachment.GetConnectString;
630 >  if assigned(FOnCreateDatabase) and (FAttachment <> nil) then
631 >    OnCreateDatabase(self);
632   end;
633  
634   procedure TIBDataBase.DropDatabase;
635   begin
636    CheckActive;
637 <  Call(isc_drop_database(StatusVector, @FHandle), True);
637 >  FAttachment.DropDatabase;
638 >  FAttachment := nil;
639   end;
640  
641   procedure TIBDataBase.DBParamsChange(Sender: TObject);
# Line 680 | Line 662 | begin
662      end;
663   end;
664  
665 < function TIBDataBase.FindDefaultTransaction: TIBTransaction;
665 >  function TIBDataBase.FindDefaultTransaction(): TIBTransaction;
666   var
667    i: Integer;
668   begin
# Line 706 | Line 688 | end;
688  
689   function TIBDataBase.GetConnected: Boolean;
690   begin
691 <  result := FHandle <> nil;
691 >  result := (FAttachment <> nil) and FAttachment.IsConnected;
692   end;
693  
694   function TIBDataBase.GetSQLObject(Index: Integer): TIBBase;
# Line 723 | Line 705 | begin
705      Inc(result);
706   end;
707  
708 < function TIBDataBase.GetDBParamByDPB( const Idx: Integer): String;
727 < var
728 <  ConstIdx, EqualsIdx: Integer;
729 < begin
730 <  if (Idx > 0) and (Idx <= isc_dpb_last_dpb_constant) then
731 <  begin
732 <    ConstIdx := IndexOfDBConst(DPBConstantNames[Idx]);
733 <    if ConstIdx = -1 then
734 <      result := ''
735 <    else
736 <    begin
737 <      result := Params[ConstIdx];
738 <      EqualsIdx := Pos('=', result); {mbcs ok}
739 <      if EqualsIdx = 0 then
740 <        result := ''
741 <      else
742 <        result := Copy(result, EqualsIdx + 1, Length(result));
743 <    end;
744 <  end
745 <  else
746 <    result := '';
747 < end;
748 <
749 < function TIBDataBase.GetIdleTimer: Integer;
708 > function TIBDataBase.GetIdleTimer: Integer;
709   begin
710    result := FTimer.Interval;
711   end;
# Line 811 | Line 770 | begin
770      end;
771    end;
772  
773 <  if (not HandleIsShared) and
774 <     (Call(isc_detach_database(StatusVector, @FHandle), False) > 0) and
816 <     (not Force) then
817 <    IBDataBaseError
818 <  else
819 <  begin
820 <    FHandle := nil;
821 <    FHandleIsShared := False;
822 <  end;
773 >  FAttachment.Disconnect(Force);
774 >  FAttachment := nil;
775  
776    if not (csDesigning in ComponentState) then
777      MonitorHook.DBDisconnect(Self);
# Line 829 | Line 781 | begin
781        SQLObjects[i].DoAfterDatabaseDisconnect;
782   end;
783  
832 procedure TIBDataBase.LoadCharSetInfo;
833 var Query: TIBSQL;
834    i: integer;
835 begin
836  if not FInternalTransaction.Active then
837    FInternalTransaction.StartTransaction;
838  Query := TIBSQL.Create(self);
839  try
840    Query.Database := Self;
841    Query.Transaction := FInternalTransaction;
842    Query.SQL.Text := 'Select RDB$CHARACTER_SET_ID, RDB$BYTES_PER_CHARACTER, RDB$CHARACTER_SET_NAME ' +
843                      'From RDB$CHARACTER_SETS Order by 1 DESC'; {do not localize}
844    Query.Prepare;
845    Query.ExecQuery;
846    if not Query.EOF then
847    begin
848      SetLength(FCharSetSizes,Query.FieldByName('RDB$CHARACTER_SET_ID').AsInteger + 1);
849      SetLength(FCharSetNames,Query.FieldByName('RDB$CHARACTER_SET_ID').AsInteger + 1);
850      for i := 0 to Length(FCharSetSizes) - 1 do FCharSetSizes[i] := 1;
851      repeat
852        FCharSetSizes[Query.FieldByName('RDB$CHARACTER_SET_ID').AsInteger] :=
853                 Query.FieldByName('RDB$BYTES_PER_CHARACTER').AsInteger;
854        FCharSetNames[Query.FieldByName('RDB$CHARACTER_SET_ID').AsInteger] :=
855                 Query.FieldByName('RDB$CHARACTER_SET_NAME').AsString;
856        Query.Next;
857      until Query.EOF;
858    end;
859  finally
860    Query.free;
861    FInternalTransaction.Commit;
862  end;
863 end;
864
784   procedure TIBDataBase.CheckStreamConnect;
785   var
786    i: integer;
# Line 884 | Line 803 | begin
803           (FDefaultTransaction.FStreamedActive) and
804           (not FDefaultTransaction.InTransaction) then
805          FDefaultTransaction.StartTransaction;
806 <      FStreamedConnected := False;
806 >      StreamedConnected := False;
807      end;
808    except
809      if csDesigning in ComponentState then
# Line 925 | Line 844 | begin
844    end;
845   end;
846  
847 < function TIBDataBase.Login: Boolean;
847 > function TIBDataBase.Login(var aDatabaseName: string): Boolean;
848   var
849    IndexOfUser, IndexOfPassword: Integer;
850    Username, Password, OldPassword: String;
# Line 961 | Line 880 | begin
880        LoginParams.Assign(Params);
881        FOnLogin(Self, LoginParams);
882        Params.Assign (LoginParams);
883 +      aDatabaseName := FDBName;
884        HidePassword;
885      finally
886        LoginParams.Free;
# Line 982 | Line 902 | begin
902                                           Length(Params[IndexOfPassword]));
903        OldPassword := password;
904      end;
905 <    result := IBGUIInterface.LoginDialogEx(DatabaseName, Username, Password, False);
905 >
906 >    result := IBGUIInterface.LoginDialogEx(aDatabaseName, Username, Password, False);
907      if result then
908      begin
909 <      if IndexOfUser = -1 then
910 <        Params.Add(DPBConstantNames[isc_dpb_user_name] + '=' + Username)
911 <      else
912 <        Params[IndexOfUser] := DPBConstantNames[isc_dpb_user_name] +
909 >      if Username <> '' then
910 >      begin
911 >        if IndexOfUser = -1 then
912 >          Params.Add(DPBConstantNames[isc_dpb_user_name] + '=' + Username)
913 >        else
914 >          Params[IndexOfUser] := DPBConstantNames[isc_dpb_user_name] +
915                                   '=' + Username;
916 +      end
917 +      else
918 +      if IndexOfUser <> -1 then
919 +        Params.Delete(IndexOfUser);
920        if (Password = OldPassword) then
921          FHiddenPassword := ''
922        else
# Line 1008 | Line 935 | begin
935    end;
936   end;
937  
938 < procedure TIBDataBase.DoConnect;
938 > procedure TIBDataBase.DoConnect;
939 >
940 >  function ExpandDBName(aDBName: string): string;
941 >  const
942 >    TmpPrefix = '$TEMP$';
943 >    DataPrefix = '$DATADIR$';
944 >  var
945 >    LocalDirName: string;
946 >  begin
947 >    if Pos(TmpPrefix,aDBName) = 1 then
948 >    begin
949 >      system.Delete(aDBName,1,Length(TmpPrefix));
950 >      Result := GetTempDir + aDBName
951 >    end
952 >    else
953 >    if Pos(DataPrefix,aDBName) = 1 then
954 >    begin
955 >      system.Delete(aDBName,1,Length(DataPrefix));
956 >      if Sysutils.VendorName <> '' then
957 >        LocalDirName :=  Sysutils.VendorName
958 >      else
959 >        LocalDirName :=  'IBX';
960 >      {$IFDEF UNIX}
961 >      LocalDirName := GetUserDir + '.' + LocalDirName;
962 >      {$ENDIF}
963 >      {$IFDEF WINDOWS}
964 >      LocalDirName := GetWindowsSpecialDir(CSIDL_LOCAL_APPDATA) + LocalDirName;
965 >      {$ENDIF}
966 >      CreateDir(LocalDirName);
967 >      Result := LocalDirName + DirectorySeparator + aDBName;
968 >    end
969 >    else
970 >      Result := aDBName;
971 >  end;
972 >
973   var
1013  DPB: String;
974    TempDBParams: TStrings;
975    I: integer;
976    aDBName: string;
977 +  Status: IStatus;
978 +  CharSetID: integer;
979 +  CharSetName: AnsiString;
980 +  DPB: IDPB;
981 +  PW: IDPBItem;
982   begin
983 +  DPB := nil;
984    CheckInactive;
985    CheckDatabaseName;
986    if (not LoginPrompt) and (FHiddenPassword <> '') then
# Line 1023 | Line 989 | begin
989      FDBParamsChanged := True;
990    end;
991    { Use builtin login prompt if requested }
992 <  if (LoginPrompt or (csDesigning in ComponentState)) and not Login then
992 >  aDBName := ExpandDBName(FDBName);
993 >
994 >  if (LoginPrompt or (csDesigning in ComponentState)) and not Login(aDBName) then
995      IBError(ibxeOperationCancelled, [nil]);
996  
997    TempDBParams := TStringList.Create;
998    try
999     TempDBParams.Assign(FDBParams);
1000 <   aDBName := FDBName;
1000 >   {$ifdef UNIX}
1001 >   {See below for WINDOWS UseDefaultSystemCodePage}
1002 >   if UseDefaultSystemCodePage then
1003 >     TempDBParams.Values['lc_ctype'] :='UTF8';
1004 >   {$endif}
1005     {Opportunity to override defaults}
1006     for i := 0 to FSQLObjects.Count - 1 do
1007     begin
# Line 1037 | Line 1009 | begin
1009           SQLObjects[i].DoBeforeDatabaseConnect(TempDBParams,aDBName);
1010     end;
1011  
1012 <   { Generate a new DPB if necessary }
1013 <   if (FDBParamsChanged or (TempDBParams.Text <> FDBParams.Text)) then
1014 <   begin
1015 <     FDBParamsChanged := False;
1016 <     if (not LoginPrompt and not (csDesigning in ComponentState)) or (FHiddenPassword = '') then
1017 <       GenerateDPB(TempDBParams, DPB, FDPBLength)
1012 >   repeat
1013 >     { Generate a new DPB if necessary }
1014 >     if (DPB = nil) or FDBParamsChanged or (TempDBParams.Text <> FDBParams.Text) then
1015 >     begin
1016 >       FDBParamsChanged := False;
1017 >       if (not LoginPrompt and not (csDesigning in ComponentState)) or (FHiddenPassword = '') then
1018 >         DPB := GenerateDPB(TempDBParams)
1019 >       else
1020 >       begin
1021 >          TempDBParams.Values['password'] := FHiddenPassword;
1022 >          DPB := GenerateDPB(TempDBParams);
1023 >       end;
1024 >     end;
1025 >
1026 >     if FCreateDatabase then
1027 >     begin
1028 >       FCreateDatabase := false;
1029 >       DPB.Add(isc_dpb_set_db_SQL_dialect).AsByte := SQLDialect; {create with this SQL Dialect}
1030 >       FAttachment := FirebirdAPI.CreateDatabase(aDBName,DPB, false);
1031 >       if FAttachment = nil then
1032 >         DPB := nil;
1033 >       if assigned(FOnCreateDatabase) and (FAttachment <> nil) then
1034 >         OnCreateDatabase(self);
1035 >     end
1036       else
1037 +       FAttachment := FirebirdAPI.OpenDatabase(aDBName,DPB,false);
1038 +
1039 +     if FAttachment = nil then
1040       begin
1041 <        TempDBParams.Add('password=' + FHiddenPassword);
1042 <        GenerateDPB(TempDBParams, DPB, FDPBLength);
1041 >       Status := FirebirdAPI.GetStatus;
1042 >       {$IFDEF UNIX}
1043 >       if GetProtocol(aDBName) = Local then
1044 >       begin
1045 >           if ((Status.GetSQLCode = -901) and (Status.GetIBErrorCode = isc_random)) {Access permissions on firebird temp}
1046 >              or
1047 >              ((Status.GetSQLCode = -902) and (Status.GetIBErrorCode = isc_sys_request)) {Security DB Problem}
1048 >              or
1049 >              ((Status.GetSQLCode = -902) and (Status.GetIBErrorCode = isc_psw_attach)) {Security DB Problem}
1050 >              or
1051 >              ((Status.GetSQLCode = -904) and (Status.GetIBErrorCode = isc_lock_dir_access)) {Lock File Problem}
1052 >              then
1053 >              begin
1054 >                aDBName := 'localhost:' + aDBName;
1055 >                Continue;
1056 >             end
1057 >       end;
1058 >       {$ENDIF}
1059 >       if ((Status.GetSQLCode = -902) and (Status.GetIBErrorCode = isc_io_error)) {Database not found}
1060 >                        and CreateIfNotExists and not (csDesigning in ComponentState) then
1061 >         FCreateDatabase := true
1062 >       else
1063 >         raise EIBInterBaseError.Create(Status);
1064       end;
1065 <     IBAlloc(FDPB, 0, FDPBLength);
1066 <     Move(DPB[1], FDPB[0], FDPBLength);
1067 <   end;
1065 >
1066 >     if UseDefaultSystemCodePage and (FAttachment <> nil) then
1067 >     {Only now can we check the codepage in use by the Attachment.
1068 >      If not that required then re-open with required LCLType.}
1069 >     begin
1070 >       {$ifdef WINDOWS}
1071 >       if Attachment.CodePage2CharSetID(GetACP,CharSetID) then
1072 >       {$else}
1073 >       if Attachment.CodePage2CharSetID(DefaultSystemCodePage,CharSetID) then
1074 >       {$endif}
1075 >       begin
1076 >         CharSetName := Attachment.GetCharsetName(CharSetID);
1077 >         if CharSetName <> AnsiUpperCase(TempDBParams.Values['lc_ctype']) then
1078 >         begin
1079 >           TempDBParams.Values['lc_ctype'] := CharSetName;
1080 >           FDBParamsChanged := True;
1081 >           FAttachment := nil;
1082 >         end
1083 >       end
1084 >     end;
1085 >
1086 >   until FAttachment <> nil;
1087 >
1088    finally
1089     TempDBParams.Free;
1090    end;
1091 <  if Call(isc_attach_database(StatusVector, Length(aDBName),
1092 <                         PChar(aDBName), @FHandle,
1093 <                         FDPBLength, FDPB), False) > 0 then
1060 <  begin
1061 <    FHandle := nil;
1062 <    IBDataBaseError;
1063 <  end;
1091 >  PW := Attachment.getDPB.Find(isc_dpb_password);
1092 >  if PW <> nil then PW.AsString := 'xxxxxxxx'; {Hide password}
1093 >
1094    if not (csDesigning in ComponentState) then
1095      FDBName := aDBName; {Synchronise at run time}
1066  FDBSQLDialect := GetDBSQLDialect;
1096    ValidateClientSQLDialect;
1097    for i := 0 to FSQLObjects.Count - 1 do
1098    begin
# Line 1072 | Line 1101 | begin
1101    end;
1102    if not (csDesigning in ComponentState) then
1103      MonitorHook.DBConnect(Self);
1075  LoadCharSetInfo;
1104   end;
1105  
1106   procedure TIBDataBase.RemoveSQLObject(Idx: Integer);
# Line 1175 | Line 1203 | begin
1203    FDefaultTransaction := Value;
1204   end;
1205  
1178 procedure TIBDataBase.SetHandle(Value: TISC_DB_HANDLE);
1179 begin
1180  if HandleIsShared then
1181    Close
1182  else
1183    CheckInactive;
1184  FHandle := Value;
1185  FHandleIsShared := (Value <> nil);
1186 end;
1187
1206   procedure TIBDataBase.SetIdleTimer(Value: Integer);
1207   begin
1208    if Value < 0 then
# Line 1229 | Line 1247 | end;
1247   begin
1248    if Connected then
1249    begin
1250 <    if FCanTimeout then
1250 >    if not FAttachment.HasActivity then
1251      begin
1252        ForceClose;
1253        if Assigned(FOnIdleTimer) then
1254          FOnIdleTimer(Self);
1255      end
1238    else
1239      FCanTimeout := True;
1256    end;
1257   end;
1258  
# Line 1258 | Line 1274 | begin
1274    DatabaseInfo.Free;
1275   end;
1276  
1261 function TIBDataBase.GetSQLDialect: Integer;
1262 begin
1263  Result := FSQLDialect;
1264 end;
1265
1277  
1278   procedure TIBDataBase.SetSQLDialect( const Value: Integer);
1279   begin
1280    if (Value < 1) then IBError(ibxeSQLDialectInvalid, [nil]);
1281 <  if ((FHandle = nil) or (Value <= FDBSQLDialect))  then
1281 >  if (Attachment = nil) or (Value <= DBSQLDialect)  then
1282      FSQLDialect := Value
1283    else
1284      IBError(ibxeSQLDialectInvalid, [nil]);
1285   end;
1286  
1287   function TIBDataBase.GetDBSQLDialect: Integer;
1277 var
1278  DatabaseInfo: TIBDatabaseInfo;
1288   begin
1289 <  DatabaseInfo := TIBDatabaseInfo.Create(self);
1290 <  DatabaseInfo.Database := self;
1291 <  result := DatabaseInfo.DBSQLDialect;
1292 <  DatabaseInfo.Free;
1289 >  CheckActive;
1290 >  Result := Attachment.GetSQLDialect;
1291 > end;
1292 >
1293 > function TIBDataBase.GetDefaultCharSetID: integer;
1294 > begin
1295 >  if (Attachment <> nil) and Attachment.HasDefaultCharSet then
1296 >    Result := Attachment.GetDefaultCharSetID
1297 >  else
1298 >    Result := 0;
1299 > end;
1300 >
1301 > function TIBDataBase.GetDefaultCharSetName: AnsiString;
1302 > begin
1303 >  if Attachment <> nil then
1304 >    Result := Attachment.GetCharsetName(DefaultCharSetID)
1305 >  else
1306 >    Result := '';
1307 > end;
1308 >
1309 > function TIBDataBase.GetDefaultCodePage: TSystemCodePage;
1310 > begin
1311 >  if Attachment <> nil then
1312 >    Attachment.CharSetID2CodePage(DefaultCharSetID,Result)
1313 >  else
1314 >    Result := CP_NONE;
1315 > end;
1316 >
1317 > function TIBDataBase.GetRemoteProtocol: string;
1318 > begin
1319 >  CheckActive;
1320 >  Result := Attachment.GetRemoteProtocol;
1321   end;
1322  
1323   procedure TIBDataBase.ValidateClientSQLDialect;
1324   begin
1325 <  if (FDBSQLDialect < FSQLDialect) then
1325 >  if (DBSQLDialect < FSQLDialect) then
1326    begin
1327 <    FSQLDialect := FDBSQLDialect;
1327 >    FSQLDialect := DBSQLDialect;
1328      if Assigned (FOnDialectDowngradeWarning) then
1329        FOnDialectDowngradeWarning(self);
1330    end;
# Line 1373 | Line 1410 | begin
1410      Query.Database := Self;
1411      Query.Transaction := FInternalTransaction;
1412      Query.SQL.Text := 'Select R.RDB$FIELD_NAME ' + {do not localize}
1413 <      'from RDB$RELATION_FIELDS R, RDB$FIELDS F ' + {do not localize}
1413 >      'from RDB$RELATION_FIELDS R ' + {do not localize}
1414        'where R.RDB$RELATION_NAME = ' + {do not localize}
1415 <      '''' +
1416 <      FormatIdentifierValue(SQLDialect, TableName) +
1380 <      ''' ' +
1381 <      'and R.RDB$FIELD_SOURCE = F.RDB$FIELD_NAME '; {do not localize}
1415 >      '''' + ExtractIdentifier(DBSQLDialect, TableName) +
1416 >      ''' and Exists(Select * From RDB$FIELDS F Where R.RDB$FIELD_SOURCE = F.RDB$FIELD_NAME)' ; {do not localize}
1417      Query.Prepare;
1418      Query.ExecQuery;
1419      with List do
# Line 1386 | Line 1421 | begin
1421        BeginUpdate;
1422        try
1423          Clear;
1424 <        while (not Query.EOF) and (Query.Next <> nil) do
1425 <          List.Add(TrimRight(Query.Current.ByName('RDB$FIELD_NAME').AsString)); {do not localize}
1424 >        while (not Query.EOF) and Query.Next  do
1425 >          List.Add(TrimRight(Query.FieldByName('RDB$FIELD_NAME').AsString)); {do not localize}
1426        finally
1427          EndUpdate;
1428        end;
# Line 1426 | Line 1461 | begin
1461          BeginUpdate;
1462          try
1463            Clear;
1464 <          while (not Query.EOF) and (Query.Next <> nil) do
1465 <            List.Add(TrimRight(Query.Current[0].AsString));
1464 >          while (not Query.EOF) and Query.Next  do
1465 >            List.Add(TrimRight(Query.Fields[0].AsString));
1466          finally
1467            EndUpdate;
1468          end;
# Line 1444 | Line 1479 | end;
1479   constructor TIBTransaction.Create(AOwner: TComponent);
1480   begin
1481    inherited Create(AOwner);
1447  FIBLoaded := False;
1448  CheckIBLoaded;
1449  FIBLoaded := True;
1450  CheckIBLoaded;
1482    FDatabases := TList.Create;
1483    FSQLObjects := TList.Create;
1453  FHandle := nil;
1484    FTPB := nil;
1455  FTPBLength := 0;
1485    FTRParams := TStringList.Create;
1486    FTRParamsChanged := True;
1487    TStringList(FTRParams).OnChange := TRParamsChange;
# Line 1468 | Line 1497 | destructor TIBTransaction.Destroy;
1497   var
1498    i: Integer;
1499   begin
1500 <  if FIBLoaded then
1501 <  begin
1502 <    if InTransaction then
1503 <      EndTransaction(FDefaultAction, True);
1504 <    for i := 0 to FSQLObjects.Count - 1 do
1505 <      if FSQLObjects[i] <> nil then
1506 <        SQLObjects[i].DoTransactionFree;
1507 <    RemoveSQLObjects;
1508 <    RemoveDatabases;
1509 <    FreeMem(FTPB);
1510 <    FTPB := nil;
1482 <    FTRParams.Free;
1483 <    FSQLObjects.Free;
1484 <    FDatabases.Free;
1485 <  end;
1500 >  if InTransaction then
1501 >    EndTransaction(FDefaultAction, True);
1502 >  for i := 0 to FSQLObjects.Count - 1 do
1503 >    if FSQLObjects[i] <> nil then
1504 >      SQLObjects[i].DoTransactionFree;
1505 >  RemoveSQLObjects;
1506 >  RemoveDatabases;
1507 >  FTPB := nil;
1508 >  FTRParams.Free;
1509 >  FSQLObjects.Free;
1510 >  FDatabases.Free;
1511    inherited Destroy;
1512   end;
1513  
1489 function TIBTransaction.Call(ErrCode: ISC_STATUS;
1490  RaiseError: Boolean): ISC_STATUS;
1491 var
1492  i: Integer;
1493 begin
1494  result := ErrCode;
1495  for i := 0 to FDatabases.Count - 1 do if FDatabases[i] <> nil then
1496    Databases[i].FCanTimeout := False;
1497  FCanTimeout := False;
1498  if RaiseError and (result > 0) then
1499    IBDataBaseError;
1500 end;
1501
1514   procedure TIBTransaction.CheckDatabasesInList;
1515   begin
1516    if GetDatabaseCount = 0 then
# Line 1509 | Line 1521 | procedure TIBTransaction.CheckInTransact
1521   begin
1522    if FStreamedActive and (not InTransaction) then
1523      Loaded;
1524 <  if (FHandle = nil) then
1524 >  if (TransactionIntf = nil) then
1525      IBError(ibxeNotInTransaction, [nil]);
1526   end;
1527  
# Line 1565 | Line 1577 | procedure TIBTransaction.EnsureNotInTran
1577   begin
1578    if csDesigning in ComponentState then
1579    begin
1580 <    if FHandle <> nil then
1580 >    if TransactionIntf <> nil then
1581        Rollback;
1582    end;
1583   end;
1584  
1585   procedure TIBTransaction.CheckNotInTransaction;
1586   begin
1587 <  if (FHandle <> nil) then
1587 >  if (TransactionIntf <> nil) and  TransactionIntf.InTransaction then
1588      IBError(ibxeInTransaction, [nil]);
1589   end;
1590  
# Line 1581 | Line 1593 | var
1593    i: Integer;
1594    NilFound: Boolean;
1595   begin
1596 +  EnsureNotInTransaction;
1597 +  CheckNotInTransaction;
1598 +  FTransactionIntf := nil;
1599 +
1600    i := FindDatabase(db);
1601    if i <> -1 then
1602    begin
# Line 1631 | Line 1647 | end;
1647   procedure TIBTransaction.EndTransaction(Action: TTransactionAction;
1648    Force: Boolean);
1649   var
1634  status: ISC_STATUS;
1650    i: Integer;
1651   begin
1652    CheckInTransaction;
# Line 1642 | Line 1657 | begin
1657    case Action of
1658      TARollback, TACommit:
1659      begin
1660 <      if (HandleIsShared) and
1661 <         (Action <> FDefaultAction) and
1662 <         (not Force) then
1663 <        IBError(ibxeCantEndSharedTransaction, [nil]);
1664 <      DoBeforeTransactionEnd;
1660 >      try
1661 >        DoBeforeTransactionEnd;
1662 >      except on E: EIBInterBaseError do
1663 >        begin
1664 >          if not Force then
1665 >            raise;
1666 >        end;
1667 >      end;
1668 >
1669        for i := 0 to FSQLObjects.Count - 1 do if FSQLObjects[i] <> nil then
1670 +      try
1671          SQLObjects[i].DoBeforeTransactionEnd(Action);
1672 +      except on E: EIBInterBaseError do
1673 +        begin
1674 +          if not Force then
1675 +              raise;
1676 +          end;
1677 +      end;
1678 +
1679        if InTransaction then
1680        begin
1681 <        if HandleIsShared then
1682 <        begin
1656 <          FHandle := nil;
1657 <          FHandleIsShared := False;
1658 <          status := 0;
1659 <        end
1660 <        else
1661 <          if (Action = TARollback) then
1662 <            status := Call(isc_rollback_transaction(StatusVector, @FHandle), False)
1663 <          else
1664 <            status := Call(isc_commit_transaction(StatusVector, @FHandle), False);
1665 <        if ((Force) and (status > 0)) then
1666 <          status := Call(isc_rollback_transaction(StatusVector, @FHandle), False);
1667 <        if Force then
1668 <          FHandle := nil
1681 >        if (Action = TARollback) then
1682 >            FTransactionIntf.Rollback(Force)
1683          else
1684 <          if (status > 0) then
1685 <            IBDataBaseError;
1686 <        for i := 0 to FSQLObjects.Count - 1 do if FSQLObjects[i] <> nil then
1687 <          SQLObjects[i].DoAfterTransactionEnd;
1688 <        DoAfterTransactionEnd;
1684 >        try
1685 >          FTransactionIntf.Commit;
1686 >        except on E: EIBInterBaseError do
1687 >          begin
1688 >            if Force then
1689 >              FTransactionIntf.Rollback(Force)
1690 >            else
1691 >              raise;
1692 >          end;
1693 >        end;
1694 >
1695 >          for i := 0 to FSQLObjects.Count - 1 do if FSQLObjects[i] <> nil then
1696 >          try
1697 >            SQLObjects[i].DoAfterTransactionEnd;
1698 >          except on E: EIBInterBaseError do
1699 >            begin
1700 >              if not Force then
1701 >                raise;
1702 >            end;
1703 >          end;
1704 >        try
1705 >          DoAfterTransactionEnd;
1706 >        except on E: EIBInterBaseError do
1707 >          begin
1708 >            if not Force then
1709 >              raise;
1710 >          end;
1711 >        end;
1712        end;
1713      end;
1714      TACommitRetaining:
1715 <      Call(isc_commit_retaining(StatusVector, @FHandle), True);
1715 >      FTransactionIntf.CommitRetaining;
1716 >
1717      TARollbackRetaining:
1718 <      Call(isc_rollback_retaining(StatusVector, @FHandle), True);
1718 >      FTransactionIntf.RollbackRetaining;
1719    end;
1720    if not (csDesigning in ComponentState) then
1721    begin
# Line 1729 | Line 1767 | end;
1767  
1768   function TIBTransaction.GetInTransaction: Boolean;
1769   begin
1770 <  result := (FHandle <> nil);
1770 >  result := (TransactionIntf <> nil) and TransactionIntf.InTransaction;
1771   end;
1772  
1773   function TIBTransaction.FindDatabase(db: TIBDatabase): Integer;
# Line 1785 | Line 1823 | procedure TIBTransaction.BeforeDatabaseD
1823   begin
1824    if InTransaction then
1825      EndTransaction(FDefaultAction, True);
1826 +  FTransactionIntf := nil;
1827   end;
1828  
1829   procedure TIBTransaction.RemoveDatabase(Idx: Integer);
# Line 1793 | Line 1832 | var
1832   begin
1833    if ((Idx >= 0) and (FDatabases[Idx] <> nil)) then
1834    begin
1835 +    EnsureNotInTransaction;
1836 +    CheckNotInTransaction;
1837 +    FTransactionIntf := nil;
1838 +
1839      DB := Databases[Idx];
1840      FDatabases[Idx] := nil;
1841      DB.RemoveTransaction(DB.FindTransaction(Self));
# Line 1805 | Line 1848 | procedure TIBTransaction.RemoveDatabases
1848   var
1849    i: Integer;
1850   begin
1851 +  EnsureNotInTransaction;
1852 +  CheckNotInTransaction;
1853 +  FTransactionIntf := nil;
1854 +
1855    for i := 0 to FDatabases.Count - 1 do if FDatabases[i] <> nil then
1856      RemoveDatabase(i);
1857   end;
# Line 1851 | Line 1898 | begin
1898          Rollback;
1899   end;
1900  
1854 procedure TIBTransaction.SetDefaultAction(Value: TTransactionAction);
1855 begin
1856 (*  if (Value = taRollbackRetaining) and (GetIBClientVersion < 6) then
1857    IBError(ibxeIB60feature, [nil]);*)
1858  FDefaultAction := Value;
1859 end;
1860
1901   procedure TIBTransaction.SetDefaultDatabase(Value: TIBDatabase);
1902   var
1903    i: integer;
# Line 1880 | Line 1920 | begin
1920    FDefaultDatabase := Value;
1921   end;
1922  
1883 procedure TIBTransaction.SetHandle(Value: TISC_TR_HANDLE);
1884 begin
1885  if (HandleIsShared) then
1886    EndTransaction(DefaultAction, True)
1887  else
1888    CheckNotInTransaction;
1889  FHandle := Value;
1890  FHandleIsShared := (Value <> nil);
1891 end;
1892
1923   procedure TIBTransaction.Notification( AComponent: TComponent;
1924                                          Operation: TOperation);
1925   var
# Line 1931 | Line 1961 | end;
1961  
1962   procedure TIBTransaction.StartTransaction;
1963   var
1934  pteb: PISC_TEB_ARRAY;
1935  TPB: String;
1964    i: Integer;
1965 +  Attachments: array of IAttachment;
1966 +  ValidDatabaseCount: integer;
1967   begin
1968    CheckNotInTransaction;
1969    CheckDatabasesInList;
1970 +  if TransactionIntf <> nil then
1971 +  begin
1972 +    TransactionIntf.Start(DefaultAction);
1973 +    Exit;
1974 +  end;
1975 +
1976    for i := 0 to FDatabases.Count - 1 do
1977     if  FDatabases[i] <> nil then
1978     begin
1979       with TIBDatabase(FDatabases[i]) do
1980       if not Connected then
1981 <       if FStreamedConnected then
1981 >       if StreamedConnected then
1982         begin
1983           Open;
1984 <         FStreamedConnected := False;
1984 >         StreamedConnected := False;
1985         end
1986         else
1987           IBError(ibxeDatabaseClosed, [nil]);
# Line 1953 | Line 1989 | begin
1989    if FTRParamsChanged then
1990    begin
1991      FTRParamsChanged := False;
1992 <    GenerateTPB(FTRParams, TPB, FTPBLength);
1957 <    if FTPBLength > 0 then
1958 <    begin
1959 <      IBAlloc(FTPB, 0, FTPBLength);
1960 <      Move(TPB[1], FTPB[0], FTPBLength);
1961 <    end;
1992 >    FTPB :=  GenerateTPB(FTRParams);
1993    end;
1994  
1995 <  pteb := nil;
1996 <  IBAlloc(pteb, 0, DatabaseCount * SizeOf(TISC_TEB));
1997 <  try
1998 <    for i := 0 to DatabaseCount - 1 do if Databases[i] <> nil then
1999 <    begin
2000 <      pteb^[i].db_handle := @(Databases[i].Handle);
2001 <      pteb^[i].tpb_length := FTPBLength;
2002 <      pteb^[i].tpb_address := FTPB;
2003 <    end;
2004 <    if Call(isc_start_multiple(StatusVector, @FHandle,
2005 <                               DatabaseCount, PISC_TEB(pteb)), False) > 0 then
2006 <    begin
2007 <      FHandle := nil;
2008 <      IBDataBaseError;
1978 <    end;
1979 <    if not (csDesigning in ComponentState) then
1980 <      MonitorHook.TRStart(Self);
1981 <  finally
1982 <    FreeMem(pteb);
1995 >  ValidDatabaseCount := 0;
1996 >  for i := 0 to DatabaseCount - 1 do
1997 >    if Databases[i] <> nil then Inc(ValidDatabaseCount);
1998 >
1999 >  if ValidDatabaseCount = 1 then
2000 >    FTransactionIntf := Databases[0].Attachment.StartTransaction(FTPB,DefaultAction)
2001 >  else
2002 >  begin
2003 >    SetLength(Attachments,ValidDatabaseCount);
2004 >    for i := 0 to DatabaseCount - 1 do
2005 >      if Databases[i] <> nil then
2006 >        Attachments[i] := Databases[i].Attachment;
2007 >
2008 >    FTransactionIntf := FirebirdAPI.StartTransaction(Attachments,FTPB,DefaultAction);
2009    end;
2010 +
2011 +  if not (csDesigning in ComponentState) then
2012 +      MonitorHook.TRStart(Self);
2013    DoOnStartTransaction;
2014   end;
2015  
# Line 1988 | Line 2017 | procedure TIBTransaction.TimeoutTransact
2017   begin
2018    if InTransaction then
2019    begin
2020 <    if FCanTimeout then
2020 >    if not TransactionIntf.HasActivity then
2021      begin
2022        EndTransaction(FDefaultAction, True);
2023        if Assigned(FOnIdleTimer) then
2024          FOnIdleTimer(Self);
2025      end
1997    else
1998      FCanTimeout := True;
2026    end;
2027   end;
2028  
# Line 2008 | Line 2035 | procedure TIBTransaction.TRParamsChangin
2035   begin
2036    EnsureNotInTransaction;
2037    CheckNotInTransaction;
2038 +  FTransactionIntf := nil;
2039   end;
2040  
2041   { TIBBase }
# Line 2023 | Line 2051 | begin
2051    inherited Destroy;
2052   end;
2053  
2026 function TIBBase.GetCharSetSize(CharSetID: integer): integer;
2027 begin
2028  if (CharSetID >= 0) and (CharSetID < Length(Database.FCharSetSizes)) then
2029    Result := Database.FCharSetSizes[CharSetID]
2030  else
2031    Result := 1; {Unknown character set}
2032 end;
2033
2034 function TIBBase.GetDefaultCharSetSize: integer;
2035 var DefaultCharSetName: string;
2036    i: integer;
2037 begin
2038  DefaultCharSetName := GetDefaultCharSetName;
2039  Result := 4; {worse case}
2040  for i := 0 to Length(Database.FCharSetSizes) - 1 do
2041    if Database.FCharSetNames[i] = DefaultCharSetName then
2042    begin
2043      Result := Database.FCharSetSizes[i];
2044      break;
2045    end;
2046 end;
2047
2048 function TIBBase.GetCharSetName(CharSetID: integer): string;
2049 begin
2050  if (CharSetID >= 0) and (CharSetID < Length(Database.FCharSetNames)) then
2051    Result := Database.FCharSetNames[CharSetID]
2052  else
2053    Result := ''; {Unknown character set}
2054 end;
2055
2056 function TIBBase.GetDefaultCharSetName: string;
2057 begin
2058  Result := AnsiUpperCase(Database.Params.Values['lc_ctype']);
2059 end;
2060
2054   procedure TIBBase.HandleException(Sender: TObject);
2055   begin
2056    if assigned(Database) then
# Line 2096 | Line 2089 | begin
2089    FTransaction.CheckInTransaction;
2090   end;
2091  
2099 function TIBBase.GetDBHandle: PISC_DB_HANDLE;
2100 begin
2101  CheckDatabase;
2102  result := @FDatabase.Handle;
2103 end;
2104
2105 function TIBBase.GetTRHandle: PISC_TR_HANDLE;
2106 begin
2107  CheckTransaction;
2108  result := @FTransaction.Handle;
2109 end;
2110
2092   procedure TIBBase.DoBeforeDatabaseConnect(DBParams: TStrings; var DBName: string
2093    );
2094   begin
# Line 2222 | Line 2203 | end;
2203    parameter buffer, and return it and its length
2204    in DPB and DPBLength, respectively. }
2205  
2206 < procedure GenerateDPB(sl: TStrings; var DPB: string; var DPBLength: Short);
2206 > function GenerateDPB(sl: TStrings): IDPB;
2207   var
2208 <  i, j, pval: Integer;
2208 >  i, j: Integer;
2209    DPBVal: UShort;
2210    ParamName, ParamValue: string;
2211   begin
2212 <  { The DPB is initially empty, with the exception that
2232 <    the DPB version must be the first byte of the string. }
2233 <  DPBLength := 1;
2234 <  DPB := Char(isc_dpb_version1);
2212 >  Result := FirebirdAPI.AllocateDPB;
2213  
2214    {Iterate through the textual database parameters, constructing
2215     a DPB on-the-fly }
# Line 2265 | Line 2243 | begin
2243      case DPBVal of
2244        isc_dpb_user_name, isc_dpb_password, isc_dpb_password_enc,
2245        isc_dpb_sys_user_name, isc_dpb_license, isc_dpb_encrypt_key,
2246 <      isc_dpb_lc_messages, isc_dpb_lc_ctype,
2246 >      isc_dpb_lc_messages, isc_dpb_lc_ctype, isc_dpb_page_size,
2247        isc_dpb_sql_role_name, isc_dpb_sql_dialect:
2248        begin
2249          if DPBVal = isc_dpb_sql_dialect then
2250            ParamValue[1] := Char(Ord(ParamValue[1]) - 48);
2251 <        DPB := DPB +
2274 <               Char(DPBVal) +
2275 <               Char(Length(ParamValue)) +
2276 <               ParamValue;
2277 <        Inc(DPBLength, 2 + Length(ParamValue));
2251 >        Result.Add(DPBVal).SetAsString(ParamValue);
2252        end;
2253 +
2254        isc_dpb_num_buffers, isc_dpb_dbkey_scope, isc_dpb_force_write,
2255        isc_dpb_no_reserve, isc_dpb_damaged, isc_dpb_verify:
2256 <      begin
2257 <        DPB := DPB +
2283 <               Char(DPBVal) +
2284 <               #1 +
2285 <               Char(StrToInt(ParamValue));
2286 <        Inc(DPBLength, 3);
2287 <      end;
2256 >        Result.Add(DPBVal).SetAsByte(byte(ParamValue[1]));
2257 >
2258        isc_dpb_sweep:
2259 <      begin
2260 <        DPB := DPB +
2291 <               Char(DPBVal) +
2292 <               #1 +
2293 <               Char(isc_dpb_records);
2294 <        Inc(DPBLength, 3);
2295 <      end;
2259 >        Result.Add(DPBVal).SetAsByte(isc_dpb_records);
2260 >
2261        isc_dpb_sweep_interval:
2262 <      begin
2263 <        pval := StrToInt(ParamValue);
2299 <        DPB := DPB +
2300 <               Char(DPBVal) +
2301 <               #4 +
2302 <               PChar(@pval)[0] +
2303 <               PChar(@pval)[1] +
2304 <               PChar(@pval)[2] +
2305 <               PChar(@pval)[3];
2306 <        Inc(DPBLength, 6);
2307 <      end;
2262 >        Result.Add(DPBVal).SetAsInteger(StrToInt(ParamValue));
2263 >
2264        isc_dpb_activate_shadow, isc_dpb_delete_shadow, isc_dpb_begin_log,
2265 <      isc_dpb_quit_log:
2266 <      begin
2311 <        DPB := DPB +
2312 <               Char(DPBVal) +
2313 <               #1 + #0;
2314 <        Inc(DPBLength, 3);
2315 <      end;
2265 >      isc_dpb_map_attach, isc_dpb_quit_log:
2266 >        Result.Add(DPBVal).SetAsByte(0);
2267        else
2268        begin
2269          if (DPBVal > 0) and
# Line 2330 | Line 2281 | end;
2281    of the transaction parameters, generate a transaction
2282    parameter buffer, and return it and its length in
2283    TPB and TPBLength, respectively. }
2284 < procedure GenerateTPB(sl: TStrings; var TPB: string; var TPBLength: Short);
2284 > function GenerateTPB(sl: TStrings): ITPB;
2285   var
2286 <  i, j, TPBVal, ParamLength: Integer;
2286 >  i, j, TPBVal: Integer;
2287    ParamName, ParamValue: string;
2288   begin
2289 <  TPB := '';
2339 <  if (sl.Count = 0) then
2340 <    TPBLength := 0
2341 <  else
2342 <  begin
2343 <    TPBLength := sl.Count + 1;
2344 <    TPB := TPB + Char(isc_tpb_version3);
2345 <  end;
2289 >  Result := FirebirdAPI.AllocateTPB;
2290    for i := 0 to sl.Count - 1 do
2291    begin
2292      if (Trim(sl[i]) =  '') then
2349    begin
2350      Dec(TPBLength);
2293        Continue;
2294 <    end;
2294 >
2295      if (Pos('=', sl[i]) = 0) then {mbcs ok}
2296        ParamName := LowerCase(sl[i]) {mbcs ok}
2297      else
# Line 2373 | Line 2315 | begin
2315        isc_tpb_concurrency, isc_tpb_shared, isc_tpb_wait, isc_tpb_nowait,
2316        isc_tpb_read, isc_tpb_write, isc_tpb_ignore_limbo,
2317        isc_tpb_read_committed, isc_tpb_rec_version, isc_tpb_no_rec_version:
2318 <        TPB := TPB + Char(TPBVal);
2318 >        Result.Add(TPBVal);
2319 >
2320        isc_tpb_lock_read, isc_tpb_lock_write:
2321 <      begin
2322 <        TPB := TPB + Char(TPBVal);
2380 <        { Now set the string parameter }
2381 <        ParamLength := Length(ParamValue);
2382 <        Inc(TPBLength, ParamLength + 1);
2383 <        TPB := TPB + Char(ParamLength) + ParamValue;
2384 <      end;
2321 >        Result.Add(TPBVal).SetAsString(ParamValue);
2322 >
2323        else
2324        begin
2325          if (TPBVal > 0) and

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines