diff --git a/.github/workflows/ci-docker.yml b/.github/workflows/ci-docker.yml index a01ad2d05..590cb60c6 100644 --- a/.github/workflows/ci-docker.yml +++ b/.github/workflows/ci-docker.yml @@ -32,8 +32,18 @@ jobs: with: submodules: true + - name: OS information for ${{ matrix.tag }} + uses: mosteo-actions/docker-run@v2 + with: + image: ghcr.io/alire-project/docker/gnat:${{matrix.tag}} + command: | + lsb_release -a || \ + cat /etc/os-release || \ + cat /etc/system-release || \ + echo "No lsb_release information" + - name: Run test script (${{ matrix.tag }}) - uses: mosteo-actions/docker-run@v1 + uses: mosteo-actions/docker-run@v2 with: image: ghcr.io/alire-project/docker/gnat:${{matrix.tag}} command: scripts/ci-github.sh diff --git a/scripts/ci-github.sh b/scripts/ci-github.sh index 06800270e..53d36e9dd 100755 --- a/scripts/ci-github.sh +++ b/scripts/ci-github.sh @@ -51,8 +51,8 @@ echo GNAT VERSION: gnatls -v echo ............................ -echo ALR VERSION: -alr version +echo "ALR VERSION (at $(which alr)):" +alr -d version echo ............................ # Set up index if not default: @@ -61,6 +61,10 @@ if [ "${INDEX:-}" != "" ]; then alr index --name default --add "$INDEX" fi +echo "ALR SETTINGS (global):" +alr settings --global +echo ............................ + echo ALR SEARCH: # List releases for the record alr -q -d search --list --external diff --git a/src/alire/alire-directories.adb b/src/alire/alire-directories.adb index f9551b4d4..8824cd6d8 100644 --- a/src/alire/alire-directories.adb +++ b/src/alire/alire-directories.adb @@ -13,6 +13,9 @@ with Alire.Platforms.Folders; with Alire.VFS; with Alire.Utils; +with Den.Filesystem; +with Den.Walk; + with GNAT.String_Hash; with GNATCOLL.VFS; @@ -392,51 +395,35 @@ package body Alire.Directories is Max_Depth : Natural := Natural'Last) return AAA.Strings.Vector is + use all type Den.Kinds; Found : AAA.Strings.Vector; - procedure Locate (Folder : String; - Current_Depth : Natural; - Max_Depth : Natural) + ----------- + -- Check -- + ----------- + + procedure Check (Item : Den.Walk.Item; + Enter : in out Boolean; + Stop : in out Boolean) is - use Ada.Directories; - Search : Search_Type; begin - Start_Search (Search, Folder, "", - Filter => (Ordinary_File => True, - Directory => True, - others => False)); - - while More_Entries (Search) loop - declare - Current : Directory_Entry_Type; - begin - Get_Next_Entry (Search, Current); - if Kind (Current) = Directory then - if Simple_Name (Current) /= "." - and then - Simple_Name (Current) /= ".." - and then - Current_Depth < Max_Depth - then - Locate (Folder / Simple_Name (Current), - Current_Depth + 1, - Max_Depth); - end if; - elsif Kind (Current) = Ordinary_File - and then Simple_Name (Current) = Simple_Name (Name) - then - Found.Append (Folder / Name); - end if; - end; - end loop; + Stop := False; - End_Search (Search); - end Locate; + if Max_Depth < Natural'Last and then Item.Depth > Max_Depth then + Enter := False; + end if; + + if Den.Kind (Item.Path) = File + and then Den.Name (Item.Path) = Den.Name (Name) + then + Found.Append (Item.Path); + end if; + end Check; - use Ada.Directories; begin - if Exists (Folder) and then Kind (Folder) = Directory then - Locate (Folder, 0, Max_Depth); + if Den.Exists (Folder) and then Den.Kind (Folder) = Den.Directory then + Den.Walk.Find (Folder, + Check'Access); end if; return Found; @@ -451,7 +438,9 @@ package body Alire.Directories is return Any_Path is begin - return AAA.Directories.Relative_Path (Parent, Child); + return Result : constant Any_Path := + Den.Filesystem.Relative (Den.Scrub (Parent), + Den.Scrub (Child)); end Find_Relative_Path; ---------------------- @@ -820,33 +809,32 @@ package body Alire.Directories is Remove_From_Source : Boolean) is - Base : constant Absolute_Path := Adirs.Full_Name (Src); - Target : constant Absolute_Path := Adirs.Full_Name (Dst); + Base : constant Absolute_Path := Den.Filesystem.Absolute (Src); + Target : constant Absolute_Path := Den.Filesystem.Absolute (Dst); ----------- -- Merge -- ----------- procedure Merge - (Item : Ada.Directories.Directory_Entry_Type; + (Item : Any_Path; Stop : in out Boolean) is - use all type Adirs.File_Kind; - + use all type Den.Kinds; + Src : constant Absolute_Path := Den.Filesystem.Absolute (Item); Rel_Path : constant Relative_Path := - Find_Relative_Path (Base, Adirs.Full_Name (Item)); + Find_Relative_Path (Base, Src); -- If this proves to be too slow, we should do our own recursion, -- building the relative path along the way, as this is recomputing -- it for every file needlessly. Dst : constant Absolute_Path := Target / Rel_Path; - Src : constant Absolute_Path := Adirs.Full_Name (Item); begin Stop := False; -- Check if we must skip (we delete source file) - if Adirs.Kind (Item) = Ordinary_File + if Den.Kind (Item) /= Directory and then Skip_Top_Level_Files and then Base = Parent (Src) then @@ -856,7 +844,7 @@ package body Alire.Directories is -- Create a new dir if necessary - if Adirs.Kind (Item) = Directory then + if Den.Kind (Item) = Directory then if not Is_Directory (Dst) then Trace.Debug (" Merge: Creating destination dir " & Dst); Create_Tree (Dst); @@ -870,15 +858,15 @@ package body Alire.Directories is -- Copy file into place Trace.Debug (" Merge: copying " - & Adirs.Full_Name (Item) + & Den.Filesystem.Absolute (Item) & " into " & Dst); - if Adirs.Exists (Dst) then + if Den.Exists (Dst) then if Fail_On_Existing_File then Recoverable_User_Error ("Cannot copy " & TTY.URL (Src) & " into place, file already exists: " & TTY.URL (Dst)); - elsif Adirs.Kind (Dst) /= Ordinary_File then + elsif Den.Kind (Dst) /= File then Raise_Checked_Error ("Cannot overwrite " & TTY.URL (Dst) & " as it is not a regular file"); else @@ -912,7 +900,11 @@ package body Alire.Directories is exception when E : others => Trace.Error - ("When copying " & Src & " --> " & Dst & ": "); + ("When copying " & Src & " (" & Den.Kind (Src)'Image + & ") --> " & Dst & ": "); + Trace.Error + ("Src item was: " + & Item & " (" & Den.Kind (Item)'Image & ")"); Log_Exception (E, Error); raise; end; @@ -941,112 +933,60 @@ package body Alire.Directories is procedure Traverse_Tree (Start : Any_Path; Doing : access procedure - (Item : Ada.Directories.Directory_Entry_Type; + (Item : Any_Path; Stop : in out Boolean); Recurse : Boolean := False; Spinner : Boolean := False) is use Ada.Directories; - Visited : AAA.Strings.Set; - -- To avoid infinite recursion in case of softlinks pointed to parent - -- folders - Progress : Simple_Logging.Ongoing := Simple_Logging.Activity (Text => "Exploring " & Start, Level => (if Spinner then Info else Debug)); - procedure Go_Down (Item : Directory_Entry_Type); - - ---------------------------- - -- Traverse_Tree_Internal -- - ---------------------------- - - procedure Traverse_Tree_Internal - (Start : Any_Path; - Doing : access procedure - (Item : Ada.Directories.Directory_Entry_Type; - Stop : in out Boolean); - Recurse : Boolean := False) - is - pragma Unreferenced (Doing, Recurse); - begin - Search (Start, - "", - (Directory => True, Ordinary_File => True, others => False), - Go_Down'Access); - end Traverse_Tree_Internal; - ------------- -- Go_Down -- ------------- - procedure Go_Down (Item : Directory_Entry_Type) is - Stop : Boolean := False; - Prune : Boolean := False; - VF : constant VFS.Virtual_File := - VFS.New_Virtual_File (VFS.From_FS (Full_Name (Item))); - -- We use this later to check whether this is a soft link + procedure Go_Down (This : Den.Walk.Item; + Enter : in out Boolean; + Stop : in out Boolean) + is + use all type Den.Kinds; + Path : constant Any_Path := This.Path; begin + Enter := True; + Stop := False; - -- Ada.Directories reports softlinks not as special files but as the - -- target of the link. This confuses users of Traverse_Tree that may - -- see files within a folder that has never been visited before. - - -- Short of introducing new file kinds for softlinks and reporting - -- them to clients, for now we just ignore softlinks to dirs, and - -- this way only actual folders are traversed. - - if VF.Is_Symbolic_Link and then Kind (Item) = Directory then - Trace.Warning ("Skipping softlink dir during tree traversal: " - & Full_Name (Item)); + begin + Doing (This.Path, Stop); + exception + when Traverse_Tree_Prune_Dir => + Enter := False; + end; + if Stop then return; end if; - if Simple_Name (Item) /= "." and then Simple_Name (Item) /= ".." then - begin - Doing (Item, Stop); - exception - when Traverse_Tree_Prune_Dir => - Prune := True; - end; - if Stop then - return; - end if; - - if not Prune and then Recurse and then Kind (Item) = Directory then - declare - Normal_Name : constant Absolute_Path - := - String (GNATCOLL.VFS.Full_Name - (VFS.New_Virtual_File (Full_Name (Item)), - Normalize => True, - Resolve_Links => True).all); - begin - if Visited.Contains (Normal_Name) then - Trace.Debug ("Not revisiting " & Normal_Name); - else - Visited.Insert (Normal_Name); - if Spinner then - Progress.Step ("Exploring .../" & Simple_Name (Item)); - end if; - Traverse_Tree_Internal (Normal_Name, Doing, Recurse); - end if; - end; - elsif Prune and then Kind (Item) = Directory then - Trace.Debug ("Skipping dir: " & Full_Name (Item)); - elsif Prune and then Kind (Item) /= Directory then - Trace.Warning ("Pruning of non-dir entry has no effect: " - & Full_Name (Item)); + if Enter and then Recurse and then Den.Kind (Path) = Directory then + if Spinner then + Progress.Step ("Exploring .../" & Simple_Name (Path)); end if; + elsif not Enter and then Den.Kind (Path) = Directory then + Trace.Debug ("Skipping dir: " & Full_Name (Path)); + elsif not Enter and then Den.Kind (Path) /= Directory then + Trace.Warning ("Pruning of non-dir entry has no effect: " + & Full_Name (Path)); end if; end Go_Down; begin Trace.Debug ("Traversing folder: " & Adirs.Full_Name (Start)); - Traverse_Tree_Internal (Start, Doing, Recurse); + Den.Walk.Find (Start, + Action => Go_Down'Access, + Options => (Enter_Regular_Dirs => Recurse, others => <>)); end Traverse_Tree; --------------- @@ -1062,7 +1002,7 @@ package body Alire.Directories is -- Accumulate -- ---------------- - procedure Accumulate (Item : Directory_Entry_Type; + procedure Accumulate (Item : Any_Path; Stop : in out Boolean) is begin diff --git a/src/alire/alire-directories.ads b/src/alire/alire-directories.ads index bbf513751..250b4b709 100644 --- a/src/alire/alire-directories.ads +++ b/src/alire/alire-directories.ads @@ -114,7 +114,7 @@ package Alire.Directories is procedure Traverse_Tree (Start : Any_Path; Doing : access procedure - (Item : Ada.Directories.Directory_Entry_Type; + (Item : Any_Path; Stop : in out Boolean); Recurse : Boolean := False; Spinner : Boolean := False); diff --git a/src/alire/alire-install.adb b/src/alire/alire-install.adb index 319237029..133fcda61 100644 --- a/src/alire/alire-install.adb +++ b/src/alire/alire-install.adb @@ -395,7 +395,7 @@ package body Alire.Install is Result : Installed_Milestones; procedure Find - (Item : Ada.Directories.Directory_Entry_Type; + (Item : Any_Path; Stop : in out Boolean) is Name : constant String := Adirs.Simple_Name (Item); diff --git a/src/alire/alire-os_lib.ads b/src/alire/alire-os_lib.ads index b2c88852a..5b0b2b81a 100644 --- a/src/alire/alire-os_lib.ads +++ b/src/alire/alire-os_lib.ads @@ -65,9 +65,7 @@ private function To_Portable (Path : Any_Path) return Portable_Path_Like - is (case GNATCOLL.OS.Constants.OS is - when MacOS | Unix => Path, - when Windows => Replace (Path, "\", "/")); + is (Replace (Path, "\", "/")); -------------------- -- To_Native_Like -- @@ -75,7 +73,7 @@ private function To_Native (Path : Portable_Path_Like) return Native_Path_Like is (case GNATCOLL.OS.Constants.OS is - when MacOS | Unix => Path, + when MacOS | Unix => Replace (String (Path), "\", "/"), when Windows => Replace (String (Path), "/", "\")); end Alire.OS_Lib; diff --git a/src/alire/alire-roots.adb b/src/alire/alire-roots.adb index 5b60531da..fae9448b8 100644 --- a/src/alire/alire-roots.adb +++ b/src/alire/alire-roots.adb @@ -19,6 +19,9 @@ with Alire.Toolchains.Solutions; with Alire.User_Pins.Maps; with Alire.Utils.TTY; with Alire.Utils.User_Input; +with Alire.VFS; + +with Den.Filesystem; with GNAT.OS_Lib; with GNAT.SHA256; @@ -1229,17 +1232,17 @@ package body Alire.Roots is Found : AAA.Strings.Set; -- Milestone --> Description procedure Check_Dir - (Item : Ada.Directories.Directory_Entry_Type; - Stop : in out Boolean) + (Item : Any_Path; + Stop : in out Boolean) is pragma Unreferenced (Stop); - use Ada.Directories; + use all type Den.Kinds; begin - if Kind (Item) /= Directory then + if Den.Kind (Item) /= Directory then return; end if; - if Simple_Name (Item) = Paths.Working_Folder_Inside_Root + if Den.Name (Item) = Paths.Working_Folder_Inside_Root then -- This is an alire metadata folder, don't go in. It could also be -- a crate named "alire" but that seems like a bad idea anyway. @@ -1250,14 +1253,23 @@ package body Alire.Roots is declare Opt : Optional.Root := - Optional.Detect_Root (Full_Name (Item)); + Optional.Detect_Root (Den.Filesystem.Full_Name (Item)); begin if Opt.Is_Valid then Found.Insert - (TTY.URL (Directories.Find_Relative_Path - (Starting_Path, Full_Name (Item))) & "/" - & Opt.Value.Release.Constant_Reference.Milestone.TTY_Image - & ": " & TTY.Emph + (TTY.URL + (String + (VFS.To_Portable + (Directories.Find_Relative_Path + (Den.Filesystem.Full_Name (Starting_Path), + Den.Filesystem.Full_Name (Item)))) + -- We use both full names because on Windows we see + -- mixed short/long names for the same path if we + -- apply Full_Name to only one of them. + & "/" + & Opt.Value.Release.Constant_Reference.Milestone.TTY_Image) + & ": " + & TTY.Emph (if Opt.Value.Release.Constant_Reference.Description /= "" then Opt.Value.Release.Constant_Reference.Description else "(no description)")); diff --git a/src/alire/alire-toml_index.adb b/src/alire/alire-toml_index.adb index b60e714b0..8534d77e1 100644 --- a/src/alire/alire-toml_index.adb +++ b/src/alire/alire-toml_index.adb @@ -58,7 +58,7 @@ package body Alire.TOML_Index is -- describes a supported index, and that the file tree follows the proper -- naming conventions, without extraneous files being present. - procedure Load_Manifest (Item : Ada.Directories.Directory_Entry_Type; + procedure Load_Manifest (Item : Any_Path; Stop : in out Boolean); -- Check if entry is a candidate to manifest file, and in that case load -- its contents. May raise Checked_Error. @@ -273,7 +273,8 @@ package body Alire.TOML_Index is end return; when others => Result := - Outcome_Failure ("Several index.toml files found in index"); + Outcome_Failure ("Several index.toml files found in index: " + & Repo_Version_Files.Flatten (";")); return ""; end case; end Locate_Root; @@ -357,7 +358,7 @@ package body Alire.TOML_Index is -- Load_Manifest -- ------------------- - procedure Load_Manifest (Item : Ada.Directories.Directory_Entry_Type; + procedure Load_Manifest (Item : Any_Path; Stop : in out Boolean) is pragma Unreferenced (Stop); diff --git a/src/alire/alire-toolchains.adb b/src/alire/alire-toolchains.adb index 560b0a499..2e70dba39 100644 --- a/src/alire/alire-toolchains.adb +++ b/src/alire/alire-toolchains.adb @@ -528,7 +528,7 @@ package body Alire.Toolchains is -- Detect -- ------------ - procedure Detect (Item : Ada.Directories.Directory_Entry_Type; + procedure Detect (Item : Any_Path; Stop : in out Boolean) is use Ada.Directories; diff --git a/src/alr/alr-commands-clean.adb b/src/alr/alr-commands-clean.adb index d22918736..1e58bc496 100644 --- a/src/alr/alr-commands-clean.adb +++ b/src/alr/alr-commands-clean.adb @@ -42,7 +42,7 @@ package body Alr.Commands.Clean is -- Add_Target -- ---------------- - procedure Add_Target (Item : Ada.Directories.Directory_Entry_Type; + procedure Add_Target (Item : Alire.Any_Path; Unused_Stop : in out Boolean) is use Ada.Directories; diff --git a/src/alr/alr-commands-test.adb b/src/alr/alr-commands-test.adb index cfe4c35dd..d16d31705 100644 --- a/src/alr/alr-commands-test.adb +++ b/src/alr/alr-commands-test.adb @@ -436,7 +436,7 @@ package body Alr.Commands.Test is -- Not_Empty -- --------------- - procedure Not_Empty (Item : Ada.Directories.Directory_Entry_Type; + procedure Not_Empty (Item : Alire.Any_Path; Stop : in out Boolean) is pragma Unreferenced (Item, Stop); diff --git a/src/alr/alr-files.adb b/src/alr/alr-files.adb index f728fa1c5..855580bec 100644 --- a/src/alr/alr-files.adb +++ b/src/alr/alr-files.adb @@ -1,4 +1,4 @@ -with Ada.Directories; +with Den.Filesystem; package body Alr.Files is @@ -7,19 +7,27 @@ package body Alr.Files is ------------------------- function Locate_Any_GPR_File return Natural is - use Ada.Directories; Candidates : AAA.Strings.Vector; - procedure Check (File : Directory_Entry_Type) is + ----------- + -- Check -- + ----------- + + procedure Check (File : Alire.Any_Path; Stop : in out Boolean) is + use AAA.Strings; begin - Candidates.Append (Full_Name (File)); + Stop := False; + if Den.Kind (File) in Den.File + and then Has_Suffix (To_Lower_Case (File), ".gpr") + then + Candidates.Append (Den.Filesystem.Full (File)); + end if; end Check; begin - Search (Current_Directory, - "*.gpr", - (Ordinary_File => True, others => False), - Check'Access); + Alire.Directories.Traverse_Tree + (Alire.Directories.Current, + Check'Access); return Natural (Candidates.Length); end Locate_Any_GPR_File; diff --git a/testsuite/run.py b/testsuite/run.py index e20b2df9f..90b8eccef 100755 --- a/testsuite/run.py +++ b/testsuite/run.py @@ -11,6 +11,7 @@ import os.path import shutil +import subprocess import sys from argparse import ArgumentTypeError @@ -41,7 +42,10 @@ def require_executable(self, name): if path is None: raise FileNotFoundError(f"{name} not found in PATH") else: - print(f"Using {name} at {path}") + print(f"Testsuite using {name} at {path} with version:", ) + print(subprocess.run([name, '--version'], + stdout=subprocess.PIPE).stdout.decode()) + sys.stdout.flush() def set_up(self): super().set_up() diff --git a/testsuite/tests/install/softlinks/test.py b/testsuite/tests/install/softlinks/test.py index a0f881d6f..6a06b6460 100644 --- a/testsuite/tests/install/softlinks/test.py +++ b/testsuite/tests/install/softlinks/test.py @@ -1,6 +1,8 @@ """ Test that binary files containing softlinks can be installed properly. The test -crate contains all kinds of pernicious links (broken, recursive, etc.): +crate contains all kinds of pernicious links (broken, recursive, etc.). + +This test is Unix-only, as Windows' tar cannot recreate the broken links: crate/ ├── bin -> subdir/bin @@ -30,8 +32,9 @@ import os import shutil +import subprocess from drivers.alr import run_alr, crate_dirname -from drivers.helpers import contents, on_windows +from drivers.helpers import contents def kind(file): @@ -42,11 +45,6 @@ def ls(path): return out.stdout -# Does not apply to Windows as it does not support softlinks -if on_windows(): - print('SKIP: on Windows, unapplicable') - sys.exit(0) - # This command should succeed normally run_alr("install", "--prefix=install", "crate") @@ -80,5 +78,6 @@ def ls(path): # Cleanup os.chdir("..") shutil.rmtree(cratedir) +shutil.rmtree("install") print('SUCCESS') diff --git a/testsuite/tests/install/softlinks/test.yaml b/testsuite/tests/install/softlinks/test.yaml index 1f89021f2..58bf7be7d 100644 --- a/testsuite/tests/install/softlinks/test.yaml +++ b/testsuite/tests/install/softlinks/test.yaml @@ -1,6 +1,6 @@ driver: python-script control: - - [SKIP, "skip_unix", "Test is Unix-only"] + - [SKIP, "skip_unix", "Test is Unix-only"] indexes: my_index: in_fixtures: false diff --git a/testsuite/tests/misc/dir-traversal/test.py b/testsuite/tests/misc/dir-traversal/test.py new file mode 100644 index 000000000..408943982 --- /dev/null +++ b/testsuite/tests/misc/dir-traversal/test.py @@ -0,0 +1,25 @@ +""" +Check that broken/recursive symlinks don't cause alr to fail +""" + +import os +from drivers.alr import run_alr, init_local_crate +# from drivers.asserts import assert_eq, assert_match + +init_local_crate() + +# Create a symbolic link to itself. This used to cause alr to fail. +os.symlink("self", "self") + +# Commands that traverse looking for things (crates, executables) shouldn't +# fail. + +run_alr("clean", "--temp") +run_alr("run") +run_alr("run", "--list") +run_alr("show", "--nested") + +# Remove the symlink, otherwise it breaks the testsuite driver +os.unlink("self") + +print("SUCCESS") \ No newline at end of file diff --git a/testsuite/tests/misc/dir-traversal/test.yaml b/testsuite/tests/misc/dir-traversal/test.yaml new file mode 100644 index 000000000..9a541fdd1 --- /dev/null +++ b/testsuite/tests/misc/dir-traversal/test.yaml @@ -0,0 +1,6 @@ +driver: python-script +build_mode: both +control: + - [SKIP, "skip_unix", "Test is Unix-only"] +indexes: + compiler_only_index: {} diff --git a/testsuite/tests/show/nested/test.py b/testsuite/tests/show/nested/test.py index 90de669f3..98744e3eb 100644 --- a/testsuite/tests/show/nested/test.py +++ b/testsuite/tests/show/nested/test.py @@ -17,7 +17,7 @@ # After entering the crate, it is no longer nested and shouldn't be detected os.chdir("xxx") -assert_match("\s*", +assert_match(r"\s*", run_alr("show", "--nested", quiet=False).out) # If we initialize another crate without entering it, it should again be