tfexpand.pp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. {
  2. This file is part of the Free Pascal test suite.
  3. Copyright (c) 1999-2004 by the Free Pascal development team.
  4. Test for possible bugs in Dos.FExpand
  5. See the file COPYING.FPC, included in this distribution,
  6. for details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************}
  11. program TFExpand;
  12. {$DEFINE DEBUG}
  13. (* Defining DEBUG causes all the source and target strings *)
  14. (* to be written to the console to make debugging easier. *)
  15. { $DEFINE DIRECT}
  16. (* Defining DIRECT causes direct embedding of fexpand.inc instead *)
  17. (* of using FExpand implementation in (previously compiled) unit Dos. *)
  18. uses
  19. {$ifdef FPC}
  20. PopupErr,
  21. {$endif FPC}
  22. Dos;
  23. {$IFDEF DIRECT}
  24. (* For testing purposes on non-native platforms *)
  25. {$DEFINE VOLUMES}
  26. {$DEFINE NODOTS}
  27. { $DEFINE AMIGA}
  28. { $DEFINE UNIX}
  29. {$DEFINE MACOS}
  30. { $DEFINE FPC_FEXPAND_DRIVES}
  31. { $DEFINE FPC_FEXPAND_UNC}
  32. {$DEFINE FPC_FEXPAND_VOLUMES}
  33. {$DEFINE FPC_FEXPAND_NO_DEFAULT_PATHS}
  34. {$DEFINE FPC_FEXPAND_DRIVESEP_IS_ROOT}
  35. { $DEFINE FPC_FEXPAND_DIRSEP_IS_UPDIR}
  36. {$DEFINE FPC_FEXPAND_NO_DOTS_UPDIR}
  37. { $DEFINE FPC_FEXPAND_NO_CURDIR}
  38. { $DEFINE FPC_FEXPAND_TILDE}
  39. { $DEFINE FPC_FEXPAND_MULTIPLE_UPDIR}
  40. {$DEFINE FPC_FEXPAND_DIRSEP_IS_CURDIR}
  41. { $DEFINE FPC_FEXPAND_GETENV_PCHAR}
  42. {$ENDIF DIRECT}
  43. {$IFDEF LINUX}
  44. {$IFNDEF UNIX}
  45. {$DEFINE UNIX}
  46. {$ENDIF UNIX}
  47. {$ENDIF LINUX}
  48. {$IFDEF AMIGA}
  49. {$DEFINE VOLUMES}
  50. {$DEFINE NODRIVEC}
  51. {$ENDIF AMIGA}
  52. {$IFDEF NETWARE}
  53. {$DEFINE VOLUMES}
  54. {$DEFINE NODRIVEC}
  55. {$ENDIF NETWARE}
  56. {$IFDEF UNIX}
  57. {$DEFINE NODRIVEC}
  58. {$ENDIF UNIX}
  59. {$IFDEF MACOS}
  60. {$DEFINE VOLUMES}
  61. {$DEFINE NODRIVEC}
  62. {$DEFINE NODOTS}
  63. {$ENDIF MACOS}
  64. const
  65. {$IFDEF MACOS}
  66. CC = 'C';
  67. {$ELSE MACOS}
  68. CC = 'C:';
  69. {$ENDIF MACOS}
  70. {$IFNDEF FPC}
  71. FileNameCaseSensitive = false;
  72. DirectorySeparator = '\';
  73. DirectorySeparator2 = '\';
  74. DirSep = '\';
  75. CDrive = 'C:';
  76. DriveSep = ':';
  77. {$ELSE FPC}
  78. (* Used for ChDir/MkDir *)
  79. DirectorySeparator2 = System.DirectorySeparator;
  80. {$IFDEF DIRECT}
  81. {$IFDEF MACOS}
  82. DirectorySeparator = ':';
  83. LFNSupport = true;
  84. FileNameCaseSensitive = false;
  85. {$ELSE MACOS}
  86. {$IFDEF UNIX}
  87. DirectorySeparator = '/';
  88. FileNameCaseSensitive = true;
  89. {$ELSE UNIX}
  90. {$IFDEF AMIGA}
  91. DirectorySeparator = ':';
  92. FileNameCaseSensitive = true;
  93. {$ELSE AMIGA}
  94. DirectorySeparator = '\';
  95. FileNameCaseSensitive = false;
  96. {$ENDIF AMIGA}
  97. {$ENDIF UNIX}
  98. {$ENDIF MACOS}
  99. {$ENDIF DIRECT}
  100. DirSep = DirectorySeparator;
  101. {$IFDEF MACOS}
  102. DriveSep = '';
  103. {$ELSE MACOS}
  104. {$IFDEF AMIGA}
  105. DriveSep = '';
  106. {$ELSE AMIGA}
  107. DriveSep = DriveSeparator;
  108. {$ENDIF AMIGA}
  109. {$ENDIF MACOS}
  110. {$IFDEF UNIX}
  111. CDrive = '';
  112. {$ELSE UNIX}
  113. {$IFDEF MACOS}
  114. CDrive = 'C';
  115. {$ELSE MACOS}
  116. {$IFDEF AMIGA}
  117. CDrive = 'C';
  118. {$ELSE AMIGA}
  119. CDrive = 'C:';
  120. {$ENDIF AMIGA}
  121. {$ENDIF MACOS}
  122. {$ENDIF UNIX}
  123. {$ENDIF FPC}
  124. TestFileName = 'testfile.tst';
  125. TestDir1Name = 'TESTDIR1';
  126. TestDir2Name = 'TESTDIR2';
  127. HasErrors: boolean = false;
  128. {$IFDEF DIRECT}
  129. procedure XToDirect (var S: string);
  130. var
  131. I: byte;
  132. begin
  133. if DirectorySeparator2 <> DirectorySeparator then
  134. for I := 1 to Length (S) do
  135. if S [I] = DirectorySeparator2 then
  136. S [I] := DirectorySeparator;
  137. if DriveSeparator = DirectorySeparator then
  138. begin
  139. I := Pos (DirectorySeparator + DirectorySeparator, S);
  140. if I <> 0 then
  141. Delete (S, I, 1);
  142. end;
  143. end;
  144. procedure GetDir (Drive: byte; var Directory: string);
  145. begin
  146. System.GetDir (Drive, Directory);
  147. XToDirect (Directory);
  148. end;
  149. {$I fexpand.inc}
  150. {$ENDIF DIRECT}
  151. var
  152. TestDir, TestDir0, OrigDir, OrigTstDir, CurDir, CDir, S: DirStr;
  153. TestDrive: string [2];
  154. I: byte;
  155. IOR: longint;
  156. F: file;
  157. function Translate (S: PathStr): PathStr;
  158. var
  159. I: byte;
  160. begin
  161. {$IFDEF UNIX}
  162. if (Length (S) > 1) and (S [2] = ':') then Delete (S, 1, 2);
  163. {$ELSE UNIX}
  164. for I := 1 to Length (S) do if S [I] = '/' then S [I] := DirSep;
  165. if (Length (S) > 0) and (S [1] in ['a'..'z']) then S [1] := UpCase (S [1]);
  166. {$ENDIF UNIX}
  167. if not (FileNameCaseSensitive) then
  168. for I := 1 to Length (S) do S [I] := UpCase (S [I]);
  169. Translate := S;
  170. end;
  171. procedure Check (Src, Rslt: PathStr);
  172. var
  173. Rslt2: PathStr;
  174. begin
  175. {$IFDEF DEBUG}
  176. WriteLn (Src, '=>', Rslt);
  177. {$ENDIF DEBUG}
  178. Rslt := Translate (Rslt);
  179. Rslt2 := FExpand (Src);
  180. if Rslt <> Rslt2 then
  181. begin
  182. WriteLn ('Error: FExpand (', Src, ') should be "', Rslt, '", not "',
  183. Rslt2, '"');
  184. HasErrors := true;
  185. end;
  186. end;
  187. begin
  188. if ParamCount <> 1 then
  189. begin
  190. WriteLn ('Warning: Parameter missing!');
  191. WriteLn ('Full path to a directory with write access' +
  192. {$IFNDEF UNIX}
  193. {$IFNDEF VOLUMES}
  194. #13#10'(preferably not on a C: drive)' +
  195. {$ENDIF VOLUMES}
  196. {$ENDIF UNIX}
  197. ' expected.');
  198. WriteLn ('Trying to use the current directory instead ' +
  199. {$IFDEF UNIX}
  200. '(not quite ideal).');
  201. {$ELSE UNIX}
  202. '(problems might arise).');
  203. {$ENDIF UNIX}
  204. {$IFDEF DIRECT}System.{$ENDIF DIRECT}GetDir (0, TestDir);
  205. end else TestDir := ParamStr (1);
  206. if TestDir [Length (TestDir)] <> DirectorySeparator2 then
  207. TestDir := TestDir + DirectorySeparator2;
  208. {$IFDEF DIRECT}System.{$ENDIF DIRECT}GetDir (0, OrigDir);
  209. {$IFDEF NODRIVEC}
  210. TestDrive := '';
  211. {$ELSE NODRIVEC}
  212. TestDrive := Copy (TestDir, 1, 2);
  213. GetDir ((Ord (TestDir [1]) and not ($20)) - 64, OrigTstDir);
  214. {$ENDIF NODRIVEC}
  215. {$I-}
  216. MkDir (TestDir + TestDir1Name);
  217. if IOResult <> 0 then ;
  218. MkDir (TestDir + TestDir1Name + DirectorySeparator2 + TestDir2Name);
  219. if IOResult <> 0 then ;
  220. {$I+}
  221. ChDir (TestDir + TestDir1Name + DirectorySeparator2 + TestDir2Name);
  222. {$I-}
  223. TestDir0 := TestDir;
  224. {$IFDEF DIRECT}
  225. XToDirect (TestDir);
  226. {$ENDIF DIRECT}
  227. Assign (F, TestFileName);
  228. Rewrite (F);
  229. Close (F);
  230. if IOResult <> 0 then ;
  231. {$IFNDEF DIRECT}
  232. Assign (F, FExpand (TestFileName));
  233. {$ENDIF DIRECT}
  234. {$I+}
  235. GetDir (0, CurDir);
  236. {$IFDEF NODRIVEC}
  237. {$IFDEF UNIX}
  238. CDir := CurDir;
  239. {$ELSE UNIX}
  240. CDir := 'C:';
  241. {$ENDIF UNIX}
  242. {$ELSE NODRIVEC}
  243. GetDir (3, CDir);
  244. {$ENDIF NODRIVEC}
  245. Check (' ', CurDir + DirSep + ' ');
  246. {$IFDEF AMIGA}
  247. Check ('', CurDir);
  248. {$ELSE AMIGA}
  249. Check ('', CurDir + DirSep);
  250. {$ENDIF AMIGA}
  251. {$IFDEF MACOS}
  252. Check (':', CurDir + DirSep);
  253. {$ELSE MACOS}
  254. Check ('.', CurDir);
  255. {$ENDIF MACOS}
  256. {$IFDEF NODOTS}
  257. Check ('C:.', 'C:.');
  258. {$ELSE NODOTS}
  259. Check ('C:.', CDir);
  260. {$ENDIF NODOTS}
  261. {$IFNDEF NODRIVEC}
  262. if CDir [Length (CDir)] = DirSep then Check ('c:anything', CDir + 'anything')
  263. else Check ('c:anything', CDir + DirSep + 'anything');
  264. {$ENDIF NODRIVEC}
  265. Check (CC + DirSep, CDrive + DirSep);
  266. {$IFDEF NODOTS}
  267. Check (CC + DirSep + '.', CDrive + DirSep + '.');
  268. Check (CC + DirSep + '..', CDrive + DirSep + '..');
  269. {$ELSE NODOTS}
  270. Check (CC + DirSep + '.', CDrive + DirSep);
  271. Check (CC + DirSep + '..', CDrive + DirSep);
  272. {$ENDIF NODOTS}
  273. Check (CC + DirSep + 'DOS', CDrive + DirSep + 'DOS');
  274. {$IFNDEF NODOTS}
  275. Check (CC + DirSep + '..' + DirSep + 'DOS', CDrive + DirSep + 'DOS');
  276. {$ENDIF NODOTS}
  277. Check (CC + DirSep + 'DOS.', CDrive + DirSep + 'DOS.');
  278. {$IFDEF AMIGA}
  279. Check (CC + DirSep + 'DOS' + DirSep, CDrive + DirSep);
  280. {$ELSE AMIGA}
  281. Check (CC + DirSep + 'DOS' + DirSep, CDrive + DirSep + 'DOS' + DirSep);
  282. {$ENDIF AMIGA}
  283. {$IFNDEF NODOTS}
  284. Check (CC + DirSep + 'DOS' + DirSep + '.', CDrive + DirSep + 'DOS');
  285. Check (CC + DirSep + 'DOS' + DirSep + '..', CDrive + DirSep);
  286. Check (CC + DirSep + 'DOS' + DirSep + '..' + DirSep, CDrive + DirSep);
  287. Check (CC + DirSep + 'DOS' + DirSep + 'TEST' + DirSep + '..', CDrive +
  288. DirSep + 'DOS');
  289. Check ('C:' + DirSep + 'DOS' + DirSep + 'TEST' + DirSep + '..' + DirSep,
  290. CDrive + DirSep + 'DOS' + DirSep);
  291. {$ENDIF NODOTS}
  292. {$IFNDEF MACOS}
  293. Check (DirSep, TestDrive + DirSep);
  294. Check (DirSep + '.', TestDrive + DirSep);
  295. Check (DirSep + '..', TestDrive + DirSep);
  296. Check (DirSep + 'DOS', TestDrive + DirSep + 'DOS');
  297. {$ENDIF MACOS}
  298. Check ('d', CurDir + DirSep + 'd');
  299. {$IFDEF MACOS}
  300. Check (DirSep + 'd', CurDir + DirSep + 'd');
  301. {$ELSE MACOS}
  302. {$IFNDEF NODOTS}
  303. Check ('.' + DirSep + 'd', CurDir + DirSep + 'd');
  304. {$ENDIF NODOTS}
  305. {$ENDIF MACOS}
  306. Check (' d', CurDir + DirSep + ' d');
  307. Check ('dd', CurDir + DirSep + 'dd');
  308. {$IFDEF MACOS}
  309. Check (DirSep + 'dd' + DirSep + 'dd', CurDir + DirSep + 'dd' + DirSep + 'dd');
  310. Check ('dd' + DirSep + 'dd', 'dd' + DirSep + 'dd');
  311. {$ELSE MACOS}
  312. Check ('dd' + DirSep + 'dd', CurDir + DirSep + 'dd' + DirSep + 'dd');
  313. {$ENDIF MACOS}
  314. Check ('ddd', CurDir + DirSep + 'ddd');
  315. {$IFDEF MACOS}
  316. Check ('dddd' + DirSep + 'eeee.ffff', 'dddd' + DirSep + 'eeee.ffff');
  317. {$ELSE MACOS}
  318. Check ('dddd' + DirSep + 'eeee.ffff', CurDir + DirSep + 'dddd' + DirSep
  319. + 'eeee.ffff');
  320. {$ENDIF MACOS}
  321. Check ('.special', CurDir + DirSep + '.special');
  322. Check ('..special', CurDir + DirSep + '..special');
  323. Check ('special..', CurDir + DirSep + 'special..');
  324. {$IFDEF AMIGA}
  325. Check ('special.' + DirSep, CurDir);
  326. {$ELSE AMIGA}
  327. {$IFDEF MACOS}
  328. Check ('special.' + DirSep, 'special.' + DirSep);
  329. {$ELSE MACOS}
  330. Check ('special.' + DirSep, CurDir + DirSep + 'special.' + DirSep);
  331. {$ENDIF MACOS}
  332. {$ENDIF AMIGA}
  333. {$IFDEF MACOS}
  334. Check (DirSep + DirSep, TestDir + TestDir1Name + DirSep);
  335. Check (DirSep + DirSep + TestFileName, TestDir + TestDir1Name + DirSep
  336. + TestFileName);
  337. {$ELSE MACOS}
  338. Check (DirSep + '.special', TestDrive + DirSep + '.special');
  339. {$IFNDEF NODOTS}
  340. Check ('..', TestDir + TestDir1Name);
  341. Check ('.' + DirSep + '..', TestDir + TestDir1Name);
  342. Check ('..' + DirSep + '.', TestDir + TestDir1Name);
  343. {$ENDIF NODOTS}
  344. {$ENDIF MACOS}
  345. {$IFDEF NETWARE}
  346. Check ('...', TestDir);
  347. {$ELSE NETWARE}
  348. Check ('...', CurDir + DirSep + '...');
  349. {$ENDIF NETWARE}
  350. Check (TestFileName, CurDir + DirSep + TestFileName);
  351. {$IFDEF UNIX}
  352. S := GetEnv ('HOME');
  353. { On m68k netbsd at least, HOME contains a final slash
  354. remove it PM }
  355. if S[length(S)]=DirSep then
  356. S:=Copy(S,1,Length(S)-1);
  357. Check ('~', S);
  358. Check ('~' + DirSep + '.', S);
  359. if (Length (S) > 0) and (S [Length (S)] <> DirSep) then S := S + DirSep;
  360. Check ('~NobodyWithThisNameShouldEverExist.test/nothing', CurDir + DirSep +
  361. '~NobodyWithThisNameShouldEverExist.test/nothing');
  362. Check ('/tmp/~NoSuchUserAgain', '/tmp/~NoSuchUserAgain');
  363. Check ('~' + DirSep, S);
  364. Check ('~' + DirSep + '.' + DirSep, S);
  365. Check ('~' + DirSep + 'directory' + DirSep + 'another',
  366. S + 'directory' + DirSep + 'another');
  367. {$ELSE UNIX}
  368. {$IFNDEF NODRIVEC}
  369. Check (TestDrive + '..', TestDir + TestDir1Name);
  370. Check (TestDrive + '..' + DirSep, TestDir + TestDir1Name + DirSep);
  371. Check (TestDrive + '.' + DirSep + '.', CurDir);
  372. Check (TestDrive + '.' + DirSep + '..', TestDir + TestDir1Name);
  373. {$I-}
  374. { $ ifndef unix
  375. { avoid a and b drives for
  376. no unix systems to reduce the
  377. probablility of getting an alert message box }
  378. (* This should not be needed - unit popuperr should solve this?! TH *)
  379. I := 3;
  380. $else unix}
  381. I := 1;
  382. { $ endif unix}
  383. repeat
  384. S := '';
  385. GetDir (I, S);
  386. IOR := IOResult;
  387. if IOR = 0 then Inc (I);
  388. until (I > 26) or (IOR <> 0);
  389. if I <= 26 then
  390. begin
  391. S := Chr (I + 64) + ':ddd';
  392. Check (S, Chr (I + 64) + ':' + DirSep + 'ddd');
  393. end else
  394. WriteLn ('Sorry, cannot test FExpand behaviour for incorrect drives here.');
  395. {$I+}
  396. {$IFDEF FPC}
  397. Check ('d\d/d', CurDir + DirSep + 'd' + DirSep + 'd' + DirSep + 'd');
  398. Check ('\\server\share\directory', '\\server\share\directory');
  399. Check ('\\server\share\directory1\directory2\..',
  400. '\\server\share\directory1');
  401. Check ('\\', '\\');
  402. Check ('\\.', '\\.\');
  403. Check ('\\.\', '\\.\');
  404. Check ('\\.\.', '\\.\.');
  405. Check ('\\.\..', '\\.\..');
  406. Check ('\\.\...', '\\.\...');
  407. Check ('\\.\TEST', '\\.\TEST');
  408. Check ('\\..\', '\\..\');
  409. Check ('\\..\TEST', '\\..\TEST');
  410. Check ('\\..\TEST\.', '\\..\TEST');
  411. Check ('\\..\TEST1\TEST2\..', '\\..\TEST1');
  412. Check ('\\..\TEST\..', '\\..\TEST');
  413. Check ('\\..\TEST\..\..', '\\..\TEST');
  414. {$ENDIF FPC}
  415. {$ENDIF NODRIVEC}
  416. {$ENDIF UNIX}
  417. {$IFDEF VOLUMES}
  418. Check ('VolName' + DriveSep + DirSep + 'DIR1', 'VolName' + DriveSep + DirSep + 'DIR1');
  419. {$IFNDEF NODOTS}
  420. Check ('VolName' + DriveSep + DirSep + 'DIR1' + DirSep + '..', 'VolName' + DriveSep + DirSep);
  421. Check ('VolName' + DriveSep + DirSep + 'DIR1' + DirSep + '..' + DirSep + '..',
  422. 'VolName' + DriveSep + DirSep);
  423. Check ('VolName' + DriveSep + DirSep + '.', 'VolName:' + DirSep);
  424. Check ('VolName' + DriveSep + DirSep + '..', 'VolName:' + DirSep);
  425. Check ('VolName' + DriveSep + DirSep + '..' + DirSep, 'VolName' + DriveSep + DirSep);
  426. {$ENDIF NODOTS}
  427. {$IFDEF NETWARE}
  428. Check ('SrvName\VolName' + DriveSep + DirSep + 'TEST', 'SrvName' + DirSep + 'VolName' +
  429. DriveSep + DirSep + 'TEST');
  430. Check ('SrvName/VolName' + DriveSep + DirSep + 'TEST', 'SrvName' + DirSep + 'VolName' +
  431. DriveSep + DirSep + 'TEST');
  432. {$ENDIF NETWARE}
  433. {$IFDEF AMIGA}
  434. {$IFDEF NODOTS}
  435. Check ('.', CurDir + DirSep + '.');
  436. {$ELSE NODOTS}
  437. Check ('.', CurDir);
  438. {$ENDIF NODOTS}
  439. {$ENDIF AMIGA}
  440. {$ENDIF VOLUMES}
  441. Erase (F);
  442. {$IFNDEF NODRIVEC}
  443. ChDir (OrigTstDir);
  444. {$ENDIF NODRIVEC}
  445. ChDir (OrigDir);
  446. RmDir (TestDir0 + TestDir1Name + DirectorySeparator2 + TestDir2Name);
  447. RmDir (TestDir0 + TestDir1Name);
  448. if HasErrors then
  449. begin
  450. WriteLn ('FExpand doesn''t work correctly.');
  451. Halt (1);
  452. end;
  453. end.