Connections.pas 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511
  1. unit Connections;
  2. interface
  3. uses
  4. ADODB, SysUtils, Classes, ZHAPI, Windows;
  5. const
  6. ProductName = 'Measure';
  7. EmptyFileVersion = '1.0.0.0';
  8. FileVersion = '1.0.1.3';
  9. EncryptVersion = 'Auto1.0';
  10. SAdoConnectStr = 'Provider=Microsoft.Jet.OLEDB.4.0;Data Source=%s;' +
  11. 'User ID=Admin;Password='''';Persist Security Info=True';
  12. type
  13. TConnection = class
  14. private
  15. FFileName: string;
  16. FOrgFile: string;
  17. function GetConnection: TADOConnection; virtual; abstract;
  18. public
  19. procedure Open(const AFileName: string); virtual; abstract;
  20. procedure Save; virtual;
  21. procedure Close;
  22. function HasOpened: Boolean; virtual; abstract;
  23. property FileName: string read FFileName;
  24. property Connection: TADOConnection read GetConnection;
  25. property OrgFile: string read FOrgFile;
  26. end;
  27. TCommonConnection = class(TConnection)
  28. private
  29. FConnection: TADOConnection;
  30. procedure InnerOpen(const AFileName: string);
  31. function GetConnection: TADOConnection; override;
  32. public
  33. constructor Create;
  34. destructor Destroy; override;
  35. procedure Open(const AFileName: string); override;
  36. function HasOpened: Boolean; override;
  37. end;
  38. TDrawingFileHead = packed record
  39. ProductName: string[20];
  40. FileType: Integer;
  41. EncryptVersion: string[20];
  42. FileVersion: string[20];
  43. // 创建项目 -- Exe版本号
  44. CreateExeVersion: string[20];
  45. // 最后一次打开 -- Exe版本号
  46. LastestExeVersion: string[20];
  47. // 历史最大 -- exe版本号
  48. MaxExeVersion: string[20];
  49. Reserve: array [0..1024 -21 -21 -21] of Char;
  50. end;
  51. TEncryptConnection = class(TCommonConnection)
  52. private
  53. FFileHead: TDrawingFileHead;
  54. function CheckFileValid(const AFileName: string): Boolean;
  55. procedure Encrypt;
  56. function Decrypt(const AFileName: string): string;
  57. procedure RefreshFileHead;
  58. procedure DamageFileHead(AFileName: string);
  59. procedure RecoverFileHead(AFileName: string);
  60. procedure WriteAndAddHead(AFileName: string);
  61. procedure ReadAndRemoveHead(AFileName: string);
  62. procedure ResetFileHead;
  63. function CheckFileEncrypt: Boolean; overload;
  64. function CheckFileEncrypt(AFileName: string): Boolean; overload;
  65. function GetNeedUpdate: Boolean;
  66. function GetOverExe: Boolean;
  67. public
  68. destructor Destroy; override;
  69. procedure Open(const AFileName: string); override;
  70. procedure Save; override;
  71. procedure UpdateFileVersion;
  72. procedure SaveDebugFile(const AFileName: string);
  73. property NeedUpdate: Boolean read GetNeedUpdate;
  74. property OverExe: Boolean read GetOverExe;
  75. end;
  76. procedure SimpleDecrypt(const ASourceFile, AResultFile: string);
  77. implementation
  78. uses
  79. CompactDB, TransFile, UtilMethods;
  80. const
  81. MDBOrgHead: array [0..15] of Byte =
  82. ($00, $01, $00, $00, $53, $74, $61, $6E, $64, $61, $72, $64, $20, $4A, $65, $74);
  83. WrongHead: array [0..15] of Char =
  84. ('z', 'o', 'n', 'g', 'h', 'e', 'n', 'g', 's', 'o', 'f', 't', 'w', 'a', 'r', 'e');
  85. procedure SimpleDecrypt(const ASourceFile, AResultFile: string);
  86. procedure RemoveHead(const AFileName: string);
  87. var
  88. sTempFile: string;
  89. FileStream, FileStreamTemp: TFileStream;
  90. begin
  91. if not FileExists(AFileName) then Exit;
  92. sTempFile := GetTempFileName;
  93. try
  94. FileStream := TFileStream.Create(AFileName, fmOpenRead);
  95. FileStreamTemp := TFileStream.Create(sTempFile, fmCreate);
  96. try
  97. FileStream.Position := SizeOf(TDrawingFileHead);
  98. FileStreamTemp.CopyFrom(FileStream, FileStream.Size - SizeOf(TDrawingFileHead));
  99. finally
  100. FileStream.Free;
  101. FileStreamTemp.Free;
  102. end;
  103. CopyFile(PChar(sTempFile), PChar(AFileName), False);
  104. finally
  105. DeleteFile(PChar(sTempFile));
  106. end;
  107. end;
  108. var
  109. sTempFile: string;
  110. begin
  111. if not FileExists(ASourceFile) then Exit;
  112. try
  113. sTempFile := GetTempFileName;
  114. CopyFile(PChar(ASourceFile), PChar(sTempFile), False);
  115. RemoveHead(sTempFile);
  116. AutoDecryptFile(sTempFile, AResultFile);
  117. finally
  118. DeleteFile(PChar(sTempFile));
  119. end;
  120. end;
  121. procedure EncryptFile(const AFileName: string);
  122. var
  123. fileStream: TFileStream;
  124. begin
  125. fileStream := TFileStream.Create(AFileName, fmOpenReadWrite);
  126. try
  127. fileStream.Position := 0;
  128. fileStream.WriteBuffer(WrongHead, SizeOf(WrongHead));
  129. finally
  130. fileStream.Free;
  131. end;
  132. end;
  133. procedure DecryptFile(const AFileName: string);
  134. var
  135. fileStream: TFileStream;
  136. begin
  137. fileStream := TFileStream.Create(AFileName, fmOpenReadWrite);
  138. try
  139. fileStream.Position := 0;
  140. fileStream.WriteBuffer(MDBOrgHead, SizeOf(MDBOrgHead));
  141. finally
  142. fileStream.Free;
  143. end;
  144. end;
  145. { TEncryptConnection }
  146. procedure TEncryptConnection.Encrypt;
  147. var
  148. sTempFile, sEncryptFile: string;
  149. begin
  150. if (FFileName <> '') and FileExists(FFileName) then
  151. begin
  152. sTempFile := GetTempFileName;
  153. sEncryptFile := GetTempFileName;
  154. CopyFile(PChar(FFileName), PChar(sTempFile), False);
  155. try
  156. AutoEncryptFile(sTempFile, sEncryptFile);
  157. RefreshFileHead;
  158. WriteAndAddHead(sEncryptFile);
  159. if CheckFileValid(sEncryptFile) then
  160. CopyFile(PChar(sEncryptFile), PChar(FOrgFile), False)
  161. else
  162. Abort;
  163. finally
  164. DeleteFile(PChar(sTempFile));
  165. DeleteFile(PChar(sEncryptFile));
  166. end;
  167. end;
  168. end;
  169. destructor TEncryptConnection.Destroy;
  170. begin
  171. Close;
  172. Encrypt;
  173. if (FFileName <> '') and FileExists(FFileName) then
  174. DeleteFile(PChar(FFileName));
  175. inherited;
  176. end;
  177. procedure TEncryptConnection.Open(const AFileName: string);
  178. begin
  179. Close;
  180. if (FFileName <> '') and FileExists(FFileName) then
  181. DeleteFile(PChar(FFileName));
  182. FFileName := Decrypt(AFileName);
  183. if FileExists(FFileName) then
  184. begin
  185. RecoverFileHead(FFileName);
  186. InnerOpen(FFileName);
  187. DamageFileHead(FFileName);
  188. end
  189. else
  190. raise Exception.Create(Format('文件[%s]不存在!', [AFileName]));
  191. end;
  192. procedure TEncryptConnection.DamageFileHead(AFileName: string);
  193. var
  194. sTempFile: string;
  195. FileStream: TFileStream;
  196. begin
  197. sTempFile := ExtractFilePath(AFileName) + ExtractSimpleFileName(AFileName) + '.tmp';
  198. CopyFile(PChar(AFileName), PChar(sTempFile), False);
  199. try
  200. if FileExists(sTempFile) then
  201. begin
  202. FileStream := TFileStream.Create(sTempFile, fmOpenWrite);
  203. try
  204. FileStream.Seek($00, soFromBeginning);
  205. FileStream.Write(WrongHead, SizeOf(WrongHead));
  206. finally
  207. FileStream.Free;
  208. end;
  209. CopyFile(PChar(sTempFile), PChar(AFileName), False);
  210. DeleteFile(PChar(sTempFile));
  211. end;
  212. except
  213. DeleteFile(PChar(sTempFile));
  214. end;
  215. end;
  216. procedure TEncryptConnection.WriteAndAddHead(AFileName: string);
  217. var
  218. sTempFile: string;
  219. FileStream, FileStreamTemp: TFileStream;
  220. begin
  221. if FileExists(AFileName) then
  222. begin
  223. sTempFile := ExtractFilePath(FFileName) + ExtractSimpleFileName(FFileName) + '.tmp';
  224. FileStream := TFileStream.Create(AFileName, fmOpenRead);
  225. FileStreamTemp := TFileStream.Create(sTempFile, fmCreate);
  226. try
  227. try
  228. FileStream.Position := 0;
  229. FileStreamTemp.WriteBuffer(FFileHead, SizeOf(TDrawingFileHead));
  230. FileStreamTemp.CopyFrom(FileStream, FileStream.Size);
  231. finally
  232. FileStream.Free;
  233. FileStreamTemp.Free;
  234. end;
  235. CopyFile(PChar(sTempFile), PChar(AFileName), False);
  236. DeleteFile(PChar(sTempFile));
  237. except
  238. DeleteFile(PChar(sTempFile))
  239. end;
  240. end;
  241. end;
  242. procedure TEncryptConnection.ReadAndRemoveHead(AFileName: string);
  243. var
  244. sTempFile: string;
  245. FileStream, FileStreamTemp: TFileStream;
  246. begin
  247. if FileExists(AFileName) then
  248. begin
  249. sTempFile := ExtractFilePath(FFileName) + ExtractSimpleFileName(FFileName) + '.tmp';
  250. FileStream := TFileStream.Create(AFileName, fmOpenRead);
  251. FileStreamTemp := TFileStream.Create(sTempFile, fmCreate);
  252. try
  253. try
  254. FileStream.Position := 0;
  255. FileStream.ReadBuffer(FFileHead, SizeOf(TDrawingFileHead));
  256. FileStreamTemp.CopyFrom(FileStream, FileStream.Size - SizeOf(TDrawingFileHead));
  257. finally
  258. FileStream.Free;
  259. FileStreamTemp.Free;
  260. end;
  261. CopyFile(PChar(sTempFile), PChar(AFileName), False);
  262. DeleteFile(PChar(sTempFile));
  263. except
  264. DeleteFile(PChar(sTempFile))
  265. end;
  266. end;
  267. end;
  268. function TEncryptConnection.Decrypt(const AFileName: string): string;
  269. var
  270. sConnectFile, sTempFile: string;
  271. begin
  272. FOrgFile := AFileName;
  273. sConnectFile := GetTempFileName;
  274. sTempFile := GetTempFileName;
  275. if FileExists(AFileName) then
  276. begin
  277. CopyFile(PChar(AFileName), PChar(sTempFile), False);
  278. if CheckFileEncrypt(sTempFile) then
  279. begin
  280. ReadAndRemoveHead(sTempFile);
  281. AutoDecryptFile(sTempFile, sConnectFile);
  282. end
  283. else
  284. begin
  285. ResetFileHead;
  286. CopyFile(PChar(sTempFile), PChar(sConnectFile), False);
  287. end;
  288. DecryptFile(sConnectFile);
  289. DeleteFile(PChar(sTempFile));
  290. end;
  291. Result := sConnectFile;
  292. end;
  293. function TEncryptConnection.CheckFileEncrypt(AFileName: string): Boolean;
  294. var
  295. FileHead: TDrawingFileHead;
  296. FileStream: TFileStream;
  297. begin
  298. FileStream := TFileStream.Create(AFileName, fmOpenRead);
  299. try
  300. FileStream.ReadBuffer(FileHead, SizeOf(TDrawingFileHead));
  301. Result := FileHead.ProductName = ProductName;
  302. finally
  303. FileStream.Free;
  304. end;
  305. end;
  306. function TEncryptConnection.CheckFileEncrypt: Boolean;
  307. begin
  308. Result := FFileHead.ProductName = ProductName;
  309. end;
  310. procedure TEncryptConnection.ResetFileHead;
  311. begin
  312. if (FFileHead.ProductName <> ProductName) then
  313. begin
  314. FFileHead.ProductName := ProductName;
  315. FFileHead.FileType := 1;
  316. FFileHead.EncryptVersion := EncryptVersion;
  317. FFileHead.FileVersion := EmptyFileVersion;
  318. FFileHead.CreateExeVersion := GetVersion;
  319. FFileHead.LastestExeVersion := GetVersion;
  320. FFileHead.MaxExeVersion := GetVersion;
  321. end;
  322. end;
  323. procedure TEncryptConnection.Save;
  324. begin
  325. RecoverFileHead(FFileName);
  326. Encrypt;
  327. end;
  328. procedure TEncryptConnection.RefreshFileHead;
  329. begin
  330. FFileHead.EncryptVersion := EncryptVersion;
  331. FFileHead.FileVersion := FileVersion;
  332. FFileHead.LastestExeVersion := GetVersion;
  333. end;
  334. procedure TEncryptConnection.UpdateFileVersion;
  335. begin
  336. FFileHead.FileVersion := FileVersion;
  337. end;
  338. procedure TEncryptConnection.RecoverFileHead(AFileName: string);
  339. var
  340. sTempFile: string;
  341. FileStream: TFileStream;
  342. begin
  343. sTempFile := ExtractFilePath(AFileName) + ExtractSimpleFileName(AFileName) + '.tmp';
  344. CopyFile(PChar(AFileName), PChar(sTempFile), False);
  345. try
  346. if FileExists(sTempFile) then
  347. begin
  348. FileStream := TFileStream.Create(sTempFile, fmOpenWrite);
  349. try
  350. FileStream.Seek($00, soFromBeginning);
  351. FileStream.Write(MDBOrgHead, SizeOf(MDBOrgHead));
  352. finally
  353. FileStream.Free;
  354. end;
  355. CopyFile(PChar(sTempFile), PChar(AFileName), False);
  356. DeleteFile(PChar(sTempFile));
  357. end;
  358. except
  359. DeleteFile(PChar(sTempFile));
  360. end;
  361. end;
  362. function TEncryptConnection.GetNeedUpdate: Boolean;
  363. begin
  364. Result := CompareCode(FileVersion, FFileHead.FileVersion, '.') > 0;
  365. end;
  366. function TEncryptConnection.GetOverExe: Boolean;
  367. begin
  368. Result := CompareCode(FileVersion, FFileHead.FileVersion, '.') < 0;
  369. end;
  370. procedure TEncryptConnection.SaveDebugFile(const AFileName: string);
  371. begin
  372. CopyFile(PChar(FFileName), PChar(AFileName), False);
  373. RecoverFileHead(AFileName);
  374. end;
  375. function TEncryptConnection.CheckFileValid(
  376. const AFileName: string): Boolean;
  377. var
  378. sConnectFile, sTempFile: string;
  379. vCon: TADOConnection;
  380. begin
  381. Result := False;
  382. vCon := TADOConnection.Create(nil);
  383. vCon.LoginPrompt := False;
  384. sConnectFile := GetTempFileName;
  385. sTempFile := GetTempFileName;
  386. try
  387. if FileExists(AFileName) then
  388. begin
  389. CopyFile(PChar(AFileName), PChar(sTempFile), False);
  390. if CheckFileEncrypt(sTempFile) then
  391. begin
  392. ReadAndRemoveHead(sTempFile);
  393. AutoDecryptFile(sTempFile, sConnectFile);
  394. end
  395. else
  396. CopyFile(PChar(sTempFile), PChar(sConnectFile), False);
  397. RecoverFileHead(sConnectFile);
  398. vCon.ConnectionString := Format(SAdoConnectStr, [sConnectFile]);
  399. vCon.Open;
  400. Result := True;
  401. end;
  402. finally
  403. DeleteFile(PChar(sConnectFile));
  404. DeleteFile(PChar(sTempFile));
  405. vCon.Free;
  406. end;
  407. end;
  408. { TConnection }
  409. procedure TConnection.Close;
  410. begin
  411. Connection.Close;
  412. end;
  413. procedure TConnection.Save;
  414. begin
  415. end;
  416. { TCommonConnection }
  417. constructor TCommonConnection.Create;
  418. begin
  419. FConnection := TADOConnection.Create(nil);
  420. FConnection.LoginPrompt := False;
  421. end;
  422. destructor TCommonConnection.Destroy;
  423. begin
  424. Close;
  425. FConnection.Free;
  426. inherited;
  427. end;
  428. function TCommonConnection.GetConnection: TADOConnection;
  429. begin
  430. Result := FConnection;
  431. end;
  432. function TCommonConnection.HasOpened: Boolean;
  433. begin
  434. Result := FConnection.Connected;
  435. end;
  436. procedure TCommonConnection.Open(const AFileName: string);
  437. begin
  438. Close;
  439. if FileExists(AFileName) then
  440. begin
  441. DecryptFile(AFileName);
  442. InnerOpen(AFileName);
  443. end
  444. else
  445. raise Exception.Create(Format('文件[%s]不存在!', [AFileName]));
  446. end;
  447. procedure TCommonConnection.InnerOpen(const AFileName: string);
  448. begin
  449. FFileName := AFileName;
  450. CompactDatabase(FFileName);
  451. FConnection.ConnectionString := Format(SAdoConnectStr, [FFileName]);
  452. FConnection.Open;
  453. end;
  454. end.