Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Locate dotnet tool ipy launcher #1936

Merged
merged 4 commits into from
Apr 2, 2025
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
66 changes: 61 additions & 5 deletions src/core/IronPython/Hosting/PythonCommandLine.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,10 @@
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.IO.MemoryMappedFiles;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Runtime.Versioning;
using System.Text;
using System.Threading;

Expand Down Expand Up @@ -261,17 +263,18 @@ private void InitializeModules() {

var name = Path.GetFileNameWithoutExtension(executable);
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) {
var runner = Path.Combine(prefix, name + ".exe");
if (File.Exists(runner)) {
var exename = name + ".exe";
var runner = Path.Combine(prefix, exename);
if (File.Exists(runner) || FindRunner(prefix, exename, executable, out runner)) {
executable = runner;
} else {
// TODO: was for .NET Core 2.1, can we drop this?
// ipy.bat is created Install-IronPython.ps1, which installs from a zip file
runner = Path.Combine(prefix, name + ".bat");
if (File.Exists(runner)) executable = runner;
}
} else if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux) || RuntimeInformation.IsOSPlatform(OSPlatform.OSX)) {
var runner = Path.Combine(prefix, name);
if (File.Exists(runner)) {
if (File.Exists(runner) || FindRunner(prefix, name, executable, out runner)) {
executable = runner;
} else {
runner = Path.Combine(prefix, name + ".sh");
Expand All @@ -289,7 +292,7 @@ private void InitializeModules() {
if (File.Exists(path)) {
foreach (var line in File.ReadAllLines(path, Encoding.UTF8)) { // TODO: this actually needs to be decoded with surrogateescape
if (line.StartsWith('#')) continue;
var split = line.Split(new[] { '=' }, 2);
var split = line.Split(['='], 2);
if (split.Length != 2) continue;
if (split[0].Trim() == "home") {
pyvenv_prefix = split[1].Trim();
Expand All @@ -309,6 +312,59 @@ private void InitializeModules() {
}

PythonContext.SetHostVariables(prefix ?? "", executable, null);


// --- Local functions -------

static bool FindRunner(string prefix, string name, string assembly, out string runner) {
runner = null;
#if NET
while (prefix != null) {
runner = Path.Combine(prefix, name);
if (File.Exists(runner)) {
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) || IsExecutable(runner)) {
break;
}
}
prefix = Path.GetDirectoryName(prefix);
}
if (runner != null && Path.GetExtension(assembly).Equals(".dll", StringComparison.OrdinalIgnoreCase)) {
// make sure that the runner refers to this DLL
var relativeAssemblyPath = assembly.Substring(prefix.Length + 1); // skip over the path separator
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, I guess prefix here could be null if it doesn't find anything?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed, the test is wrong. Thanks!

byte[] fsAssemblyPath = Encoding.UTF8.GetBytes(relativeAssemblyPath);
byte fsap0 = fsAssemblyPath[0];

try {
using var mmf = MemoryMappedFile.CreateFromFile(runner, FileMode.Open, null, 0, MemoryMappedFileAccess.Read);
using var accessor = mmf.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read);

for (long i = accessor.Capacity - fsAssemblyPath.Length; i >= 0; i--) { // the path should be close to the end of the file
if (accessor.ReadByte(i) != fsap0) continue;

bool found = true;
for (int j = 1; j < fsAssemblyPath.Length; j++) {
if (accessor.ReadByte(i + j) != fsAssemblyPath[j]) {
found = false;
break;
}
}
if (found) return true;
}
} catch { }
Copy link
Preview

Copilot AI Mar 31, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[nitpick] Swallowing exceptions silently in the try-catch block may obscure underlying issues during runner validation. Consider logging or handling the exception explicitly to aid debugging.

Suggested change
} catch { }
} catch (Exception ex) {
Console.WriteLine($"Exception occurred while validating runner: {ex}");
}

Copilot is powered by AI, so mistakes are possible. Review output carefully before use.

}
#endif
return false;
}

#if NET
[UnsupportedOSPlatform("windows")]
static bool IsExecutable(string filePath) {
var fileInfo = new Mono.Unix.UnixFileInfo(filePath);
var fileMode = fileInfo.FileAccessPermissions;

return (fileMode & (Mono.Unix.FileAccessPermissions.UserExecute | Mono.Unix.FileAccessPermissions.GroupExecute | Mono.Unix.FileAccessPermissions.OtherExecute)) != 0;
}
#endif
}

/// <summary>
Expand Down
4 changes: 2 additions & 2 deletions src/core/IronPython/Runtime/ClrModule.cs
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@
[assembly: PythonModule("clr", typeof(IronPython.Runtime.ClrModule))]
namespace IronPython.Runtime {
/// <summary>
/// this class contains objecs and static methods used for
/// .NET/CLS interop with Python.
/// This class contains objects and static methods used for
/// .NET/CLS interop with Python.
/// </summary>
public static class ClrModule {

Expand Down
Loading