tfexpand.pp 14 KB

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