Skip to content

Conversation

@charles-zablit
Copy link

@charles-zablit charles-zablit commented Aug 1, 2025

This patch adds an option to install the embeddable version of Python directly from the toolchain installer on Windows.

The installer will pick up the python files from T:\Program Files\Swift\Python and install it in toolchain-install-dir/Python-3.x.x.

This PR depends on the following being merged and cherry-picked:

rdar://157773554

Version="$(NonSemVerProductVersion)"
Scope="$(PackageScope)">

<Media Id="1" Cabinet="$(VariantCabinetName)" EmbedCab="$(ArePackageCabsEmbedded)" />
Copy link
Contributor

Choose a reason for hiding this comment

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

this is not going to build without these variables defined. i would recommend moving this file to a wxi, and defining these variables in a wxs file that imports it. see https://github.com/swiftlang/swift-installer-scripts/blob/main/platforms/Windows/bld/asserts/bld.asserts.wxs

Copy link
Author

@charles-zablit charles-zablit Aug 4, 2025

Choose a reason for hiding this comment

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

Fixed, thanks! However, I think the file structure in swift-installer-scripts\platforms\Windows\python is not correct: the subdirectory should not be named asserts. Do you have a suggestion of what this should be named? Or should I just adopt a flat directory structure?

Copy link
Member

Choose a reason for hiding this comment

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

I think that cpy.wxs (as this is CPython) is fine as a name.

Copy link
Author

Choose a reason for hiding this comment

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

Right now, the file structure is:

python/
├─ asserts/
│  ├─ python.wxs
│  ├─ python.wixproject
python.wxi

I will rename the python files to cpy but what should the asserts folder be named?

Copy link
Author

Choose a reason for hiding this comment

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

@mhegazy regarding the variant name, I would really like to remove asserts for Python as it's not an asserts build, just a regular Python distribution. Would the following file structure be OK?

python/
├─ python.wxs
├─ python.wixproject
├─ python.wxi

Copy link
Author

Choose a reason for hiding this comment

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

I have tried to install the embeddable Python alongside the Asserts and NoAsserts directories, with LLDB_PYTHON_HOME=../../Python and tried other CMake defines like LLDB_PYTHON_EXE_RELATIVE_PATH.

This does not work because lldb still tries to look for python310.dll in the Path.

I see 2 possible solutions going forward:

1

Use Windows APIs to manually add specify the location of python310.dll before starting lldb. Since we optionally install Embeddable Python, this adds some hidden logic, if the user uses regular Python instead, in which case the DLL will actually be in the PATH. For this reason, I'm not in favor of this solution.

2

Put the contents of the embeddable Python folder in usr/bin. We would have something like this:

Asserts/
├─ usr/
│  ├─ bin/
│  │  ├─ lldb.exe
│  │  ├─ python.exe
│  │  ├─ python310.dll
NoAsserts/
├─ usr/
│  ├─ bin/
│  │  ├─ lldb.exe
│  │  ├─ python.exe
│  │  ├─ python310.dll

I have tested this scenario and it works fine.

Now your concern here was regarding this scenario:

Asserts/
├─ usr/
│  ├─ bin/
│  │  ├─ lldb.exe
│  │  ├─ python.exe
│  │  ├─ python310.dll
NoAsserts/
├─ usr/
│  ├─ bin/
│  │  ├─ lldb.exe

In this scenario, the Asserts variant works fine. The NoAsserts variant should work fine as well because Asserts/usr/bin will be in the PATH. If it's not, the user will have to install regular Python manually. I think this is OK because they would have had to install it either way. To I don't think that the following is true.

so if we only install to one, the other variant is broken since lldb expects it to be in the path, and it will not be.


I think we should go with the 2nd solution, because it works with minimal changes to lldb's code.

Copy link
Contributor

Choose a reason for hiding this comment

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

I do not think option 2 is a good one. Installing the component per variant is not the best path as we discussed earlier.

Here is the thing, we install usr/bin in the PATH. so if we put python.exe in usr/bin users will find it if they just run python. this is not diffrent from putting it in ../../Python and then adding this to the PATH so that lldb find it.

Copy link
Contributor

Choose a reason for hiding this comment

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

I think I have another option. lldb needs 2 things, python310.dll since it was linked against it, and cannot load without it, and the rest of the embeded python installation to execute script, and find imports etc..

For the first, we would add a new component under LLDB ComponentGroup for python310.dll - this will be in the PATH since the whole folder is, but it is not an issue since this is the dll and not the python.exe.

For the second, we would install the new python.msi to the ..\..\Python as we discussed before, and set LLDB_PYTHON_HOME=../../Python as part of the installation.

@compnerd what do you think?

Copy link
Member

Choose a reason for hiding this comment

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

That would be problematic as the python.exe (interpreter) is then going to fail to find the DLL. I think that the best option here is to do a LoadLibraryW for python3.10.dll and in the failing case use SetDllDirectory to inject the additional library search path. I just want to verify that it does not change the DLL lookup order if we do that (as in would block the previous search order from being honoured).

Copy link
Author

Choose a reason for hiding this comment

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

We will be using SetDllDirectory once the following PR is merged:

</Component>

<Component Directory="toolchain_$(VariantName)_usr_bin">
<File Source="$(PythonRoot)\python.exe" />
Copy link
Contributor

Choose a reason for hiding this comment

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

do we really need the executable? i thought that all lldb needed is the dll and the built in modules?

Copy link
Member

Choose a reason for hiding this comment

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

Yes, we should include thee executable. Ultimately, the issue with all this is that LLDB is meant to be usable as a scriptable debugger, including for post-mortem scenarios. In such a case, you want to be able to execute a script, which is going to require the interpreter.

Copy link
Author

Choose a reason for hiding this comment

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

do we really need the executable? i thought that all lldb needed is the dll and the built in modules?

On Windows, following this discussion, we want to bundle an executable Python with the toolchain installer.

Scope="$(PackageScope)">

<?define PlatformRoot = "$(ImageRoot)\Platforms\Windows.platform"?>
<?define PythonRoot = "$(ImageRoot)\python"?>
Copy link
Contributor

Choose a reason for hiding this comment

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

this should be dedined in python msi project and not here.

Copy link
Member

Choose a reason for hiding this comment

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

+1

Copy link
Author

Choose a reason for hiding this comment

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

Fixed, thanks!

<ProjectReference Include="..\bld\asserts\bld.asserts.wixproj" BindName="bld.asserts" />
<ProjectReference Include="..\cli\asserts\cli.asserts.wixproj" BindName="cli.asserts" />
<ProjectReference Include="..\dbg\asserts\dbg.asserts.wixproj" BindName="dbg.asserts" />
<ProjectReference Include="..\python\python.wixproj" BindName="python" />
Copy link
Member

Choose a reason for hiding this comment

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

Lets shuffle this down after all ide MSI. I wonder if it makes sense to rename this to match the 3-letter naming.

Copy link
Author

Choose a reason for hiding this comment

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

I moved it down below ide. Should it be at the very bottom?

Regarding the 3-letter renaming, I'm not opposed to it, however I think it would be good to differentiate between embedded Python and regular Python if we ever decide to bundle/chain the full Python msi with the toolchain installer.

If we are sure we are never going to chain the full Python msi, then we could name this py3?

Copy link
Member

Choose a reason for hiding this comment

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

I am certainly curious about why the differentiation matters.

Ultimately, the difference between the embedded and full python is something that I think that we should somewhat blur. The actual difference is the lack of an installer (not a problem), a slightly smaller standard library (is that truly a concern?), and the missing pip which we can inject. So, difference-wise it is pretty small.

They also do not author MSIs, which means that the chaining would be more complicated than if they provided a MSM/MSI.

Copy link
Author

@charles-zablit charles-zablit Oct 7, 2025

Choose a reason for hiding this comment

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

and the missing pip which we can inject. So, difference-wise it is pretty small.

Embedded Python does not support pip.

I think the scope of this patch is to fix the issues with users being confused by the missing python310.dll, where lldb crashes immediately (without an error message when executed from powershell):

There is a PR in llvm/llvm-project to address those crashes:

I might be misunderstanding, but I don't see the benefits of having a constrained version of Python with an unsupported pip installation that is not in the PATH. It seems reasonable, to me, to expect that a user who wants to do lldb Python scripting has a fully configured Python on their system.

<Checkbox Name="OptionsInstallAndroidSDKAMD64" X="210" Y="363" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsInstallAndroidPlatform">#(loc.Sdk_ProductName_Android_amd64)</Checkbox>
<Checkbox Name="OptionsInstallAndroidSDKARM" X="210" Y="381" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsInstallAndroidPlatform">#(loc.Sdk_ProductName_Android_armv7)</Checkbox>
<Checkbox Name="OptionsInstallAndroidSDKX86" X="210" Y="399" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsInstallAndroidPlatform">#(loc.Sdk_ProductName_Android_x86)</Checkbox>
<Checkbox Name="OptionsInstallEmbeddedPython" X="192" Y="165" Width="-11" Height="17" TabStop="yes" FontId="3">#(loc.EmbeddedPython_ProductName)</Checkbox>
Copy link
Member

Choose a reason for hiding this comment

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

I think that this should come at the very end as it is not really part of the toolchain, it is a third party dependency we are installing.

Copy link
Author

Choose a reason for hiding this comment

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

Fixed, thanks 👍

Scope="$(PackageScope)">

<?define PlatformRoot = "$(ImageRoot)\Platforms\Windows.platform"?>
<?define PythonRoot = "$(ImageRoot)\python"?>
Copy link
Member

Choose a reason for hiding this comment

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

+1

Name="$(VariantProductName)"
UpgradeCode="$(VariantUpgradeCode)"
Version="$(NonSemVerProductVersion)"
Scope="$(PackageScope)">
Copy link
Member

Choose a reason for hiding this comment

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

How do we version this component? Should we version it as per the Python version or the Swift toolchain version?

Copy link
Author

Choose a reason for hiding this comment

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

I think we should version it per the Python version, if we don't change the Python version between Swift versions, there is no need to reinstall Python.

</Component>

<Component Directory="toolchain_$(VariantName)_usr_bin">
<File Source="$(PythonRoot)\python.exe" />
Copy link
Member

Choose a reason for hiding this comment

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

Yes, we should include thee executable. Ultimately, the issue with all this is that LLDB is meant to be usable as a scriptable debugger, including for post-mortem scenarios. In such a case, you want to be able to execute a script, which is going to require the interpreter.

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from d806397 to 91159fc Compare August 4, 2025 14:38
@charles-zablit charles-zablit changed the title [Draft][windows] add Python 3.10.1 to the installer [windows] add Python 3.10.1 to the installer Aug 4, 2025
@charles-zablit charles-zablit self-assigned this Aug 4, 2025
<?define VariantCabinetName = python.asserts.cab?>
<?define ToolchainVersionedVariantDirectory = ToolchainVersionedAsserts ?>
<?define VariantEnvironmentComponentGUID = 30629e0c-b376-47bc-bedf-fefb7d4ca61d?>
<?if $(ProductArchitecture) = "arm64" ?>
Copy link
Contributor

Choose a reason for hiding this comment

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

this is not variant specific. i would recommend moving it to the wxi to avoid it being duplicated for diffrent variants.

Copy link
Member

Choose a reason for hiding this comment

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

Yeah, this seems like we should sink it into the wxi.

<?define VariantCabinetName = python.asserts.cab?>
<?define ToolchainVersionedVariantDirectory = ToolchainVersionedAsserts ?>
<?define VariantEnvironmentComponentGUID = 30629e0c-b376-47bc-bedf-fefb7d4ca61d?>
<?if $(ProductArchitecture) = "arm64" ?>
Copy link
Member

Choose a reason for hiding this comment

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

Yeah, this seems like we should sink it into the wxi.

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from 796d69c to a0e6271 Compare September 9, 2025 15:22
@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from a0e6271 to 66022f0 Compare October 6, 2025 15:58
@charles-zablit charles-zablit changed the title [windows] add Python 3.10.1 to the installer [windows] add an option to install Embeddable Python 3.10.1 in the toolchain installer Oct 6, 2025
<Checkbox Name="OptionsInstallAssertsToolchain" X="210" Y="435" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsIncludeNoAsserts and OptionsInstallNoAssertsToolchain">#(loc.Asserts_Toolchain_ProductName)</Checkbox>
<Checkbox Name="OptionsInstallNoAssertsToolchain" X="210" Y="453" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsIncludeNoAsserts and OptionsInstallAssertsToolchain">#(loc.NoAsserts_Toolchain_ProductName)</Checkbox>

<Checkbox Name="OptionsInstallEmbeddedPython" X="192" Y="471" Width="-11" Height="17" TabStop="yes" FontId="3" EnableCondition="OptionsInstallDBG">#(loc.EmbeddedPython_ProductName)</Checkbox>
Copy link
Contributor

Choose a reason for hiding this comment

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

can we nest this under OptionsInstallDBG?

Copy link
Member

Choose a reason for hiding this comment

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

@mhegazy, just to be clear - you mean nesting as in indented?

Copy link
Contributor

@mhegazy mhegazy Oct 8, 2025

Choose a reason for hiding this comment

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

correct. a few changes are included in that:

  • move it up so that it is immediately under OptionsInstallDBG
  • indent it so it looks like a sub feature
  • set the enableCondition to track OptionsInstallDBG (EnableCondition="OptionsInstallDBG"), so if the user did not choose OptionsInstallDBG in the first place, we do not allow them to choose to install python.

And if we do that we should also change the install condition in installer.wxs from InstallCondition="OptionsInstallEmbeddedPython = 1" to be InstallCondition="OptionsInstallEmbeddedPython = 1 and OptionsInstallDBG = 1" this way we only install it if both options are set

Copy link
Author

Choose a reason for hiding this comment

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

It's now indented and under OptionsInstallDBG. I also changed the install condition as you suggested.

@@ -0,0 +1,15 @@
<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs">
Copy link
Contributor

Choose a reason for hiding this comment

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

no need for spliting the file into python.wxs and python.wxi. just have them all in python.wxs

Copy link
Author

Choose a reason for hiding this comment

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

That's what I originally did but I ended up splitting it into 2 files on your advice here: #447 (comment)

I'm happy to revert it, I think having only one file makes more sense given the small size of the 2 files.

Copy link
Contributor

Choose a reason for hiding this comment

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

That was my bad, originally i assumed we wanted to ship them as part of the toolchain folder which is keyed on the variant. but since this is sitting in its own folder now there should not be any need to depend on the variant.
appologies for giving incorrect suggessions earlier.

Copy link
Author

Choose a reason for hiding this comment

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

No problem! Just wanted to confirm :)

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from 66022f0 to 497fa7e Compare October 7, 2025 12:36
<Component>
<File Source="$(PythonRoot)\libffi-7.dll" />
</Component>

Copy link
Member

Choose a reason for hiding this comment

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

Why the extra whitespace?

Copy link
Author

Choose a reason for hiding this comment

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

Leftover from there were whitespaces on every line. Fixed, thanks!

<ComponentGroupRef Id="EmbeddedPythonLicense" />
</Feature>
</Package>
</Include> No newline at end of file
Copy link
Member

Choose a reason for hiding this comment

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

Missing newline

Copy link
Author

Choose a reason for hiding this comment

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

Fixed 👍

Comment on lines 2 to 5
<?define VariantUpgradeCode = $(PythonUpgradeCode)?>
<?define VariantProductName = !(loc.EmbeddedPython_ProductName)?>
<?define VariantCabinetName = python.cab?>
<?define VariantEnvironmentComponentGUID = 30629e0c-b376-47bc-bedf-fefb7d4ca61d?>
Copy link
Member

Choose a reason for hiding this comment

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

I think that we can inline these values and I don't think that we need the Variant prefix as we do not have variants.

Copy link
Author

Choose a reason for hiding this comment

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

I inlined all of them apart from the $PythonRoot one.

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch 2 times, most recently from 51aad5c to 58832c9 Compare October 9, 2025 14:09
<Variable Name="OptionsInstallBLD" Value="1" />
<Variable Name="OptionsInstallCLI" bal:Overridable="yes" Persisted="yes" Value="1" />
<Variable Name="OptionsInstallDBG" bal:Overridable="yes" Persisted="yes" Value="1" />
<Variable Name="OptionsInstallEmbeddedPython" bal:Overridable="yes" Persisted="yes" Value="1" />
Copy link
Member

Choose a reason for hiding this comment

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

Hmm, I think that we want the default to be 0? We shouldn't install the additional python copy by default I think.

Copy link
Contributor

Choose a reason for hiding this comment

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

lldb would not start if python3.10 is not installed. chances that ppl have that version and not say 3.9 or 3.14 are low. i think we should optimize for the majority of users vs advanced users who will need to run custom debugging scripts. so we should keep it on by default

Copy link
Member

Choose a reason for hiding this comment

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

What do you think of OptionsInstallePy?


<MsiPackage
SourceFile="!(bindpath.python)\python.msi"
InstallCondition="OptionsInstallEmbeddedPython = 1 and OptionsInstallDBG = 1"
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
InstallCondition="OptionsInstallEmbeddedPython = 1 and OptionsInstallDBG = 1"
InstallCondition="OptionsInstallEmbeddedPython = 1 AND OptionsInstallDBG = 1"

Copy link
Author

Choose a reason for hiding this comment

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

All the other and in this file are lowercase.

Copy link
Member

Choose a reason for hiding this comment

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

There are others in the file which are uppercase. I know that I ran into some issues with the case in MSBuild. It is unfortunate that they are not consistently case-insensitive.

Manufacturer="!(loc.ManufacturerName)"
Name="!(loc.EmbeddedPython_ProductName)"
UpgradeCode="$(PythonUpgradeCode)"
Version="$(NonSemVerProductVersion)"
Copy link
Member

Choose a reason for hiding this comment

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

A question here: should we use the swift toolchain version for the package or the python version?

Copy link
Author

Choose a reason for hiding this comment

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

I think we should use the python version. There won't be any difference between toolchain version unless we bump the Python version.


<Fragment>
<DirectoryRef Id="INSTALLROOT">
<Directory Id="Python" Name="Python-$(PythonVersion)" >
Copy link
Member

Choose a reason for hiding this comment

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

Why have this in the shared bit? Why not sink this into the Python MSI? It isn't reusable

Copy link
Author

Choose a reason for hiding this comment

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

Fixed, thanks!

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from 58832c9 to ea5e9b4 Compare October 16, 2025 13:53
@charles-zablit
Copy link
Author

Note that the latest changes require the following to be merged:

@charles-zablit charles-zablit force-pushed the charles-zablit/windows/embed-python-in-the-installer branch from ea5e9b4 to 0567695 Compare October 16, 2025 14:06
<ProjectReference Include="..\cli\asserts\cli.asserts.wixproj" BindName="cli.asserts" />
<ProjectReference Include="..\dbg\asserts\dbg.asserts.wixproj" BindName="dbg.asserts" />
<ProjectReference Include="..\ide\asserts\ide.asserts.wixproj" BindName="ide.asserts" />
<ProjectReference Include="..\python\python.wixproj" BindName="python" />
Copy link
Member

Choose a reason for hiding this comment

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

I'm half tempted to say that dep\python\python.wixproj is ideal.

<Variable Name="OptionsInstallBLD" Value="1" />
<Variable Name="OptionsInstallCLI" bal:Overridable="yes" Persisted="yes" Value="1" />
<Variable Name="OptionsInstallDBG" bal:Overridable="yes" Persisted="yes" Value="1" />
<Variable Name="OptionsInstallEmbeddedPython" bal:Overridable="yes" Persisted="yes" Value="1" />
Copy link
Member

Choose a reason for hiding this comment

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

What do you think of OptionsInstallePy?


<MsiPackage
SourceFile="!(bindpath.python)\python.msi"
InstallCondition="OptionsInstallEmbeddedPython = 1 and OptionsInstallDBG = 1"
Copy link
Member

Choose a reason for hiding this comment

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

There are others in the file which are uppercase. I know that I ran into some issues with the case in MSBuild. It is unfortunate that they are not consistently case-insensitive.

<Directory Id="python_licenses" Name="licenses" />
</Directory>
</DirectoryRef>
</Fragment>
Copy link
Member

Choose a reason for hiding this comment

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

Why is this fragment here rather than in a regular tree layout like we do in the platform MSIs?

<FeatureGroupRef Id="SideBySideUpgradeStrategy" />

<ComponentGroup Id="EmbeddedPython" Directory="Python">
<Files Include="$(PythonRoot)/**" />
Copy link
Member

Choose a reason for hiding this comment

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

Please explicitly list the files. This is to ensure that we know exactly what is being shipped for licensing purposes.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants